mirror of
https://github.com/opentofu/opentofu.git
synced 2025-01-27 17:06:27 -06:00
df8529719c
This augments backend-config to also accept key=value pairs. This should make Terraform easier to script rather than having to generate a JSON file. You must still specify the backend type as a minimal amount in configurations, example: ``` terraform { backend "consul" {} } ``` This is required because Terraform needs to be able to detect the _absense_ of that value for unsetting, if that is necessary at some point.
300 lines
4.6 KiB
Go
300 lines
4.6 KiB
Go
package variables
|
|
|
|
import (
|
|
"flag"
|
|
"fmt"
|
|
"io/ioutil"
|
|
"reflect"
|
|
"testing"
|
|
|
|
"github.com/davecgh/go-spew/spew"
|
|
)
|
|
|
|
func TestFlagAny_impl(t *testing.T) {
|
|
var _ flag.Value = new(FlagAny)
|
|
}
|
|
|
|
func TestFlagAny(t *testing.T) {
|
|
cases := []struct {
|
|
Input interface{}
|
|
Output map[string]interface{}
|
|
Error bool
|
|
}{
|
|
{
|
|
"=value",
|
|
nil,
|
|
true,
|
|
},
|
|
|
|
{
|
|
" =value",
|
|
nil,
|
|
true,
|
|
},
|
|
|
|
{
|
|
"key=value",
|
|
map[string]interface{}{"key": "value"},
|
|
false,
|
|
},
|
|
|
|
{
|
|
"key=",
|
|
map[string]interface{}{"key": ""},
|
|
false,
|
|
},
|
|
|
|
{
|
|
"key=foo=bar",
|
|
map[string]interface{}{"key": "foo=bar"},
|
|
false,
|
|
},
|
|
|
|
{
|
|
"key=false",
|
|
map[string]interface{}{"key": "false"},
|
|
false,
|
|
},
|
|
|
|
{
|
|
"key =value",
|
|
map[string]interface{}{"key": "value"},
|
|
false,
|
|
},
|
|
|
|
{
|
|
"key = value",
|
|
map[string]interface{}{"key": " value"},
|
|
false,
|
|
},
|
|
|
|
{
|
|
`key = "value"`,
|
|
map[string]interface{}{"key": "value"},
|
|
false,
|
|
},
|
|
|
|
{
|
|
"map.key=foo",
|
|
map[string]interface{}{"map.key": "foo"},
|
|
false,
|
|
},
|
|
|
|
{
|
|
"key",
|
|
nil,
|
|
true,
|
|
},
|
|
|
|
{
|
|
`key=["hello", "world"]`,
|
|
map[string]interface{}{"key": []interface{}{"hello", "world"}},
|
|
false,
|
|
},
|
|
|
|
{
|
|
`key={"hello" = "world", "foo" = "bar"}`,
|
|
map[string]interface{}{
|
|
"key": map[string]interface{}{
|
|
"hello": "world",
|
|
"foo": "bar",
|
|
},
|
|
},
|
|
false,
|
|
},
|
|
|
|
{
|
|
`key={"hello" = "world", "foo" = "bar"}\nkey2="invalid"`,
|
|
nil,
|
|
true,
|
|
},
|
|
|
|
{
|
|
"key=/path",
|
|
map[string]interface{}{"key": "/path"},
|
|
false,
|
|
},
|
|
|
|
{
|
|
"key=1234.dkr.ecr.us-east-1.amazonaws.com/proj:abcdef",
|
|
map[string]interface{}{"key": "1234.dkr.ecr.us-east-1.amazonaws.com/proj:abcdef"},
|
|
false,
|
|
},
|
|
|
|
// simple values that can parse as numbers should remain strings
|
|
{
|
|
"key=1",
|
|
map[string]interface{}{
|
|
"key": "1",
|
|
},
|
|
false,
|
|
},
|
|
{
|
|
"key=1.0",
|
|
map[string]interface{}{
|
|
"key": "1.0",
|
|
},
|
|
false,
|
|
},
|
|
{
|
|
"key=0x10",
|
|
map[string]interface{}{
|
|
"key": "0x10",
|
|
},
|
|
false,
|
|
},
|
|
|
|
// Test setting multiple times
|
|
{
|
|
[]string{
|
|
"foo=bar",
|
|
"bar=baz",
|
|
},
|
|
map[string]interface{}{
|
|
"foo": "bar",
|
|
"bar": "baz",
|
|
},
|
|
false,
|
|
},
|
|
|
|
// Test map merging
|
|
{
|
|
[]string{
|
|
`foo={ foo = "bar" }`,
|
|
`foo={ bar = "baz" }`,
|
|
},
|
|
map[string]interface{}{
|
|
"foo": map[string]interface{}{
|
|
"foo": "bar",
|
|
"bar": "baz",
|
|
},
|
|
},
|
|
false,
|
|
},
|
|
}
|
|
|
|
for i, tc := range cases {
|
|
t.Run(fmt.Sprintf("%d-%s", i, tc.Input), func(t *testing.T) {
|
|
var input []string
|
|
switch v := tc.Input.(type) {
|
|
case string:
|
|
input = []string{v}
|
|
case []string:
|
|
input = v
|
|
default:
|
|
t.Fatalf("bad input type: %T", tc.Input)
|
|
}
|
|
|
|
f := new(FlagAny)
|
|
for i, single := range input {
|
|
err := f.Set(single)
|
|
|
|
// Only check for expected errors on the final input
|
|
expected := tc.Error && i == len(input)-1
|
|
if err != nil != expected {
|
|
t.Fatalf("bad error. Input: %#v\n\nError: %s", single, err)
|
|
}
|
|
}
|
|
|
|
actual := map[string]interface{}(*f)
|
|
if !reflect.DeepEqual(actual, tc.Output) {
|
|
t.Fatalf("bad:\nexpected: %s\n\n got: %s\n", spew.Sdump(tc.Output), spew.Sdump(actual))
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestFlagAny_file(t *testing.T) {
|
|
inputLibucl := `
|
|
foo = "bar"
|
|
`
|
|
inputMap := `
|
|
foo = {
|
|
k = "v"
|
|
}`
|
|
|
|
inputJson := `{
|
|
"foo": "bar"}`
|
|
|
|
cases := []struct {
|
|
Input interface{}
|
|
Output map[string]interface{}
|
|
Error bool
|
|
}{
|
|
{
|
|
inputLibucl,
|
|
map[string]interface{}{"foo": "bar"},
|
|
false,
|
|
},
|
|
|
|
{
|
|
inputJson,
|
|
map[string]interface{}{"foo": "bar"},
|
|
false,
|
|
},
|
|
|
|
{
|
|
`map.key = "foo"`,
|
|
map[string]interface{}{"map.key": "foo"},
|
|
false,
|
|
},
|
|
|
|
{
|
|
inputMap,
|
|
map[string]interface{}{
|
|
"foo": map[string]interface{}{
|
|
"k": "v",
|
|
},
|
|
},
|
|
false,
|
|
},
|
|
|
|
{
|
|
[]string{
|
|
`foo = { "k" = "v"}`,
|
|
`foo = { "j" = "v" }`,
|
|
},
|
|
map[string]interface{}{
|
|
"foo": map[string]interface{}{
|
|
"k": "v",
|
|
"j": "v",
|
|
},
|
|
},
|
|
false,
|
|
},
|
|
}
|
|
|
|
path := testTempFile(t)
|
|
|
|
for i, tc := range cases {
|
|
t.Run(fmt.Sprintf("%d", i), func(t *testing.T) {
|
|
var input []string
|
|
switch i := tc.Input.(type) {
|
|
case string:
|
|
input = []string{i}
|
|
case []string:
|
|
input = i
|
|
default:
|
|
t.Fatalf("bad input type: %T", i)
|
|
}
|
|
|
|
f := new(FlagAny)
|
|
for _, input := range input {
|
|
if err := ioutil.WriteFile(path, []byte(input), 0644); err != nil {
|
|
t.Fatalf("err: %s", err)
|
|
}
|
|
|
|
err := f.Set(path)
|
|
if err != nil != tc.Error {
|
|
t.Fatalf("bad error. Input: %#v, err: %s", input, err)
|
|
}
|
|
}
|
|
|
|
actual := map[string]interface{}(*f)
|
|
if !reflect.DeepEqual(actual, tc.Output) {
|
|
t.Fatalf("bad: %#v", actual)
|
|
}
|
|
})
|
|
}
|
|
}
|