Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

add terraform clean --everything and terraform clean --force to delete terraform.tfstate.d folder #727

Merged
merged 86 commits into from
Dec 5, 2024
Merged
Show file tree
Hide file tree
Changes from 7 commits
Commits
Show all changes
86 commits
Select commit Hold shift + click to select a range
10f8570
add terraform clean delete terraform.tfstate.d folder if the `--ever…
haitham911 Oct 15, 2024
594be52
add comment
haitham911 Oct 15, 2024
aaf17de
Merge branch 'main' into DEV-346
aknysh Oct 16, 2024
61136f4
Refactor processArgsAndFlags function to handle additional options wi…
haitham911 Oct 17, 2024
447105b
Refactor path_utils.go and add new functions for finding folders with…
haitham911 Oct 17, 2024
014dd09
add --everything provide additional options for the 'atmos terraform …
haitham911 Oct 17, 2024
ff42d22
Refactor path_utils.go to improve folder search and deletion function…
haitham911 Oct 17, 2024
23b320a
add comment to function
haitham911 Oct 17, 2024
36e946d
Merge branch 'main' into DEV-346
haitham911 Oct 17, 2024
e290655
Refactor processArgsAndFlags function to handle options and component…
haitham911 Oct 17, 2024
5713f90
Refactor help.go to improve 'atmos terraform clean' command documenta…
haitham911 Oct 17, 2024
5749a47
use filepath ensure cross-platform compatibility
haitham911 Oct 17, 2024
5985dcb
findFoldersNamesWithPrefix function to clarify search levels
haitham911 Oct 17, 2024
10a097c
Refactor ExecuteTerraform function to use descriptive variable names …
haitham911 Oct 17, 2024
9dda130
Refactor ExecuteTerraform function to use descriptive variable names …
haitham911 Oct 17, 2024
fd16997
Refactor processArgsAndFlags function to handle additional arguments …
haitham911 Oct 17, 2024
34e7b6f
Merge branch 'main' into DEV-346
osterman Oct 20, 2024
bcdbbc5
Refactor help.go to clarify Terraform state file deletion and add for…
haitham911 Oct 23, 2024
2fd446b
Refactor deleteFilesAndFoldersRecursive function to improve error han…
haitham911 Oct 23, 2024
c08a46a
rename file bubble_msg.go
haitham911 Oct 23, 2024
0610b30
use log package for deletion logging
haitham911 Oct 23, 2024
57953db
Refactor deleteFilesAndFoldersRecursive function to improve error han…
haitham911 Oct 24, 2024
33a6a07
Refactor bubble_msg.go to improve confirmation dialog state handling
haitham911 Oct 24, 2024
2843529
Refactor Confirm function to handle confirmation dialog state and mod…
haitham911 Oct 24, 2024
47ca61c
Refactor bubble_msg.go to improve confirmation dialog state handling …
haitham911 Oct 24, 2024
b14b1e9
command with --everything or --force flags
haitham911 Oct 24, 2024
4944db7
remove comment
haitham911 Oct 24, 2024
760c725
Refactor error handling in ExecuteTerraform function
haitham911 Oct 24, 2024
5a67dee
Refactor error handling in findFoldersNamesWithPrefix function
haitham911 Oct 24, 2024
f56b15d
Refactor help.go to improve 'atmos terraform clean' command documenta…
haitham911 Oct 24, 2024
beee637
Refactor error handling in findFoldersNamesWithPrefix and ExecuteTerr…
haitham911 Oct 24, 2024
76943f1
Refactor confirm delete msg
haitham911 Oct 24, 2024
9a3eca0
log waring with no confirm msg
haitham911 Oct 24, 2024
a7b57c2
Refactor error handling in findFoldersNamesWithPrefix function
haitham911 Oct 24, 2024
893b725
Refactor clean command to improve Terraform state file deletion
haitham911 Oct 24, 2024
0fc70e1
Refactor clean command to improve Terraform state file deletion
haitham911 Oct 24, 2024
3d57f25
confirm msg color
haitham911 Oct 25, 2024
2045cce
add log clean all components
haitham911 Oct 25, 2024
4ff0fa9
check file exist before delete
haitham911 Oct 25, 2024
3d0fd44
use filepath pkg
haitham911 Oct 25, 2024
b5042eb
modify log
haitham911 Oct 25, 2024
2c30fd3
use DeletePathTerraform utility
haitham911 Oct 25, 2024
5378abd
Refactor clean subcommand to handle terraform component cleanup
haitham911 Oct 27, 2024
c24eac9
Refactor clean subcommand to use filepath package for path manipulation
haitham911 Oct 27, 2024
1c8a4b9
Refactor clean empty dir
haitham911 Oct 27, 2024
483c19e
remove print line for debug
haitham911 Oct 27, 2024
fa94d4f
Refactor clean subcommand to handle relative path correctly
haitham911 Oct 27, 2024
32d01f6
Refactor clean subcommand to handle relative path correctly
haitham911 Oct 27, 2024
21e92d8
Refactor help message for 'atmos terraform clean' command
haitham911 Oct 28, 2024
130560b
Refactor clean subcommand
haitham911 Oct 28, 2024
0350c8a
Refactor clean subcommand to use u.PrintMessage instead of u.LogInfo …
haitham911 Oct 28, 2024
5b54e15
Refactor clean TF_DATA_DIR with everything
haitham911 Oct 28, 2024
a87b585
Refactor handleTFDataDir to handle relative path correctly
haitham911 Oct 28, 2024
bb688a3
Merge branch 'main' into DEV-346
aknysh Oct 29, 2024
be055a8
Refactor error messages for invalid TF_DATA_DIR and missing stack
haitham911 Oct 30, 2024
3e65888
Update internal/exec/terraform_clean.go
osterman Oct 30, 2024
3274fbc
Refactor handleCleanSubCommand to improve deletion messaging and stre…
haitham911 Oct 31, 2024
ee133a6
Calculate total objects to delete by counting files in folders
haitham911 Oct 31, 2024
5153cea
Merge branch 'main' into DEV-346
haitham911 Oct 31, 2024
bb63396
Update dependencies in go.mod and go.sum to include new packages
haitham911 Oct 31, 2024
dc5934b
Merge branch 'main' into DEV-346
haitham911 Nov 2, 2024
fbaa62a
update dependencies: upgrade lipgloss to v1.0.0 and x/ansi to v0.4.2;
haitham911 Nov 2, 2024
9155801
Merge branch 'main' into DEV-346
aknysh Nov 6, 2024
9390f77
Merge branch 'main' into DEV-346
haitham911 Nov 13, 2024
55b89c5
chore: update charmbracelet/x dependencies in go.mod
haitham911 Nov 13, 2024
30e8b05
fix: remove duplicate help message for clean operation in help.go
haitham911 Nov 14, 2024
1fcf12d
Merge branch 'main' into DEV-346
osterman Nov 14, 2024
d5cdb29
feat: enhance terraform clean command with --everything and --force o…
haitham911 Nov 14, 2024
f774bfe
Update website/docs/cli/commands/terraform/terraform-clean.mdx
osterman Nov 14, 2024
e2b4e96
Update website/docs/cli/commands/terraform/terraform-clean.mdx
osterman Nov 14, 2024
4759e13
Update website/docs/cli/commands/terraform/usage.mdx
osterman Nov 14, 2024
345fb38
Update website/docs/cli/commands/terraform/usage.mdx
osterman Nov 14, 2024
9d9ee69
Update website/docs/cli/commands/terraform/usage.mdx
osterman Nov 14, 2024
1ad31a6
Update website/docs/cli/commands/terraform/usage.mdx
osterman Nov 14, 2024
fad5e6d
Refactor error handling in handleCleanSubCommand for TF_DATA_DIR vali…
haitham911 Nov 15, 2024
191c444
Merge branch 'main' into DEV-346
osterman Nov 17, 2024
cee4ffc
fix: correct typo in documentation
haitham911 Nov 17, 2024
59940ce
fix html doc
haitham911 Nov 17, 2024
870ce20
fix: validate that the base path exists in CollectDirectoryObjects
haitham911 Nov 18, 2024
cbecdcf
fix: pass cliConfig to findFoldersNamesWithPrefix and getStackTerrafo…
haitham911 Nov 19, 2024
813e9eb
fix: enhance DeletePathTerraform to handle symbolic links and improve…
haitham911 Nov 21, 2024
f452447
fix: improve error handling in deleteFolders function for better dele…
haitham911 Nov 24, 2024
149f4c9
Merge branch 'main' into DEV-346
haitham911 Nov 24, 2024
5e0c92f
fix: streamline error handling in deleteFolders function for improved…
haitham911 Nov 24, 2024
5564dc0
Merge branch 'main' into DEV-346
osterman Dec 4, 2024
d7cfbbf
Merge branch 'main' into DEV-346
aknysh Dec 5, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
14 changes: 10 additions & 4 deletions internal/exec/help.go
Original file line number Diff line number Diff line change
Expand Up @@ -30,8 +30,10 @@ func processHelp(componentType string, command string) error {
u.PrintMessage(" - 'atmos terraform apply' and 'atmos terraform deploy' commands commands support '--planfile' flag to specify the path " +
"to a planfile. The '--planfile' flag should be used instead of the planfile argument in the native 'terraform apply <planfile>' command")
u.PrintMessage(" - 'atmos terraform clean' command deletes the '.terraform' folder, '.terraform.lock.hcl' lock file, " +
"and the previously generated 'planfile', 'varfile' and 'backend.tf.json' file for the specified component and stack. " +
"Use --skip-lock-file flag to skip deleting the lock file.")
"and the previously generated 'planfile', 'varfile', and 'backend.tf.json' file for the specified component and stack. " +
"Use the --everything flag to also delete the state files ('terraform.tfstate.d') for the component. " +
"Use --skip-lock-file to skip deleting the lock file. " +
"If no component or stack is specified, the clean operation will apply globally to all components.")
haitham911 marked this conversation as resolved.
Show resolved Hide resolved
u.PrintMessage(" - 'atmos terraform workspace' command first runs 'terraform init -reconfigure', then 'terraform workspace select', " +
"and if the workspace was not created before, it then runs 'terraform workspace new'")
u.PrintMessage(" - 'atmos terraform import' command searches for 'region' in the variables for the specified component and stack, " +
Expand Down Expand Up @@ -72,10 +74,14 @@ func processHelp(componentType string, command string) error {
" - '.terraform.lock.hcl' file\n" +
" - generated varfile for the component in the stack\n" +
" - generated planfile for the component in the stack\n" +
" - generated 'backend.tf.json' file\n\n" +
" - generated 'backend.tf.json' file\n" +
" - 'terraform.tfstate.d' folder (if '--everything' flag is used)\n\n" +
"Usage: atmos terraform clean <component> -s <stack> <flags>\n\n" +
"Use '--skip-lock-file' flag to skip deleting the lock file.\n\n" +
"Use '--everything' flag to also delete the Terraform state files ('terraform.tfstate.d').\n" +
"Use '--skip-lock-file' flag to skip deleting the '.terraform.lock.hcl' file.\n\n" +
"If no component or stack is specified, the clean operation will apply globally to all components.\n\n" +
"For more details refer to https://atmos.tools/cli/commands/terraform/clean\n")

haitham911 marked this conversation as resolved.
Show resolved Hide resolved
} else if componentType == "terraform" && command == "deploy" {
u.PrintMessage("\n'atmos terraform deploy' command executes 'terraform apply -auto-approve' on an Atmos component in an Atmos stack.\n\n" +
"Usage: atmos terraform deploy <component> -s <stack> <flags>\n\n" +
Expand Down
95 changes: 95 additions & 0 deletions internal/exec/path_utils.go
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,10 @@ package exec

import (
"fmt"
"os"
"path"
"path/filepath"
"strings"
haitham911 marked this conversation as resolved.
Show resolved Hide resolved

"github.com/cloudposse/atmos/pkg/schema"
)
Expand Down Expand Up @@ -85,3 +88,95 @@ func constructHelmfileComponentVarfilePath(cliConfig schema.CliConfiguration, in
constructHelmfileComponentVarfileName(info),
)
}

// findFoldersNamesWithPrefix finds all the folders with the specified prefix return the list of folder names
// If prefix is empty, it returns all the folders in the root
func findFoldersNamesWithPrefix(root, prefix string) ([]string, error) {
haitham911 marked this conversation as resolved.
Show resolved Hide resolved
var folderNames []string

// First, read the directories at the root level (level 1)
level1Dirs, err := os.ReadDir(root)
if err != nil {
return nil, err
}

for _, dir := range level1Dirs {
if dir.IsDir() {
// If the directory at level 1 matches the prefix, add it
if prefix == "" || strings.HasPrefix(dir.Name(), prefix) {
folderNames = append(folderNames, dir.Name())
}
haitham911 marked this conversation as resolved.
Show resolved Hide resolved

// Now, explore one level deeper (level 2)
level2Path := filepath.Join(root, dir.Name())
level2Dirs, err := os.ReadDir(level2Path)
if err != nil {
return nil, err
}

for _, subDir := range level2Dirs {
if subDir.IsDir() && (prefix == "" || strings.HasPrefix(subDir.Name(), prefix)) {
folderNames = append(folderNames, filepath.Join(dir.Name(), subDir.Name()))
}
}
}
}

return folderNames, nil
}
haitham911 marked this conversation as resolved.
Show resolved Hide resolved

// deleteFilesAndFoldersRecursive deletes files and folders from the given list if they exist under the specified path,
// including files and folders in the second-level subdirectories.
func deleteFilesAndFoldersRecursive(basePath string, items []string) error {
// First, delete files and folders directly under the base path
for _, item := range items {
fullPath := filepath.Join(basePath, item)

// Check if the file or folder exists
if _, err := os.Stat(fullPath); err == nil {
// File or folder exists, attempt to delete
err := os.RemoveAll(fullPath)
if err != nil {
if os.IsNotExist(err) {
continue
}
return fmt.Errorf("failed to delete %s: %w", fullPath, err)
}
lastFolderName := filepath.Base(basePath)
fmt.Printf("Deleted: %s/%s\n", lastFolderName, item)
}
haitham911 marked this conversation as resolved.
Show resolved Hide resolved
haitham911 marked this conversation as resolved.
Show resolved Hide resolved
}

// Now, delete matching files and folders from immediate subdirectories
entries, err := os.ReadDir(basePath)
if err != nil {
return fmt.Errorf("error reading the base path %s: %v", basePath, err)
}

for _, entry := range entries {
// Only proceed if the entry is a directory
if entry.IsDir() {
subDirPath := filepath.Join(basePath, entry.Name())

for _, item := range items {
fullPath := filepath.Join(subDirPath, item)

// Check if the file or folder exists in the subdirectory
if _, err := os.Stat(fullPath); err == nil {
// File or folder exists, attempt to delete
err := os.RemoveAll(fullPath)
if err != nil {
if os.IsNotExist(err) {
continue
}
return fmt.Errorf("failed to delete %s: %v", fullPath, err)
}
lastFolderName := filepath.Base(subDirPath)
fmt.Printf("Deleted: %s/%s\n", lastFolderName, item)
}
haitham911 marked this conversation as resolved.
Show resolved Hide resolved
}
}
}

return nil
}
haitham911 marked this conversation as resolved.
Show resolved Hide resolved
haitham911 marked this conversation as resolved.
Show resolved Hide resolved
63 changes: 54 additions & 9 deletions internal/exec/terraform.go
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@ const (
outFlag = "-out"
varFileFlag = "-var-file"
skipTerraformLockFileFlag = "--skip-lock-file"
everythingFlag = "--everything"
)

// ExecuteTerraformCmd parses the provided arguments and flags and executes terraform commands
Expand All @@ -29,7 +30,6 @@ func ExecuteTerraformCmd(cmd *cobra.Command, args []string, additionalArgsAndFla
if err != nil {
return err
}

return ExecuteTerraform(info)
}

Expand Down Expand Up @@ -61,20 +61,32 @@ func ExecuteTerraform(info schema.ConfigAndStacksInfo) error {
return nil
}

info, err = ProcessStacks(cliConfig, info, true, true)
if err != nil {
return err
NeedProcessStacks := true
CheckStack := true
// If the user runs `atmos terraform clean --everything`, don't process stacks
if info.SubCommand == "clean" && u.SliceContainsString(info.AdditionalArgsAndFlags, everythingFlag) {
if info.ComponentFromArg == "" {
NeedProcessStacks = false
}
fmt.Print("componentFromArg: ", info.ComponentFromArg, "\n")
CheckStack = false

haitham911 marked this conversation as resolved.
Show resolved Hide resolved
}

if len(info.Stack) < 1 {
return errors.New("stack must be specified")
if NeedProcessStacks {
info, err = ProcessStacks(cliConfig, info, CheckStack, true)
if err != nil {
return err
}
if len(info.Stack) < 1 && CheckStack {
return errors.New("stack must be specified")
}
}

err = checkTerraformConfig(cliConfig)
if err != nil {
return err
}

// Check if the component (or base component) exists as Terraform component
componentPath := path.Join(cliConfig.TerraformDirAbsolutePath, info.ComponentFolderPrefix, info.FinalComponent)
componentPathExists, err := u.IsDirectory(componentPath)
Expand All @@ -94,14 +106,47 @@ func ExecuteTerraform(info schema.ConfigAndStacksInfo) error {

varFile := constructTerraformComponentVarfileName(info)
planFile := constructTerraformComponentPlanfileName(info)

if info.SubCommand == "clean" {
// If the --everything flag is provided, delete the Terraform state folder for this component
if u.SliceContainsString(info.AdditionalArgsAndFlags, everythingFlag) {
listOfClear := []string{"backend.tf.json", ".terraform", "terraform.tfstate.d", ".terraform.lock.hcl"}
// If the component is not specified, delete the Terraform state folder for all components
if info.ComponentFromArg == "" {
err := deleteFilesAndFoldersRecursive(componentPath, listOfClear)
if err != nil {
u.LogWarning(cliConfig, err.Error())
}
return nil
// If the component is specified, delete the Terraform state folder for the specified component
} else if info.ComponentFromArg != "" && info.StackFromArg == "" {
haitham911 marked this conversation as resolved.
Show resolved Hide resolved
componentPath = path.Join(componentPath, info.Component)
err := deleteFilesAndFoldersRecursive(componentPath, listOfClear)
if err != nil {
u.LogWarning(cliConfig, err.Error())
}
return nil
} else {
// If the component and stack are specified, delete the Terraform state folder for the specified component and stack
tfStateFolderPath := path.Join(componentPath, "terraform.tfstate.d")
tfStateFolderNames, err := findFoldersNamesWithPrefix(tfStateFolderPath, info.StackFromArg)
if err != nil {
u.LogWarning(cliConfig, err.Error())
}
for _, folderName := range tfStateFolderNames {
tfStateFolderPath := path.Join(componentPath, "terraform.tfstate.d", folderName)
u.LogInfo(cliConfig, fmt.Sprintf("Deleting 'terraform.tfstate.d/%s' folder", folderName))
err = os.RemoveAll(tfStateFolderPath)
if err != nil {
u.LogWarning(cliConfig, err.Error())
}
}
haitham911 marked this conversation as resolved.
Show resolved Hide resolved
}
}
haitham911 marked this conversation as resolved.
Show resolved Hide resolved
u.LogInfo(cliConfig, "Deleting '.terraform' folder")
err = os.RemoveAll(path.Join(componentPath, ".terraform"))
if err != nil {
u.LogWarning(cliConfig, err.Error())
}

if !u.SliceContainsString(info.AdditionalArgsAndFlags, skipTerraformLockFileFlag) {
u.LogInfo(cliConfig, "Deleting '.terraform.lock.hcl' file")
_ = os.Remove(path.Join(componentPath, ".terraform.lock.hcl"))
Expand Down
11 changes: 9 additions & 2 deletions internal/exec/utils.go
Original file line number Diff line number Diff line change
Expand Up @@ -391,6 +391,9 @@ func ProcessStacks(
}

if foundStackCount == 0 {
if !checkStack {
return configAndStacksInfo, nil
}
haitham911 marked this conversation as resolved.
Show resolved Hide resolved
cliConfigYaml := ""

if cliConfig.Logs.Level == u.LogLevelTrace {
Expand Down Expand Up @@ -960,7 +963,6 @@ func processArgsAndFlags(componentType string, inputArgsAndFlags []string) (sche
// Handle terraform two-words commands
// https://developer.hashicorp.com/terraform/cli/commands
if componentType == "terraform" {

// Handle the custom legacy command `terraform write varfile` (NOTE: use `terraform generate varfile` instead)
if additionalArgsAndFlags[0] == "write" && additionalArgsAndFlags[1] == "varfile" {
info.SubCommand = "write"
Expand Down Expand Up @@ -997,7 +999,12 @@ func processArgsAndFlags(componentType string, inputArgsAndFlags []string) (sche
}
} else {
info.SubCommand = additionalArgsAndFlags[0]
info.ComponentFromArg = additionalArgsAndFlags[1]
// check if additionalArgsAndFlags[1] is a option with --
if strings.HasPrefix(additionalArgsAndFlags[1], "--") {
info.AdditionalArgsAndFlags = append(info.AdditionalArgsAndFlags, additionalArgsAndFlags[1])
} else {
info.ComponentFromArg = additionalArgsAndFlags[1]
}
haitham911 marked this conversation as resolved.
Show resolved Hide resolved
if len(additionalArgsAndFlags) > 2 {
info.AdditionalArgsAndFlags = additionalArgsAndFlags[2:]
}
Expand Down