2017-02-23 19:13:28 +01:00
|
|
|
package command
|
|
|
|
|
|
|
|
import (
|
2017-04-01 21:42:13 +02:00
|
|
|
"context"
|
2017-02-23 19:13:28 +01:00
|
|
|
"fmt"
|
|
|
|
"os"
|
|
|
|
"strings"
|
|
|
|
|
2017-04-01 20:58:19 +02:00
|
|
|
"github.com/hashicorp/terraform/command/clistate"
|
2017-02-23 19:13:28 +01:00
|
|
|
"github.com/hashicorp/terraform/terraform"
|
|
|
|
"github.com/mitchellh/cli"
|
2017-09-26 04:02:12 +02:00
|
|
|
"github.com/posener/complete"
|
2017-02-23 19:13:28 +01:00
|
|
|
)
|
|
|
|
|
2017-05-31 00:06:13 +02:00
|
|
|
type WorkspaceNewCommand struct {
|
2017-02-23 19:13:28 +01:00
|
|
|
Meta
|
2017-05-31 00:06:13 +02:00
|
|
|
LegacyName bool
|
2017-02-23 19:13:28 +01:00
|
|
|
}
|
|
|
|
|
2017-05-31 00:06:13 +02:00
|
|
|
func (c *WorkspaceNewCommand) Run(args []string) int {
|
2017-03-08 05:09:48 +01:00
|
|
|
args, err := c.Meta.process(args, true)
|
|
|
|
if err != nil {
|
|
|
|
return 1
|
|
|
|
}
|
2017-02-23 19:13:28 +01:00
|
|
|
|
2017-05-31 00:06:13 +02:00
|
|
|
envCommandShowWarning(c.Ui, c.LegacyName)
|
|
|
|
|
2017-02-23 19:13:28 +01:00
|
|
|
statePath := ""
|
|
|
|
|
2017-05-31 00:06:13 +02:00
|
|
|
cmdFlags := c.Meta.flagSet("workspace new")
|
2017-02-23 19:13:28 +01:00
|
|
|
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()
|
2017-02-23 20:14:51 +01:00
|
|
|
if len(args) == 0 {
|
2017-03-01 21:59:40 +01:00
|
|
|
c.Ui.Error("Expected a single argument: NAME.\n")
|
2017-02-23 19:13:28 +01:00
|
|
|
return cli.RunResultHelp
|
|
|
|
}
|
|
|
|
|
|
|
|
newEnv := args[0]
|
|
|
|
|
2017-05-31 02:13:43 +02:00
|
|
|
if !validWorkspaceName(newEnv) {
|
2017-03-27 22:16:09 +02:00
|
|
|
c.Ui.Error(fmt.Sprintf(envInvalidName, newEnv))
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
2017-05-31 00:06:13 +02:00
|
|
|
// You can't ask to create a workspace when you're overriding the
|
|
|
|
// workspace name to be something different.
|
2017-05-31 02:13:43 +02:00
|
|
|
if current, isOverridden := c.WorkspaceOverridden(); current != newEnv && isOverridden {
|
2017-05-12 22:53:29 +02:00
|
|
|
c.Ui.Error(envIsOverriddenNewError)
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
2017-02-23 20:14:51 +01:00
|
|
|
configPath, err := ModulePath(args[1:])
|
|
|
|
if err != nil {
|
|
|
|
c.Ui.Error(err.Error())
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
2017-05-01 23:47:53 +02:00
|
|
|
conf, err := c.Config(configPath)
|
|
|
|
if err != nil {
|
|
|
|
c.Ui.Error(fmt.Sprintf("Failed to load root config module: %s", err))
|
|
|
|
}
|
|
|
|
|
2017-02-23 19:13:28 +01:00
|
|
|
// Load the backend
|
2017-05-01 23:47:53 +02:00
|
|
|
b, err := c.Backend(&BackendOpts{
|
|
|
|
Config: conf,
|
|
|
|
})
|
|
|
|
|
2017-02-23 19:13:28 +01:00
|
|
|
if err != nil {
|
|
|
|
c.Ui.Error(fmt.Sprintf("Failed to load backend: %s", err))
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
2017-02-28 19:13:03 +01:00
|
|
|
states, err := b.States()
|
2017-07-11 17:01:02 +02:00
|
|
|
if err != nil {
|
|
|
|
c.Ui.Error(fmt.Sprintf("Failed to get configured named states: %s", err))
|
2017-07-20 00:46:21 +02:00
|
|
|
return 1
|
2017-07-11 17:01:02 +02:00
|
|
|
}
|
2017-02-23 19:13:28 +01:00
|
|
|
for _, s := range states {
|
|
|
|
if newEnv == s {
|
|
|
|
c.Ui.Error(fmt.Sprintf(envExists, newEnv))
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-02-28 19:13:03 +01:00
|
|
|
_, err = b.State(newEnv)
|
2017-02-23 19:13:28 +01:00
|
|
|
if err != nil {
|
|
|
|
c.Ui.Error(err.Error())
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
2017-05-31 00:06:13 +02:00
|
|
|
// now set the current workspace locally
|
2017-05-31 02:13:43 +02:00
|
|
|
if err := c.SetWorkspace(newEnv); err != nil {
|
2017-05-31 00:06:13 +02:00
|
|
|
c.Ui.Error(fmt.Sprintf("Error selecting new workspace: %s", err))
|
2017-02-28 19:13:03 +01:00
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
2017-03-01 21:59:40 +01:00
|
|
|
c.Ui.Output(c.Colorize().Color(fmt.Sprintf(
|
|
|
|
strings.TrimSpace(envCreated), newEnv)))
|
2017-02-23 19:13:28 +01:00
|
|
|
|
|
|
|
if statePath == "" {
|
|
|
|
// if we're not loading a state, then we're done
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
|
|
|
|
// load the new Backend state
|
2017-02-28 19:13:03 +01:00
|
|
|
sMgr, err := b.State(newEnv)
|
2017-02-23 19:13:28 +01:00
|
|
|
if err != nil {
|
|
|
|
c.Ui.Error(err.Error())
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
2017-04-01 21:42:13 +02:00
|
|
|
if c.stateLock {
|
2018-02-23 17:28:47 +01:00
|
|
|
stateLocker := clistate.NewLocker(context.Background(), c.stateLockTimeout, c.Ui, c.Colorize())
|
|
|
|
if err := stateLocker.Lock(sMgr, "workspace_delete"); err != nil {
|
2017-04-01 21:42:13 +02:00
|
|
|
c.Ui.Error(fmt.Sprintf("Error locking state: %s", err))
|
|
|
|
return 1
|
|
|
|
}
|
2018-02-23 17:28:47 +01:00
|
|
|
defer stateLocker.Unlock(nil)
|
2017-02-23 19:13:28 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// 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
|
|
|
|
}
|
2017-04-12 19:42:23 +02:00
|
|
|
err = sMgr.PersistState()
|
|
|
|
if err != nil {
|
|
|
|
c.Ui.Error(err.Error())
|
|
|
|
return 1
|
|
|
|
}
|
2017-02-23 19:13:28 +01:00
|
|
|
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
|
2017-09-26 04:02:12 +02:00
|
|
|
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"),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-05-31 00:06:13 +02:00
|
|
|
func (c *WorkspaceNewCommand) Help() string {
|
2017-02-23 19:13:28 +01:00
|
|
|
helpText := `
|
2017-05-31 00:06:13 +02:00
|
|
|
Usage: terraform workspace new [OPTIONS] NAME [DIR]
|
2017-02-23 19:13:28 +01:00
|
|
|
|
2017-05-31 00:06:13 +02:00
|
|
|
Create a new Terraform workspace.
|
2017-02-23 19:13:28 +01:00
|
|
|
|
|
|
|
|
|
|
|
Options:
|
|
|
|
|
2017-05-31 00:06:13 +02:00
|
|
|
-state=path Copy an existing state file into the new workspace.
|
2017-02-23 19:13:28 +01:00
|
|
|
`
|
|
|
|
return strings.TrimSpace(helpText)
|
|
|
|
}
|
|
|
|
|
2017-05-31 00:06:13 +02:00
|
|
|
func (c *WorkspaceNewCommand) Synopsis() string {
|
|
|
|
return "Create a new workspace"
|
2017-02-23 19:13:28 +01:00
|
|
|
}
|