2021-09-22 16:53:33 -05:00
|
|
|
//go:build e2e
|
|
|
|
// +build e2e
|
|
|
|
|
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"io/ioutil"
|
|
|
|
"log"
|
2021-10-11 16:44:44 -05:00
|
|
|
"os"
|
2021-09-22 16:53:33 -05:00
|
|
|
"testing"
|
|
|
|
|
2021-10-11 16:44:44 -05:00
|
|
|
expect "github.com/Netflix/go-expect"
|
2021-09-22 16:53:33 -05:00
|
|
|
tfe "github.com/hashicorp/go-tfe"
|
|
|
|
"github.com/hashicorp/terraform/internal/e2e"
|
|
|
|
)
|
|
|
|
|
|
|
|
func Test_terraform_apply_autoApprove(t *testing.T) {
|
|
|
|
ctx := context.Background()
|
2021-10-11 16:44:44 -05:00
|
|
|
tfVersion := "1.1.0-tfc-integration"
|
|
|
|
if !hasTerraformVersion(version) {
|
|
|
|
t.Skip("Skipping test because TFC does not have current terraform version.")
|
|
|
|
}
|
|
|
|
|
2021-09-22 16:53:33 -05:00
|
|
|
cases := map[string]struct {
|
2021-10-11 16:44:44 -05:00
|
|
|
operations []operationSets
|
|
|
|
validations func(t *testing.T, orgName string)
|
2021-09-22 16:53:33 -05:00
|
|
|
}{
|
2021-10-11 16:44:44 -05:00
|
|
|
"workspace manual apply, terraform apply without auto-approve, expect prompt": {
|
|
|
|
operations: []operationSets{
|
2021-09-22 16:53:33 -05:00
|
|
|
{
|
2021-10-11 16:44:44 -05:00
|
|
|
prep: func(t *testing.T, orgName, dir string) {
|
|
|
|
wsName := "app"
|
|
|
|
_ = createWorkspace(t, orgName, tfe.WorkspaceCreateOptions{
|
|
|
|
Name: tfe.String(wsName),
|
|
|
|
TerraformVersion: tfe.String(tfVersion),
|
|
|
|
AutoApply: tfe.Bool(false),
|
|
|
|
})
|
|
|
|
tfBlock := terraformConfigCloudBackendName(orgName, wsName)
|
|
|
|
writeMainTF(t, tfBlock, dir)
|
|
|
|
},
|
|
|
|
commands: []tfCommand{
|
|
|
|
{
|
|
|
|
command: []string{"init"},
|
|
|
|
expectedCmdOutput: `Successfully configured the backend "cloud"!`,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
command: []string{"apply"},
|
|
|
|
expectedCmdOutput: `Do you want to perform these actions in workspace "app"?`,
|
|
|
|
userInput: []string{"yes"},
|
|
|
|
postInputOutput: []string{`Apply complete!`},
|
|
|
|
},
|
|
|
|
},
|
2021-09-22 16:53:33 -05:00
|
|
|
},
|
|
|
|
},
|
2021-10-11 16:44:44 -05:00
|
|
|
validations: func(t *testing.T, orgName string) {
|
|
|
|
workspace, err := tfeClient.Workspaces.ReadWithOptions(ctx, orgName, "app", &tfe.WorkspaceReadOptions{Include: "current_run"})
|
2021-09-22 16:53:33 -05:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if workspace.CurrentRun == nil {
|
|
|
|
t.Fatal("Expected workspace to have run, but got nil")
|
|
|
|
}
|
2021-10-11 16:44:44 -05:00
|
|
|
if workspace.CurrentRun.Status != tfe.RunApplied {
|
|
|
|
t.Fatalf("Expected run status to be `applied`, but is %s", workspace.CurrentRun.Status)
|
2021-09-22 16:53:33 -05:00
|
|
|
}
|
|
|
|
},
|
|
|
|
},
|
2021-10-11 16:44:44 -05:00
|
|
|
"workspace auto apply, terraform apply without auto-approve, expect prompt": {
|
|
|
|
operations: []operationSets{
|
2021-09-22 16:53:33 -05:00
|
|
|
{
|
2021-10-11 16:44:44 -05:00
|
|
|
prep: func(t *testing.T, orgName, dir string) {
|
|
|
|
wsName := "app"
|
|
|
|
_ = createWorkspace(t, orgName, tfe.WorkspaceCreateOptions{
|
|
|
|
Name: tfe.String(wsName),
|
|
|
|
TerraformVersion: tfe.String(tfVersion),
|
|
|
|
AutoApply: tfe.Bool(true),
|
|
|
|
})
|
|
|
|
tfBlock := terraformConfigCloudBackendName(orgName, wsName)
|
|
|
|
writeMainTF(t, tfBlock, dir)
|
|
|
|
},
|
|
|
|
commands: []tfCommand{
|
|
|
|
{
|
|
|
|
command: []string{"init"},
|
|
|
|
expectedCmdOutput: `Successfully configured the backend "cloud"!`,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
command: []string{"apply"},
|
|
|
|
expectedCmdOutput: `Do you want to perform these actions in workspace "app"?`,
|
|
|
|
userInput: []string{"yes"},
|
|
|
|
postInputOutput: []string{`Apply complete!`},
|
|
|
|
},
|
|
|
|
},
|
2021-09-22 16:53:33 -05:00
|
|
|
},
|
|
|
|
},
|
2021-10-11 16:44:44 -05:00
|
|
|
validations: func(t *testing.T, orgName string) {
|
|
|
|
workspace, err := tfeClient.Workspaces.ReadWithOptions(ctx, orgName, "app", &tfe.WorkspaceReadOptions{Include: "current_run"})
|
2021-09-22 16:53:33 -05:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if workspace.CurrentRun == nil {
|
2021-10-11 16:44:44 -05:00
|
|
|
t.Fatal("Expected workspace to have run, but got nil")
|
2021-09-22 16:53:33 -05:00
|
|
|
}
|
2021-10-11 16:44:44 -05:00
|
|
|
if workspace.CurrentRun.Status != tfe.RunApplied {
|
|
|
|
t.Fatalf("Expected run status to be `applied`, but is %s", workspace.CurrentRun.Status)
|
2021-09-22 16:53:33 -05:00
|
|
|
}
|
|
|
|
},
|
|
|
|
},
|
2021-10-11 16:44:44 -05:00
|
|
|
"workspace manual apply, terraform apply with auto-approve, no prompt": {
|
|
|
|
operations: []operationSets{
|
2021-09-22 16:53:33 -05:00
|
|
|
{
|
2021-10-11 16:44:44 -05:00
|
|
|
prep: func(t *testing.T, orgName, dir string) {
|
|
|
|
wsName := "app"
|
|
|
|
_ = createWorkspace(t, orgName, tfe.WorkspaceCreateOptions{
|
|
|
|
Name: tfe.String(wsName),
|
|
|
|
TerraformVersion: tfe.String(tfVersion),
|
|
|
|
AutoApply: tfe.Bool(false),
|
|
|
|
})
|
|
|
|
tfBlock := terraformConfigCloudBackendName(orgName, wsName)
|
|
|
|
writeMainTF(t, tfBlock, dir)
|
|
|
|
},
|
|
|
|
commands: []tfCommand{
|
|
|
|
{
|
|
|
|
command: []string{"init"},
|
|
|
|
expectedCmdOutput: `Successfully configured the backend "cloud"!`,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
command: []string{"apply", "-auto-approve"},
|
|
|
|
expectedCmdOutput: `Apply complete!`,
|
|
|
|
},
|
|
|
|
},
|
2021-09-22 16:53:33 -05:00
|
|
|
},
|
|
|
|
},
|
2021-10-11 16:44:44 -05:00
|
|
|
validations: func(t *testing.T, orgName string) {
|
|
|
|
workspace, err := tfeClient.Workspaces.ReadWithOptions(ctx, orgName, "app", &tfe.WorkspaceReadOptions{Include: "current_run"})
|
2021-09-22 16:53:33 -05:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if workspace.CurrentRun == nil {
|
2021-10-11 16:44:44 -05:00
|
|
|
t.Fatal("Expected workspace to have run, but got nil")
|
2021-09-22 16:53:33 -05:00
|
|
|
}
|
|
|
|
if workspace.CurrentRun.Status != tfe.RunApplied {
|
|
|
|
t.Fatalf("Expected run status to be `applied`, but is %s", workspace.CurrentRun.Status)
|
|
|
|
}
|
|
|
|
},
|
|
|
|
},
|
2021-10-11 16:44:44 -05:00
|
|
|
"workspace auto apply, terraform apply with auto-approve, no prompt": {
|
|
|
|
operations: []operationSets{
|
2021-09-22 16:53:33 -05:00
|
|
|
{
|
2021-10-11 16:44:44 -05:00
|
|
|
prep: func(t *testing.T, orgName, dir string) {
|
|
|
|
wsName := "app"
|
|
|
|
_ = createWorkspace(t, orgName, tfe.WorkspaceCreateOptions{
|
|
|
|
Name: tfe.String(wsName),
|
|
|
|
TerraformVersion: tfe.String(tfVersion),
|
|
|
|
AutoApply: tfe.Bool(true),
|
|
|
|
})
|
|
|
|
tfBlock := terraformConfigCloudBackendName(orgName, wsName)
|
|
|
|
writeMainTF(t, tfBlock, dir)
|
|
|
|
},
|
|
|
|
commands: []tfCommand{
|
|
|
|
{
|
|
|
|
command: []string{"init"},
|
|
|
|
expectedCmdOutput: `Successfully configured the backend "cloud"!`,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
command: []string{"apply", "-auto-approve"},
|
|
|
|
expectedCmdOutput: `Apply complete!`,
|
|
|
|
},
|
|
|
|
},
|
2021-09-22 16:53:33 -05:00
|
|
|
},
|
|
|
|
},
|
2021-10-11 16:44:44 -05:00
|
|
|
validations: func(t *testing.T, orgName string) {
|
|
|
|
workspace, err := tfeClient.Workspaces.ReadWithOptions(ctx, orgName, "app", &tfe.WorkspaceReadOptions{Include: "current_run"})
|
2021-09-22 16:53:33 -05:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if workspace.CurrentRun == nil {
|
2021-10-11 16:44:44 -05:00
|
|
|
t.Fatal("Expected workspace to have run, but got nil")
|
2021-09-22 16:53:33 -05:00
|
|
|
}
|
|
|
|
if workspace.CurrentRun.Status != tfe.RunApplied {
|
|
|
|
t.Fatalf("Expected run status to be `applied`, but is %s", workspace.CurrentRun.Status)
|
|
|
|
}
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
for name, tc := range cases {
|
|
|
|
log.Println("Test: ", name)
|
2021-10-11 16:44:44 -05:00
|
|
|
|
|
|
|
organization, cleanup := createOrganization(t)
|
2021-09-22 16:53:33 -05:00
|
|
|
defer cleanup()
|
2021-10-11 16:44:44 -05:00
|
|
|
exp, err := expect.NewConsole(expect.WithStdout(os.Stdout), expect.WithDefaultTimeout(expectConsoleTimeout))
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer exp.Close()
|
2021-09-22 16:53:33 -05:00
|
|
|
|
|
|
|
tmpDir, err := ioutil.TempDir("", "terraform-test")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2021-10-11 16:44:44 -05:00
|
|
|
defer os.RemoveAll(tmpDir)
|
|
|
|
|
2021-09-22 16:53:33 -05:00
|
|
|
tf := e2e.NewBinary(terraformBin, tmpDir)
|
2021-10-11 16:44:44 -05:00
|
|
|
tf.AddEnv("TF_LOG=info")
|
2021-09-22 16:53:33 -05:00
|
|
|
tf.AddEnv(cliConfigFileEnv)
|
2021-10-11 16:44:44 -05:00
|
|
|
defer tf.Close()
|
2021-09-22 16:53:33 -05:00
|
|
|
|
2021-10-11 16:44:44 -05:00
|
|
|
for _, op := range tc.operations {
|
|
|
|
op.prep(t, organization.Name, tf.WorkDir())
|
|
|
|
for _, tfCmd := range op.commands {
|
|
|
|
cmd := tf.Cmd(tfCmd.command...)
|
|
|
|
cmd.Stdin = exp.Tty()
|
|
|
|
cmd.Stdout = exp.Tty()
|
|
|
|
cmd.Stderr = exp.Tty()
|
|
|
|
|
|
|
|
err = cmd.Start()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if tfCmd.expectedCmdOutput != "" {
|
|
|
|
_, err := exp.ExpectString(tfCmd.expectedCmdOutput)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2021-09-22 16:53:33 -05:00
|
|
|
}
|
|
|
|
|
2021-10-11 16:44:44 -05:00
|
|
|
lenInput := len(tfCmd.userInput)
|
|
|
|
lenInputOutput := len(tfCmd.postInputOutput)
|
|
|
|
if lenInput > 0 {
|
|
|
|
for i := 0; i < lenInput; i++ {
|
|
|
|
input := tfCmd.userInput[i]
|
|
|
|
exp.SendLine(input)
|
|
|
|
// use the index to find the corresponding
|
|
|
|
// output that matches the input.
|
|
|
|
if lenInputOutput-1 >= i {
|
|
|
|
output := tfCmd.postInputOutput[i]
|
|
|
|
_, err := exp.ExpectString(output)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
err = cmd.Wait()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2021-09-22 16:53:33 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-11 16:44:44 -05:00
|
|
|
if tc.validations != nil {
|
|
|
|
tc.validations(t, organization.Name)
|
|
|
|
}
|
2021-09-22 16:53:33 -05:00
|
|
|
}
|
|
|
|
}
|