2023-08-23 10:35:42 +02:00
|
|
|
/*
|
|
|
|
Copyright (c) Edgeless Systems GmbH
|
|
|
|
|
|
|
|
SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
*/
|
|
|
|
|
|
|
|
package cloudcmd
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2023-10-31 12:46:40 +01:00
|
|
|
"errors"
|
2023-08-23 10:35:42 +02:00
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"os"
|
|
|
|
|
|
|
|
"github.com/edgelesssys/constellation/v2/cli/internal/terraform"
|
|
|
|
"github.com/edgelesssys/constellation/v2/internal/file"
|
|
|
|
)
|
|
|
|
|
2023-10-31 12:46:40 +01:00
|
|
|
// plan prepares a workspace and plans the possible Terraform actions.
|
|
|
|
// This will either create a new workspace or update an existing one.
|
2023-08-23 10:35:42 +02:00
|
|
|
// In case of possible migrations, the diff is written to outWriter and this function returns true.
|
2023-10-31 12:46:40 +01:00
|
|
|
func plan(
|
|
|
|
ctx context.Context, tfClient tfPlanner, fileHandler file.Handler,
|
2023-08-23 10:35:42 +02:00
|
|
|
outWriter io.Writer, logLevel terraform.LogLevel, vars terraform.Variables,
|
2023-10-26 10:55:50 +02:00
|
|
|
templateDir, existingWorkspace, backupDir string,
|
2023-08-23 10:35:42 +02:00
|
|
|
) (bool, error) {
|
2023-10-31 12:46:40 +01:00
|
|
|
isNewWorkspace, err := fileHandler.IsEmpty(existingWorkspace)
|
|
|
|
if err != nil {
|
|
|
|
if !errors.Is(err, os.ErrNotExist) {
|
|
|
|
return false, fmt.Errorf("checking if workspace is empty: %w", err)
|
|
|
|
}
|
|
|
|
isNewWorkspace = true
|
2023-08-23 10:35:42 +02:00
|
|
|
}
|
|
|
|
|
2023-10-31 12:46:40 +01:00
|
|
|
// Backup old workspace if it exists
|
|
|
|
if !isNewWorkspace {
|
|
|
|
if err := ensureFileNotExist(fileHandler, backupDir); err != nil {
|
|
|
|
return false, fmt.Errorf("backup directory %s already exists: %w", backupDir, err)
|
|
|
|
}
|
|
|
|
if err := fileHandler.CopyDir(existingWorkspace, backupDir); err != nil {
|
|
|
|
return false, fmt.Errorf("backing up old workspace: %w", err)
|
|
|
|
}
|
2023-10-26 10:55:50 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Move the new embedded Terraform files into the workspace.
|
|
|
|
if err := tfClient.PrepareWorkspace(templateDir, vars); err != nil {
|
2023-08-23 10:35:42 +02:00
|
|
|
return false, fmt.Errorf("preparing terraform workspace: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
hasDiff, err := tfClient.Plan(ctx, logLevel)
|
|
|
|
if err != nil {
|
|
|
|
return false, fmt.Errorf("terraform plan: %w", err)
|
|
|
|
}
|
|
|
|
|
2023-11-20 11:17:16 +01:00
|
|
|
// If we are planning in a new workspace, we don't want to show the plan
|
2023-10-31 12:46:40 +01:00
|
|
|
if isNewWorkspace {
|
2023-11-20 11:17:16 +01:00
|
|
|
return hasDiff, nil
|
2023-10-31 12:46:40 +01:00
|
|
|
}
|
|
|
|
|
2023-08-23 10:35:42 +02:00
|
|
|
if hasDiff {
|
|
|
|
if err := tfClient.ShowPlan(ctx, logLevel, outWriter); err != nil {
|
|
|
|
return false, fmt.Errorf("terraform show plan: %w", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return hasDiff, nil
|
|
|
|
}
|
|
|
|
|
2023-09-14 11:51:20 +02:00
|
|
|
// restoreBackup replaces the existing Terraform workspace with the backup.
|
2023-11-20 11:17:16 +01:00
|
|
|
// If no backup exists, this function simply removes workingDir.
|
2023-09-14 11:51:20 +02:00
|
|
|
func restoreBackup(fileHandler file.Handler, workingDir, backupDir string) error {
|
|
|
|
if err := fileHandler.RemoveAll(workingDir); err != nil {
|
|
|
|
return fmt.Errorf("removing existing workspace: %w", err)
|
2023-08-23 10:35:42 +02:00
|
|
|
}
|
|
|
|
if err := fileHandler.CopyDir(
|
2023-09-14 11:51:20 +02:00
|
|
|
backupDir,
|
|
|
|
workingDir,
|
2023-11-20 11:17:16 +01:00
|
|
|
); err != nil && !errors.Is(err, os.ErrNotExist) { // ignore not found error because backup does not exist for new clusters
|
2023-09-14 11:51:20 +02:00
|
|
|
return fmt.Errorf("replacing terraform workspace with backup: %w", err)
|
2023-08-23 10:35:42 +02:00
|
|
|
}
|
|
|
|
|
2023-09-14 11:51:20 +02:00
|
|
|
if err := fileHandler.RemoveAll(backupDir); err != nil {
|
|
|
|
return fmt.Errorf("removing backup directory: %w", err)
|
2023-08-23 10:35:42 +02:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// ensureFileNotExist checks if a single file or directory does not exist, returning an error if it does.
|
|
|
|
func ensureFileNotExist(fileHandler file.Handler, fileName string) error {
|
|
|
|
if _, err := fileHandler.Stat(fileName); err != nil {
|
|
|
|
if !os.IsNotExist(err) {
|
|
|
|
return fmt.Errorf("checking %q: %w", fileName, err)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return fmt.Errorf("%q already exists", fileName)
|
|
|
|
}
|