mirror of
https://github.com/opentofu/opentofu.git
synced 2024-12-30 10:47:14 -06:00
ebafa51723
This is a rather-messy, complex change to get the "command" package building again against the new backend API that was updated for the new configuration loader. A lot of this is mechanical rewriting to the new API, but meta_config.go and meta_backend.go in particular saw some major changes to interface with the new loader APIs and to deal with the change in order of steps in the backend API.
189 lines
3.9 KiB
Go
189 lines
3.9 KiB
Go
package command
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"os"
|
|
"strings"
|
|
|
|
"github.com/hashicorp/terraform/command/clistate"
|
|
"github.com/hashicorp/terraform/terraform"
|
|
"github.com/hashicorp/terraform/tfdiags"
|
|
"github.com/mitchellh/cli"
|
|
"github.com/posener/complete"
|
|
)
|
|
|
|
type WorkspaceNewCommand struct {
|
|
Meta
|
|
LegacyName bool
|
|
}
|
|
|
|
func (c *WorkspaceNewCommand) Run(args []string) int {
|
|
args, err := c.Meta.process(args, true)
|
|
if err != nil {
|
|
return 1
|
|
}
|
|
|
|
envCommandShowWarning(c.Ui, c.LegacyName)
|
|
|
|
statePath := ""
|
|
|
|
cmdFlags := c.Meta.flagSet("workspace new")
|
|
cmdFlags.StringVar(&statePath, "state", "", "terraform state file")
|
|
cmdFlags.Usage = func() { c.Ui.Error(c.Help()) }
|
|
if err := cmdFlags.Parse(args); err != nil {
|
|
return 1
|
|
}
|
|
args = cmdFlags.Args()
|
|
if len(args) == 0 {
|
|
c.Ui.Error("Expected a single argument: NAME.\n")
|
|
return cli.RunResultHelp
|
|
}
|
|
|
|
newEnv := args[0]
|
|
|
|
if !validWorkspaceName(newEnv) {
|
|
c.Ui.Error(fmt.Sprintf(envInvalidName, newEnv))
|
|
return 1
|
|
}
|
|
|
|
// You can't ask to create a workspace when you're overriding the
|
|
// workspace name to be something different.
|
|
if current, isOverridden := c.WorkspaceOverridden(); current != newEnv && isOverridden {
|
|
c.Ui.Error(envIsOverriddenNewError)
|
|
return 1
|
|
}
|
|
|
|
configPath, err := ModulePath(args[1:])
|
|
if err != nil {
|
|
c.Ui.Error(err.Error())
|
|
return 1
|
|
}
|
|
|
|
var diags tfdiags.Diagnostics
|
|
|
|
backendConfig, backendDiags := c.loadBackendConfig(configPath)
|
|
diags = diags.Append(backendDiags)
|
|
if diags.HasErrors() {
|
|
c.showDiagnostics(diags)
|
|
return 1
|
|
}
|
|
|
|
// Load the backend
|
|
b, backendDiags := c.Backend(&BackendOpts{
|
|
Config: backendConfig,
|
|
})
|
|
diags = diags.Append(backendDiags)
|
|
if backendDiags.HasErrors() {
|
|
c.showDiagnostics(diags)
|
|
return 1
|
|
}
|
|
|
|
states, err := b.States()
|
|
if err != nil {
|
|
c.Ui.Error(fmt.Sprintf("Failed to get configured named states: %s", err))
|
|
return 1
|
|
}
|
|
for _, s := range states {
|
|
if newEnv == s {
|
|
c.Ui.Error(fmt.Sprintf(envExists, newEnv))
|
|
return 1
|
|
}
|
|
}
|
|
|
|
_, err = b.State(newEnv)
|
|
if err != nil {
|
|
c.Ui.Error(err.Error())
|
|
return 1
|
|
}
|
|
|
|
// now set the current workspace locally
|
|
if err := c.SetWorkspace(newEnv); err != nil {
|
|
c.Ui.Error(fmt.Sprintf("Error selecting new workspace: %s", err))
|
|
return 1
|
|
}
|
|
|
|
c.Ui.Output(c.Colorize().Color(fmt.Sprintf(
|
|
strings.TrimSpace(envCreated), newEnv)))
|
|
|
|
if statePath == "" {
|
|
// if we're not loading a state, then we're done
|
|
return 0
|
|
}
|
|
|
|
// load the new Backend state
|
|
sMgr, err := b.State(newEnv)
|
|
if err != nil {
|
|
c.Ui.Error(err.Error())
|
|
return 1
|
|
}
|
|
|
|
if c.stateLock {
|
|
stateLocker := clistate.NewLocker(context.Background(), c.stateLockTimeout, c.Ui, c.Colorize())
|
|
if err := stateLocker.Lock(sMgr, "workspace_delete"); err != nil {
|
|
c.Ui.Error(fmt.Sprintf("Error locking state: %s", err))
|
|
return 1
|
|
}
|
|
defer stateLocker.Unlock(nil)
|
|
}
|
|
|
|
// read the existing state file
|
|
stateFile, err := os.Open(statePath)
|
|
if err != nil {
|
|
c.Ui.Error(err.Error())
|
|
return 1
|
|
}
|
|
|
|
s, err := terraform.ReadState(stateFile)
|
|
if err != nil {
|
|
c.Ui.Error(err.Error())
|
|
return 1
|
|
}
|
|
|
|
// save the existing state in the new Backend.
|
|
err = sMgr.WriteState(s)
|
|
if err != nil {
|
|
c.Ui.Error(err.Error())
|
|
return 1
|
|
}
|
|
err = sMgr.PersistState()
|
|
if err != nil {
|
|
c.Ui.Error(err.Error())
|
|
return 1
|
|
}
|
|
|
|
return 0
|
|
}
|
|
|
|
func (c *WorkspaceNewCommand) AutocompleteArgs() complete.Predictor {
|
|
return completePredictSequence{
|
|
complete.PredictNothing, // the "new" subcommand itself (already matched)
|
|
complete.PredictAnything,
|
|
complete.PredictDirs(""),
|
|
}
|
|
}
|
|
|
|
func (c *WorkspaceNewCommand) AutocompleteFlags() complete.Flags {
|
|
return complete.Flags{
|
|
"-state": complete.PredictFiles("*.tfstate"),
|
|
}
|
|
}
|
|
|
|
func (c *WorkspaceNewCommand) Help() string {
|
|
helpText := `
|
|
Usage: terraform workspace new [OPTIONS] NAME [DIR]
|
|
|
|
Create a new Terraform workspace.
|
|
|
|
|
|
Options:
|
|
|
|
-state=path Copy an existing state file into the new workspace.
|
|
`
|
|
return strings.TrimSpace(helpText)
|
|
}
|
|
|
|
func (c *WorkspaceNewCommand) Synopsis() string {
|
|
return "Create a new workspace"
|
|
}
|