Skip to content

Commit

Permalink
Added comprehensive test for determining the current project.
Browse files Browse the repository at this point in the history
  • Loading branch information
mitchell-as committed Dec 12, 2023
1 parent 25b620b commit e6183ee
Show file tree
Hide file tree
Showing 2 changed files with 90 additions and 57 deletions.
14 changes: 5 additions & 9 deletions pkg/projectfile/projectfile.go
Original file line number Diff line number Diff line change
Expand Up @@ -758,6 +758,10 @@ func (p *Project) SetBranch(branch string) error {
}

// GetProjectFilePath returns the path to the project activestate.yaml
// It considers projects in the following order:
// 1. Environment variable (e.g. `state shell` sets one)
// 2. Working directory (i.e. walk up directory tree looking for activestate.yaml)
// 3. Fall back on default project
func GetProjectFilePath() (string, error) {
defer profile.Measure("GetProjectFilePath", time.Now())
lookup := []func() (string, error){
Expand All @@ -779,21 +783,13 @@ func GetProjectFilePath() (string, error) {
}

func getProjectFilePathFromEnv() (string, error) {
var projectFilePath string

if activatedProjectDirPath := os.Getenv(constants.ActivatedStateEnvVarName); activatedProjectDirPath != "" {
projectFilePath = filepath.Join(activatedProjectDirPath, constants.ConfigFileName)
} else {
projectFilePath = os.Getenv(constants.ProjectEnvVarName)
}

projectFilePath := os.Getenv(constants.ProjectEnvVarName)
if projectFilePath != "" {
if fileutils.FileExists(projectFilePath) {
return projectFilePath, nil
}
return "", &ErrorNoProjectFromEnv{locale.NewInputError("err_project_env_file_not_exist", "", projectFilePath)}
}

return "", nil
}

Expand Down
133 changes: 85 additions & 48 deletions pkg/projectfile/projectfile_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@ import (
"github.com/ActiveState/cli/internal/constants"
"github.com/ActiveState/cli/internal/environment"
"github.com/ActiveState/cli/internal/errs"
"github.com/ActiveState/cli/internal/fileutils"
"github.com/ActiveState/cli/internal/language"
"github.com/ActiveState/cli/internal/locale"
"github.com/ActiveState/cli/internal/osutils"
Expand Down Expand Up @@ -207,59 +208,95 @@ func TestSave(t *testing.T) {
os.Remove(tmpfile.Name())
}

// Call getProjectFilePath
func TestGetProjectFilePath(t *testing.T) {
Reset()

root, err := environment.GetRootPath()
assert.NoError(t, err, "Should detect root path")
cwd, err := osutils.Getwd()
assert.NoError(t, err, "Should fetch cwd")
defer os.Chdir(cwd) // restore
os.Chdir(filepath.Join(root, "pkg", "projectfile", "testdata"))

configPath, err := GetProjectFilePath()
require.Nil(t, err)
expectedPath := filepath.Join(root, "pkg", "projectfile", "testdata", constants.ConfigFileName)
assert.Equal(t, expectedPath, configPath, "Project path is properly detected")
os.Chdir(cwd) // restore

defer os.Unsetenv(constants.ProjectEnvVarName)

os.Setenv(constants.ProjectEnvVarName, "/some/path")
configPath, err = GetProjectFilePath()
errt := &ErrorNoProjectFromEnv{}
require.ErrorAs(t, err, &errt)

expectedPath = filepath.Join(root, "pkg", "projectfile", "testdata", constants.ConfigFileName)
os.Setenv(constants.ProjectEnvVarName, expectedPath)
configPath, err = GetProjectFilePath()
require.Nil(t, err)
assert.Equal(t, expectedPath, configPath, "Project path is properly detected using the ProjectEnvVarName")

os.Unsetenv(constants.ProjectEnvVarName)
currentDir, err := os.Getwd()
require.NoError(t, err)
defer os.Chdir(currentDir)

rootDir, err := fileutils.ResolvePath(fileutils.TempDirUnsafe())
assert.NoError(t, err)
defer os.RemoveAll(rootDir)

// First, set up a new project with a subproject.
projectDir := filepath.Join(rootDir, "project")
require.NoError(t, fileutils.Mkdir(projectDir))
projectYaml := filepath.Join(projectDir, constants.ConfigFileName)
require.NoError(t, fileutils.Touch(projectYaml))
subprojectDir := filepath.Join(projectDir, "subproject")
require.NoError(t, fileutils.Mkdir(subprojectDir))
subprojectYaml := filepath.Join(subprojectDir, constants.ConfigFileName)
require.NoError(t, fileutils.Mkdir(subprojectDir))
require.NoError(t, fileutils.Touch(subprojectYaml))

// Then set up a separate, default project.
defaultDir := filepath.Join(rootDir, "default")
require.NoError(t, fileutils.Mkdir(defaultDir))
defaultYaml := filepath.Join(defaultDir, constants.ConfigFileName)
require.NoError(t, fileutils.Touch(defaultYaml))
cfg, err := config.New()
require.NoError(t, err)
defer func() { require.NoError(t, cfg.Close()) }()
cfg.Set(constants.GlobalDefaultPrefname, "") // ensure it is unset
tmpDir, err := ioutil.TempDir("", "")
assert.NoError(t, err, "Should create temp dir")
defer os.RemoveAll(tmpDir)
os.Chdir(tmpDir)
_, err = GetProjectFilePath()
assert.Error(t, err, "GetProjectFilePath should fail")
cfg.Set(constants.GlobalDefaultPrefname, expectedPath)
configPath, err = GetProjectFilePath()
assert.NoError(t, err, "GetProjectFilePath should succeed")
assert.Equal(t, expectedPath, configPath, "Project path is properly detected using default path from config")

// The activestate.yaml for an activated project should be used no matter what.
defer os.Unsetenv(constants.ActivatedStateEnvVarName)
os.Setenv(constants.ActivatedStateEnvVarName, filepath.Dir(expectedPath))
configPath, err = GetProjectFilePath()
require.Nil(t, err)
assert.Equal(t, expectedPath, configPath, "Project path is properly detected using the ActivatedStateEnvVarName")
os.Unsetenv(constants.ActivatedStateEnvVarName)
cfg.Set(constants.GlobalDefaultPrefname, defaultDir)

// Now set up an empty directory.
emptyDir := filepath.Join(rootDir, "empty")
require.NoError(t, fileutils.Mkdir(emptyDir))

// Now change to the project directory and assert GetProjectFilePath() returns it over the
// default project.
require.NoError(t, os.Chdir(projectDir))
path, err := GetProjectFilePath()
assert.NoError(t, err)
assert.Equal(t, projectYaml, path)

// `state shell` sets an environment variable, so run `state shell` in this project and then
// change to the subproject directory. Assert GetProjectFilePath() still returns the parent
// project.
require.NoError(t, os.Setenv(constants.ProjectEnvVarName, projectYaml))
defer os.Unsetenv(constants.ProfileEnvVarName)
require.NoError(t, os.Chdir(subprojectDir))
path, err = GetProjectFilePath()
assert.NoError(t, err)
assert.Equal(t, projectYaml, path)

// If the project were to not exist, GetProjectFilePath() should return a typed error.
require.NoError(t, os.Setenv(constants.ProjectEnvVarName, filepath.Join(rootDir, "does-not-exist", constants.ConfigFileName)))
path, err = GetProjectFilePath()
errNoProjectFromEnv := &ErrorNoProjectFromEnv{}
assert.ErrorAs(t, err, &errNoProjectFromEnv)

// After exiting out of the shell, the environment variable is no longer set. Assert
// GetProjectFilePath() returns the subproject.
require.NoError(t, os.Unsetenv(constants.ProjectEnvVarName))
path, err = GetProjectFilePath()
assert.NoError(t, err)
assert.Equal(t, subprojectYaml, path)

// If a project's subdirectory does not contain an activestate.yaml file, GetProjectFilePath()
// should walk up the tree until it finds one.
require.NoError(t, os.Remove(subprojectYaml))
path, err = GetProjectFilePath()
assert.NoError(t, err)
assert.Equal(t, projectYaml, path)

// Change to an empty directory and assert GetProjectFilePath() returns the default project.
require.NoError(t, os.Chdir(emptyDir))
path, err = GetProjectFilePath()
assert.NoError(t, err)
assert.Equal(t, defaultYaml, path)

// If the default project no longer exists, GetProjectFilePath() should return a typed error.
cfg.Set(constants.GlobalDefaultPrefname, filepath.Join(rootDir, "does-not-exist"))
path, err = GetProjectFilePath()
errNoDefaultProject := &ErrorNoDefaultProject{}
assert.ErrorAs(t, err, &errNoDefaultProject)

// If none of the above, GetProjectFilePath() should return a typed error.
cfg.Set(constants.GlobalDefaultPrefname, "")
path, err = GetProjectFilePath()
errNoProject := &ErrorNoProject{}
assert.ErrorAs(t, err, &errNoProject)
}

// TestGet the config
Expand Down

0 comments on commit e6183ee

Please sign in to comment.