mirror of
https://github.com/opentofu/opentofu.git
synced 2025-01-17 20:22:58 -06:00
034e944070
This is part of a general effort to move all of Terraform's non-library package surface under internal in order to reinforce that these are for internal use within Terraform only. If you were previously importing packages under this prefix into an external codebase, you could pin to an earlier release tag as an interim solution until you've make a plan to achieve the same functionality some other way.
390 lines
9.7 KiB
Go
390 lines
9.7 KiB
Go
package arguments
|
|
|
|
import (
|
|
"strings"
|
|
"testing"
|
|
|
|
"github.com/google/go-cmp/cmp"
|
|
"github.com/google/go-cmp/cmp/cmpopts"
|
|
"github.com/hashicorp/terraform/internal/addrs"
|
|
"github.com/hashicorp/terraform/internal/plans"
|
|
)
|
|
|
|
func TestParseApply_basicValid(t *testing.T) {
|
|
testCases := map[string]struct {
|
|
args []string
|
|
want *Apply
|
|
}{
|
|
"defaults": {
|
|
nil,
|
|
&Apply{
|
|
AutoApprove: false,
|
|
InputEnabled: true,
|
|
PlanPath: "",
|
|
ViewType: ViewHuman,
|
|
State: &State{Lock: true},
|
|
Vars: &Vars{},
|
|
Operation: &Operation{
|
|
PlanMode: plans.NormalMode,
|
|
Parallelism: 10,
|
|
Refresh: true,
|
|
},
|
|
},
|
|
},
|
|
"auto-approve, disabled input, and plan path": {
|
|
[]string{"-auto-approve", "-input=false", "saved.tfplan"},
|
|
&Apply{
|
|
AutoApprove: true,
|
|
InputEnabled: false,
|
|
PlanPath: "saved.tfplan",
|
|
ViewType: ViewHuman,
|
|
State: &State{Lock: true},
|
|
Vars: &Vars{},
|
|
Operation: &Operation{
|
|
PlanMode: plans.NormalMode,
|
|
Parallelism: 10,
|
|
Refresh: true,
|
|
},
|
|
},
|
|
},
|
|
"destroy mode": {
|
|
[]string{"-destroy"},
|
|
&Apply{
|
|
AutoApprove: false,
|
|
InputEnabled: true,
|
|
PlanPath: "",
|
|
ViewType: ViewHuman,
|
|
State: &State{Lock: true},
|
|
Vars: &Vars{},
|
|
Operation: &Operation{
|
|
PlanMode: plans.DestroyMode,
|
|
Parallelism: 10,
|
|
Refresh: true,
|
|
},
|
|
},
|
|
},
|
|
"JSON view disables input": {
|
|
[]string{"-json", "-auto-approve"},
|
|
&Apply{
|
|
AutoApprove: true,
|
|
InputEnabled: false,
|
|
PlanPath: "",
|
|
ViewType: ViewJSON,
|
|
State: &State{Lock: true},
|
|
Vars: &Vars{},
|
|
Operation: &Operation{
|
|
PlanMode: plans.NormalMode,
|
|
Parallelism: 10,
|
|
Refresh: true,
|
|
},
|
|
},
|
|
},
|
|
}
|
|
|
|
cmpOpts := cmpopts.IgnoreUnexported(Operation{}, Vars{}, State{})
|
|
|
|
for name, tc := range testCases {
|
|
t.Run(name, func(t *testing.T) {
|
|
got, diags := ParseApply(tc.args)
|
|
if len(diags) > 0 {
|
|
t.Fatalf("unexpected diags: %v", diags)
|
|
}
|
|
if diff := cmp.Diff(tc.want, got, cmpOpts); diff != "" {
|
|
t.Errorf("unexpected result\n%s", diff)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestParseApply_json(t *testing.T) {
|
|
testCases := map[string]struct {
|
|
args []string
|
|
wantSuccess bool
|
|
}{
|
|
"-json": {
|
|
[]string{"-json"},
|
|
false,
|
|
},
|
|
"-json -auto-approve": {
|
|
[]string{"-json", "-auto-approve"},
|
|
true,
|
|
},
|
|
"-json saved.tfplan": {
|
|
[]string{"-json", "saved.tfplan"},
|
|
true,
|
|
},
|
|
}
|
|
|
|
for name, tc := range testCases {
|
|
t.Run(name, func(t *testing.T) {
|
|
got, diags := ParseApply(tc.args)
|
|
|
|
if tc.wantSuccess {
|
|
if len(diags) > 0 {
|
|
t.Errorf("unexpected diags: %v", diags)
|
|
}
|
|
} else {
|
|
if got, want := diags.Err().Error(), "Plan file or auto-approve required"; !strings.Contains(got, want) {
|
|
t.Errorf("wrong diags\n got: %s\nwant: %s", got, want)
|
|
}
|
|
}
|
|
|
|
if got.ViewType != ViewJSON {
|
|
t.Errorf("unexpected view type. got: %#v, want: %#v", got.ViewType, ViewJSON)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestParseApply_invalid(t *testing.T) {
|
|
got, diags := ParseApply([]string{"-frob"})
|
|
if len(diags) == 0 {
|
|
t.Fatal("expected diags but got none")
|
|
}
|
|
if got, want := diags.Err().Error(), "flag provided but not defined"; !strings.Contains(got, want) {
|
|
t.Fatalf("wrong diags\n got: %s\nwant: %s", got, want)
|
|
}
|
|
if got.ViewType != ViewHuman {
|
|
t.Fatalf("wrong view type, got %#v, want %#v", got.ViewType, ViewHuman)
|
|
}
|
|
}
|
|
|
|
func TestParseApply_tooManyArguments(t *testing.T) {
|
|
got, diags := ParseApply([]string{"saved.tfplan", "please"})
|
|
if len(diags) == 0 {
|
|
t.Fatal("expected diags but got none")
|
|
}
|
|
if got, want := diags.Err().Error(), "Too many command line arguments"; !strings.Contains(got, want) {
|
|
t.Fatalf("wrong diags\n got: %s\nwant: %s", got, want)
|
|
}
|
|
if got.ViewType != ViewHuman {
|
|
t.Fatalf("wrong view type, got %#v, want %#v", got.ViewType, ViewHuman)
|
|
}
|
|
}
|
|
|
|
func TestParseApply_targets(t *testing.T) {
|
|
foobarbaz, _ := addrs.ParseTargetStr("foo_bar.baz")
|
|
boop, _ := addrs.ParseTargetStr("module.boop")
|
|
testCases := map[string]struct {
|
|
args []string
|
|
want []addrs.Targetable
|
|
wantErr string
|
|
}{
|
|
"no targets by default": {
|
|
args: nil,
|
|
want: nil,
|
|
},
|
|
"one target": {
|
|
args: []string{"-target=foo_bar.baz"},
|
|
want: []addrs.Targetable{foobarbaz.Subject},
|
|
},
|
|
"two targets": {
|
|
args: []string{"-target=foo_bar.baz", "-target", "module.boop"},
|
|
want: []addrs.Targetable{foobarbaz.Subject, boop.Subject},
|
|
},
|
|
"invalid traversal": {
|
|
args: []string{"-target=foo."},
|
|
want: nil,
|
|
wantErr: "Dot must be followed by attribute name",
|
|
},
|
|
"invalid target": {
|
|
args: []string{"-target=data[0].foo"},
|
|
want: nil,
|
|
wantErr: "A data source name is required",
|
|
},
|
|
}
|
|
|
|
for name, tc := range testCases {
|
|
t.Run(name, func(t *testing.T) {
|
|
got, diags := ParseApply(tc.args)
|
|
if len(diags) > 0 {
|
|
if tc.wantErr == "" {
|
|
t.Fatalf("unexpected diags: %v", diags)
|
|
} else if got := diags.Err().Error(); !strings.Contains(got, tc.wantErr) {
|
|
t.Fatalf("wrong diags\n got: %s\nwant: %s", got, tc.wantErr)
|
|
}
|
|
}
|
|
if !cmp.Equal(got.Operation.Targets, tc.want) {
|
|
t.Fatalf("unexpected result\n%s", cmp.Diff(got.Operation.Targets, tc.want))
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestParseApply_replace(t *testing.T) {
|
|
foobarbaz, _ := addrs.ParseAbsResourceInstanceStr("foo_bar.baz")
|
|
foobarbeep, _ := addrs.ParseAbsResourceInstanceStr("foo_bar.beep")
|
|
testCases := map[string]struct {
|
|
args []string
|
|
want []addrs.AbsResourceInstance
|
|
wantErr string
|
|
}{
|
|
"no addresses by default": {
|
|
args: nil,
|
|
want: nil,
|
|
},
|
|
"one address": {
|
|
args: []string{"-replace=foo_bar.baz"},
|
|
want: []addrs.AbsResourceInstance{foobarbaz},
|
|
},
|
|
"two addresses": {
|
|
args: []string{"-replace=foo_bar.baz", "-replace", "foo_bar.beep"},
|
|
want: []addrs.AbsResourceInstance{foobarbaz, foobarbeep},
|
|
},
|
|
"non-resource-instance address": {
|
|
args: []string{"-replace=module.boop"},
|
|
want: nil,
|
|
wantErr: "A resource instance address is required here.",
|
|
},
|
|
"data resource address": {
|
|
args: []string{"-replace=data.foo.bar"},
|
|
want: nil,
|
|
wantErr: "Only managed resources can be used",
|
|
},
|
|
"invalid traversal": {
|
|
args: []string{"-replace=foo."},
|
|
want: nil,
|
|
wantErr: "Dot must be followed by attribute name",
|
|
},
|
|
"invalid address": {
|
|
args: []string{"-replace=data[0].foo"},
|
|
want: nil,
|
|
wantErr: "A data source name is required",
|
|
},
|
|
}
|
|
|
|
for name, tc := range testCases {
|
|
t.Run(name, func(t *testing.T) {
|
|
got, diags := ParseApply(tc.args)
|
|
if len(diags) > 0 {
|
|
if tc.wantErr == "" {
|
|
t.Fatalf("unexpected diags: %v", diags)
|
|
} else if got := diags.Err().Error(); !strings.Contains(got, tc.wantErr) {
|
|
t.Fatalf("wrong diags\n got: %s\nwant: %s", got, tc.wantErr)
|
|
}
|
|
}
|
|
if !cmp.Equal(got.Operation.ForceReplace, tc.want) {
|
|
t.Fatalf("unexpected result\n%s", cmp.Diff(got.Operation.Targets, tc.want))
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestParseApply_vars(t *testing.T) {
|
|
testCases := map[string]struct {
|
|
args []string
|
|
want []FlagNameValue
|
|
}{
|
|
"no var flags by default": {
|
|
args: nil,
|
|
want: nil,
|
|
},
|
|
"one var": {
|
|
args: []string{"-var", "foo=bar"},
|
|
want: []FlagNameValue{
|
|
{Name: "-var", Value: "foo=bar"},
|
|
},
|
|
},
|
|
"one var-file": {
|
|
args: []string{"-var-file", "cool.tfvars"},
|
|
want: []FlagNameValue{
|
|
{Name: "-var-file", Value: "cool.tfvars"},
|
|
},
|
|
},
|
|
"ordering preserved": {
|
|
args: []string{
|
|
"-var", "foo=bar",
|
|
"-var-file", "cool.tfvars",
|
|
"-var", "boop=beep",
|
|
},
|
|
want: []FlagNameValue{
|
|
{Name: "-var", Value: "foo=bar"},
|
|
{Name: "-var-file", Value: "cool.tfvars"},
|
|
{Name: "-var", Value: "boop=beep"},
|
|
},
|
|
},
|
|
}
|
|
|
|
for name, tc := range testCases {
|
|
t.Run(name, func(t *testing.T) {
|
|
got, diags := ParseApply(tc.args)
|
|
if len(diags) > 0 {
|
|
t.Fatalf("unexpected diags: %v", diags)
|
|
}
|
|
if vars := got.Vars.All(); !cmp.Equal(vars, tc.want) {
|
|
t.Fatalf("unexpected result\n%s", cmp.Diff(vars, tc.want))
|
|
}
|
|
if got, want := got.Vars.Empty(), len(tc.want) == 0; got != want {
|
|
t.Fatalf("expected Empty() to return %t, but was %t", want, got)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestParseApplyDestroy_basicValid(t *testing.T) {
|
|
testCases := map[string]struct {
|
|
args []string
|
|
want *Apply
|
|
}{
|
|
"defaults": {
|
|
nil,
|
|
&Apply{
|
|
AutoApprove: false,
|
|
InputEnabled: true,
|
|
ViewType: ViewHuman,
|
|
State: &State{Lock: true},
|
|
Vars: &Vars{},
|
|
Operation: &Operation{
|
|
PlanMode: plans.DestroyMode,
|
|
Parallelism: 10,
|
|
Refresh: true,
|
|
},
|
|
},
|
|
},
|
|
"auto-approve and disabled input": {
|
|
[]string{"-auto-approve", "-input=false"},
|
|
&Apply{
|
|
AutoApprove: true,
|
|
InputEnabled: false,
|
|
ViewType: ViewHuman,
|
|
State: &State{Lock: true},
|
|
Vars: &Vars{},
|
|
Operation: &Operation{
|
|
PlanMode: plans.DestroyMode,
|
|
Parallelism: 10,
|
|
Refresh: true,
|
|
},
|
|
},
|
|
},
|
|
}
|
|
|
|
cmpOpts := cmpopts.IgnoreUnexported(Operation{}, Vars{}, State{})
|
|
|
|
for name, tc := range testCases {
|
|
t.Run(name, func(t *testing.T) {
|
|
got, diags := ParseApplyDestroy(tc.args)
|
|
if len(diags) > 0 {
|
|
t.Fatalf("unexpected diags: %v", diags)
|
|
}
|
|
if diff := cmp.Diff(tc.want, got, cmpOpts); diff != "" {
|
|
t.Errorf("unexpected result\n%s", diff)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestParseApplyDestroy_invalid(t *testing.T) {
|
|
t.Run("explicit destroy mode", func(t *testing.T) {
|
|
got, diags := ParseApplyDestroy([]string{"-destroy"})
|
|
if len(diags) == 0 {
|
|
t.Fatal("expected diags but got none")
|
|
}
|
|
if got, want := diags.Err().Error(), "Invalid mode option:"; !strings.Contains(got, want) {
|
|
t.Fatalf("wrong diags\n got: %s\nwant: %s", got, want)
|
|
}
|
|
if got.ViewType != ViewHuman {
|
|
t.Fatalf("wrong view type, got %#v, want %#v", got.ViewType, ViewHuman)
|
|
}
|
|
})
|
|
}
|