2023-05-02 10:33:06 -05:00
|
|
|
// Copyright (c) HashiCorp, Inc.
|
|
|
|
// SPDX-License-Identifier: MPL-2.0
|
|
|
|
|
2023-08-27 09:51:49 -05:00
|
|
|
package opentf
|
2016-08-17 13:28:58 -05:00
|
|
|
|
|
|
|
import (
|
|
|
|
"testing"
|
2018-05-04 21:24:06 -05:00
|
|
|
|
2023-09-20 06:35:35 -05:00
|
|
|
"github.com/opentofu/opentofu/internal/configs"
|
2018-05-04 21:24:06 -05:00
|
|
|
"github.com/zclconf/go-cty/cty"
|
2016-08-17 13:28:58 -05:00
|
|
|
)
|
|
|
|
|
2020-05-18 13:10:19 -05:00
|
|
|
func TestCheckInputVariables(t *testing.T) {
|
|
|
|
c := testModule(t, "input-variables")
|
2018-11-20 14:17:57 -06:00
|
|
|
|
2020-05-18 13:10:19 -05:00
|
|
|
t.Run("No variables set", func(t *testing.T) {
|
|
|
|
// No variables set
|
|
|
|
diags := checkInputVariables(c.Module.Variables, nil)
|
|
|
|
if !diags.HasErrors() {
|
|
|
|
t.Fatal("check succeeded, but want errors")
|
|
|
|
}
|
2018-11-20 14:17:57 -06:00
|
|
|
|
2020-05-18 13:10:19 -05:00
|
|
|
// Required variables set, optional variables unset
|
|
|
|
// This is still an error at this layer, since it's the caller's
|
|
|
|
// responsibility to have already merged in any default values.
|
|
|
|
diags = checkInputVariables(c.Module.Variables, InputValues{
|
|
|
|
"foo": &InputValue{
|
|
|
|
Value: cty.StringVal("bar"),
|
|
|
|
SourceType: ValueFromCLIArg,
|
|
|
|
},
|
|
|
|
})
|
|
|
|
if !diags.HasErrors() {
|
|
|
|
t.Fatal("check succeeded, but want errors")
|
|
|
|
}
|
2018-11-20 14:17:57 -06:00
|
|
|
})
|
|
|
|
|
2020-05-18 13:10:19 -05:00
|
|
|
t.Run("All variables set", func(t *testing.T) {
|
|
|
|
diags := checkInputVariables(c.Module.Variables, InputValues{
|
|
|
|
"foo": &InputValue{
|
|
|
|
Value: cty.StringVal("bar"),
|
|
|
|
SourceType: ValueFromCLIArg,
|
|
|
|
},
|
|
|
|
"bar": &InputValue{
|
|
|
|
Value: cty.StringVal("baz"),
|
|
|
|
SourceType: ValueFromCLIArg,
|
|
|
|
},
|
|
|
|
"map": &InputValue{
|
|
|
|
Value: cty.StringVal("baz"), // okay because config has no type constraint
|
|
|
|
SourceType: ValueFromCLIArg,
|
|
|
|
},
|
|
|
|
"object_map": &InputValue{
|
|
|
|
Value: cty.MapVal(map[string]cty.Value{
|
|
|
|
"uno": cty.ObjectVal(map[string]cty.Value{
|
|
|
|
"foo": cty.StringVal("baz"),
|
|
|
|
"bar": cty.NumberIntVal(2), // type = any
|
|
|
|
}),
|
|
|
|
"dos": cty.ObjectVal(map[string]cty.Value{
|
|
|
|
"foo": cty.StringVal("bat"),
|
|
|
|
"bar": cty.NumberIntVal(99), // type = any
|
|
|
|
}),
|
|
|
|
}),
|
|
|
|
SourceType: ValueFromCLIArg,
|
|
|
|
},
|
|
|
|
"object_list": &InputValue{
|
|
|
|
Value: cty.ListVal([]cty.Value{
|
|
|
|
cty.ObjectVal(map[string]cty.Value{
|
|
|
|
"foo": cty.StringVal("baz"),
|
|
|
|
"bar": cty.NumberIntVal(2), // type = any
|
|
|
|
}),
|
|
|
|
cty.ObjectVal(map[string]cty.Value{
|
|
|
|
"foo": cty.StringVal("bang"),
|
|
|
|
"bar": cty.NumberIntVal(42), // type = any
|
|
|
|
}),
|
|
|
|
}),
|
|
|
|
SourceType: ValueFromCLIArg,
|
|
|
|
},
|
|
|
|
})
|
|
|
|
if diags.HasErrors() {
|
|
|
|
t.Fatalf("unexpected errors: %s", diags.Err())
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("Invalid Complex Types", func(t *testing.T) {
|
|
|
|
diags := checkInputVariables(c.Module.Variables, InputValues{
|
|
|
|
"foo": &InputValue{
|
|
|
|
Value: cty.StringVal("bar"),
|
|
|
|
SourceType: ValueFromCLIArg,
|
|
|
|
},
|
|
|
|
"bar": &InputValue{
|
|
|
|
Value: cty.StringVal("baz"),
|
|
|
|
SourceType: ValueFromCLIArg,
|
|
|
|
},
|
|
|
|
"map": &InputValue{
|
|
|
|
Value: cty.StringVal("baz"), // okay because config has no type constraint
|
|
|
|
SourceType: ValueFromCLIArg,
|
|
|
|
},
|
|
|
|
"object_map": &InputValue{
|
|
|
|
Value: cty.MapVal(map[string]cty.Value{
|
|
|
|
"uno": cty.ObjectVal(map[string]cty.Value{
|
|
|
|
"foo": cty.StringVal("baz"),
|
|
|
|
"bar": cty.NumberIntVal(2), // type = any
|
|
|
|
}),
|
|
|
|
"dos": cty.ObjectVal(map[string]cty.Value{
|
|
|
|
"foo": cty.StringVal("bat"),
|
|
|
|
"bar": cty.NumberIntVal(99), // type = any
|
|
|
|
}),
|
|
|
|
}),
|
|
|
|
SourceType: ValueFromCLIArg,
|
|
|
|
},
|
|
|
|
"object_list": &InputValue{
|
|
|
|
Value: cty.TupleVal([]cty.Value{
|
|
|
|
cty.ObjectVal(map[string]cty.Value{
|
|
|
|
"foo": cty.StringVal("baz"),
|
|
|
|
"bar": cty.NumberIntVal(2), // type = any
|
|
|
|
}),
|
|
|
|
cty.ObjectVal(map[string]cty.Value{
|
|
|
|
"foo": cty.StringVal("bang"),
|
|
|
|
"bar": cty.StringVal("42"), // type = any, but mismatch with the first list item
|
|
|
|
}),
|
|
|
|
}),
|
|
|
|
SourceType: ValueFromCLIArg,
|
|
|
|
},
|
|
|
|
})
|
|
|
|
|
|
|
|
if diags.HasErrors() {
|
|
|
|
t.Fatalf("unexpected errors: %s", diags.Err())
|
|
|
|
}
|
2018-11-20 14:17:57 -06:00
|
|
|
})
|
|
|
|
}
|
2021-12-20 18:38:52 -06:00
|
|
|
|
|
|
|
// testInputValuesUnset is a helper for constructing InputValues values for
|
|
|
|
// situations where all of the root module variables are optional and a
|
|
|
|
// test case intends to just use those default values and not override them
|
|
|
|
// at all.
|
|
|
|
//
|
|
|
|
// In other words, this constructs an InputValues with one entry per given
|
|
|
|
// input variable declaration where all of them are declared as unset.
|
|
|
|
func testInputValuesUnset(decls map[string]*configs.Variable) InputValues {
|
|
|
|
if len(decls) == 0 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
ret := make(InputValues, len(decls))
|
|
|
|
for name := range decls {
|
|
|
|
ret[name] = &InputValue{
|
|
|
|
Value: cty.NilVal,
|
|
|
|
SourceType: ValueFromUnknown,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ret
|
|
|
|
}
|