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