From 530f5777bfff6c5f4a29f3549cc57e76a6a8f110 Mon Sep 17 00:00:00 2001 From: Jessica Yuen <im.jessicayuen@gmail.com> Date: Tue, 12 Sep 2017 22:40:41 -0700 Subject: [PATCH] Add subcommand 'env rm' 'env rm <env-name>' deletes an environment within a ksonnet project. This is the same as removing the <env-name> environment directory and all files contained. Empty parent directories will also be removed. --- cmd/env.go | 34 ++++++++++++++ metadata/environment.go | 49 ++++++++++++++++++++ metadata/environment_test.go | 89 +++++++++++++++++++++++++++--------- metadata/interface.go | 2 +- pkg/kubecfg/env.go | 21 +++++++++ 5 files changed, 172 insertions(+), 23 deletions(-) diff --git a/cmd/env.go b/cmd/env.go index 80f32481..b128a259 100644 --- a/cmd/env.go +++ b/cmd/env.go @@ -28,6 +28,7 @@ import ( func init() { RootCmd.AddCommand(envCmd) envCmd.AddCommand(envAddCmd) + envCmd.AddCommand(envRmCmd) envCmd.AddCommand(envListCmd) // TODO: We need to make this default to checking the `kubeconfig` file. envAddCmd.PersistentFlags().String(flagAPISpec, "version:v1.7.0", @@ -107,6 +108,39 @@ Below is an example directory structure: ksonnet env add default localhost:8000`, } +var envRmCmd = &cobra.Command{ + Use: "rm <env-name>", + Short: "Delete an environment within a ksonnet project", + RunE: func(cmd *cobra.Command, args []string) error { + if len(args) != 1 { + return fmt.Errorf("'env rm' takes a single argument, that is the name of the environment") + } + + envName := args[0] + + appDir, err := os.Getwd() + if err != nil { + return err + } + appRoot := metadata.AbsPath(appDir) + + c, err := kubecfg.NewEnvRmCmd(envName, appRoot) + if err != nil { + return err + } + + return c.Run() + }, + Long: `Delete an environment within a ksonnet project. This is the same +as removing the <env-name> environment directory and all files contained. +Empty parent directories will also be removed. +`, + Example: ` # Remove the directory 'us-west/staging' and all contents + # in the 'environments' directory. This will also remove the parent directory + # 'us-west' if it is empty. + ksonnet env rm us-west/staging`, +} + var envListCmd = &cobra.Command{ Use: "list", Short: "List all environments within a ksonnet project", diff --git a/metadata/environment.go b/metadata/environment.go index 0edffef6..78ef4493 100644 --- a/metadata/environment.go +++ b/metadata/environment.go @@ -17,6 +17,7 @@ package metadata import ( "encoding/json" + "errors" "os" "path/filepath" "strings" @@ -96,6 +97,54 @@ func (m *manager) CreateEnvironment(name, uri string, spec ClusterSpec, extensio return afero.WriteFile(m.appFS, string(envSpecPath), envSpecData, os.ModePerm) } +func (m *manager) DeleteEnvironment(name string) error { + envPath := string(appendToAbsPath(m.environmentsDir, name)) + + envs, err := m.GetEnvironments() + if err != nil { + return err + } + + // Check whether this environment exists + envExists := false + for _, env := range envs { + if env.Path == envPath { + envExists = true + break + } + } + if !envExists { + return errors.New("Environment \"" + name + "\" does not exist.") + } + + // Remove the directory and all files within the environment path. + err = m.appFS.RemoveAll(envPath) + if err != nil { + return err + } + + // Need to ensure empty parent directories are also removed. + dirs := strings.Split(name, "/") + parentDir := name + for i := len(dirs) - 2; i >= 0; i-- { + parentDir = strings.TrimSuffix(parentDir, "/"+dirs[i+1]) + parentPath := string(appendToAbsPath(m.environmentsDir, parentDir)) + + isEmpty, err := afero.IsEmpty(m.appFS, parentPath) + if err != nil { + return err + } + if isEmpty { + err := m.appFS.RemoveAll(parentPath) + if err != nil { + return err + } + } + } + + return nil +} + func (m *manager) GetEnvironments() ([]Environment, error) { envs := []Environment{} diff --git a/metadata/environment_test.go b/metadata/environment_test.go index 67751f9e..532c2c8e 100644 --- a/metadata/environment_test.go +++ b/metadata/environment_test.go @@ -18,6 +18,7 @@ package metadata import ( "fmt" "os" + "strings" "testing" "github.com/spf13/afero" @@ -26,58 +27,102 @@ import ( const ( mockSpecJSON = "spec.json" mockSpecJSONURI = "localhost:8080" + mockEnvName = "us-west/test" + mockEnvName2 = "us-west/prod" + mockEnvName3 = "us-east/test" ) -func TestGetEnvironments(t *testing.T) { +func mockEnvironments(t *testing.T, appName string) *manager { spec, err := parseClusterSpec(fmt.Sprintf("file:%s", blankSwagger), testFS) if err != nil { t.Fatalf("Failed to parse cluster spec: %v", err) } - appPath := AbsPath("/test-app") + appPath := AbsPath(appName) m, err := initManager(appPath, spec, testFS) if err != nil { t.Fatalf("Failed to init cluster spec: %v", err) } - defaultEnvDir := appendToAbsPath(environmentsDir, defaultEnvName) - anotherEnvDir := appendToAbsPath(environmentsDir, mockEnvName) + envDir := appendToAbsPath(appPath, environmentsDir) + testDirExists(t, string(envDir)) - path := appendToAbsPath(appPath, environmentsDir) - exists, err := afero.DirExists(testFS, string(path)) - if err != nil { - t.Fatalf("Expected to create directory '%s', but failed:\n%v", environmentsDir, err) - } else if !exists { - t.Fatalf("Expected to create directory '%s', but failed", path) + envNames := []string{defaultEnvName, mockEnvName, mockEnvName2, mockEnvName3} + for _, env := range envNames { + envPath := appendToAbsPath(envDir, env) + + specPath := appendToAbsPath(envPath, mockSpecJSON) + specData, err := generateSpecData(mockSpecJSONURI) + if err != nil { + t.Fatalf("Expected to marshal:\n%s\n, but failed", mockSpecJSONURI) + } + err = afero.WriteFile(testFS, string(specPath), specData, os.ModePerm) + if err != nil { + t.Fatalf("Could not write file at path: %s", specPath) + } + + testDirExists(t, string(envPath)) } - defaultEnvPath := appendToAbsPath(appPath, string(defaultEnvDir)) - anotherEnvPath := appendToAbsPath(appPath, string(anotherEnvDir)) - specDefaultEnvPath := appendToAbsPath(defaultEnvPath, mockSpecJSON) - specAnotherEnvPath := appendToAbsPath(anotherEnvPath, mockSpecJSON) + return m +} - specData, err := generateSpecData(mockSpecJSONURI) +func testDirExists(t *testing.T, path string) { + exists, err := afero.DirExists(testFS, path) if err != nil { - t.Fatalf("Expected to marshal:\n%s\n, but failed", mockSpecJSONURI) + t.Fatalf("Expected directory at '%s' to exist, but failed:\n%v", path, err) + } else if !exists { + t.Fatalf("Expected directory at '%s' to exist, but it does not", path) + } +} + +func testDirNotExists(t *testing.T, path string) { + exists, err := afero.DirExists(testFS, path) + if err != nil { + t.Fatalf("Expected directory at '%s' to be removed, but failed:\n%v", path, err) + } else if exists { + t.Fatalf("Expected directory at '%s' to be removed, but it exists", path) } +} + +func TestDeleteEnvironment(t *testing.T) { + appName := "test-delete-envs" + m := mockEnvironments(t, appName) - err = afero.WriteFile(testFS, string(specDefaultEnvPath), specData, os.ModePerm) + // Test that both directory and empty parent directory is deleted. + expectedPath := appendToAbsPath(m.environmentsDir, mockEnvName3) + parentDir := strings.Split(mockEnvName3, "/")[0] + expectedParentPath := appendToAbsPath(m.environmentsDir, parentDir) + err := m.DeleteEnvironment(mockEnvName3) if err != nil { - t.Fatalf("Could not write file at path: %s", specDefaultEnvPath) + t.Fatalf("Expected %s to be deleted but got err:\n %s", mockEnvName3, err) } - err = afero.WriteFile(testFS, string(specAnotherEnvPath), specData, os.ModePerm) + testDirNotExists(t, string(expectedPath)) + testDirNotExists(t, string(expectedParentPath)) + + // Test that only leaf directory is deleted if parent directory is shared + expectedPath = appendToAbsPath(m.environmentsDir, mockEnvName2) + parentDir = strings.Split(mockEnvName2, "/")[0] + expectedParentPath = appendToAbsPath(m.environmentsDir, parentDir) + err = m.DeleteEnvironment(mockEnvName2) if err != nil { - t.Fatalf("Could not write file at path: %s", specAnotherEnvPath) + t.Fatalf("Expected %s to be deleted but got err:\n %s", mockEnvName3, err) } + testDirNotExists(t, string(expectedPath)) + testDirExists(t, string(expectedParentPath)) +} + +func TestGetEnvironments(t *testing.T) { + m := mockEnvironments(t, "test-get-envs") envs, err := m.GetEnvironments() if err != nil { t.Fatalf("Expected to successfully get environments but failed:\n %s", err) } - if len(envs) != 2 { - t.Fatalf("Expected to get %d environments, got %d", 2, len(envs)) + if len(envs) != 4 { + t.Fatalf("Expected to get %d environments, got %d", 4, len(envs)) } if envs[0].URI != mockSpecJSONURI { diff --git a/metadata/interface.go b/metadata/interface.go index 25a5d7dc..d450bd03 100644 --- a/metadata/interface.go +++ b/metadata/interface.go @@ -38,6 +38,7 @@ type Manager interface { LibPaths(envName string) (libPath, envLibPath AbsPath) GenerateKsonnetLibData(spec ClusterSpec) ([]byte, []byte, error) CreateEnvironment(name, uri string, spec ClusterSpec, extensionsLibData, k8sLibData []byte) error + DeleteEnvironment(name string) error GetEnvironments() ([]Environment, error) // // TODO: Fill in methods as we need them. @@ -45,7 +46,6 @@ type Manager interface { // GetPrototype(id string) Protoype // SearchPrototypes(query string) []Protoype // VendorLibrary(uri, version string) error - // DeleteEnv(name string) error // } diff --git a/pkg/kubecfg/env.go b/pkg/kubecfg/env.go index 4524d702..47b4ac2f 100644 --- a/pkg/kubecfg/env.go +++ b/pkg/kubecfg/env.go @@ -55,6 +55,27 @@ func (c *EnvAddCmd) Run() error { // ================================================================== +type EnvRmCmd struct { + name string + + rootPath metadata.AbsPath +} + +func NewEnvRmCmd(name string, rootPath metadata.AbsPath) (*EnvRmCmd, error) { + return &EnvRmCmd{name: name, rootPath: rootPath}, nil +} + +func (c *EnvRmCmd) Run() error { + manager, err := metadata.Find(c.rootPath) + if err != nil { + return err + } + + return manager.DeleteEnvironment(c.name) +} + +// ================================================================== + type EnvListCmd struct { rootPath metadata.AbsPath } -- GitLab