mirror of https://github.com/docker/cli.git
278 lines
6.4 KiB
Go
278 lines
6.4 KiB
Go
package template
|
|
|
|
import (
|
|
"fmt"
|
|
"reflect"
|
|
"testing"
|
|
|
|
"gotest.tools/assert"
|
|
is "gotest.tools/assert/cmp"
|
|
)
|
|
|
|
var defaults = map[string]string{
|
|
"FOO": "first",
|
|
"BAR": "",
|
|
}
|
|
|
|
func defaultMapping(name string) (string, bool) {
|
|
val, ok := defaults[name]
|
|
return val, ok
|
|
}
|
|
|
|
func TestEscaped(t *testing.T) {
|
|
result, err := Substitute("$${foo}", defaultMapping)
|
|
assert.NilError(t, err)
|
|
assert.Check(t, is.Equal("${foo}", result))
|
|
}
|
|
|
|
func TestSubstituteNoMatch(t *testing.T) {
|
|
result, err := Substitute("foo", defaultMapping)
|
|
assert.NilError(t, err)
|
|
assert.Equal(t, "foo", result)
|
|
}
|
|
|
|
func TestInvalid(t *testing.T) {
|
|
invalidTemplates := []string{
|
|
"${",
|
|
"$}",
|
|
"${}",
|
|
"${ }",
|
|
"${ foo}",
|
|
"${foo }",
|
|
"${foo!}",
|
|
}
|
|
|
|
for _, template := range invalidTemplates {
|
|
_, err := Substitute(template, defaultMapping)
|
|
assert.ErrorContains(t, err, "Invalid template")
|
|
}
|
|
}
|
|
|
|
func TestNoValueNoDefault(t *testing.T) {
|
|
for _, template := range []string{"This ${missing} var", "This ${BAR} var"} {
|
|
result, err := Substitute(template, defaultMapping)
|
|
assert.NilError(t, err)
|
|
assert.Check(t, is.Equal("This var", result))
|
|
}
|
|
}
|
|
|
|
func TestValueNoDefault(t *testing.T) {
|
|
for _, template := range []string{"This $FOO var", "This ${FOO} var"} {
|
|
result, err := Substitute(template, defaultMapping)
|
|
assert.NilError(t, err)
|
|
assert.Check(t, is.Equal("This first var", result))
|
|
}
|
|
}
|
|
|
|
func TestNoValueWithDefault(t *testing.T) {
|
|
for _, template := range []string{"ok ${missing:-def}", "ok ${missing-def}"} {
|
|
result, err := Substitute(template, defaultMapping)
|
|
assert.NilError(t, err)
|
|
assert.Check(t, is.Equal("ok def", result))
|
|
}
|
|
}
|
|
|
|
func TestEmptyValueWithSoftDefault(t *testing.T) {
|
|
result, err := Substitute("ok ${BAR:-def}", defaultMapping)
|
|
assert.NilError(t, err)
|
|
assert.Check(t, is.Equal("ok def", result))
|
|
}
|
|
|
|
func TestEmptyValueWithHardDefault(t *testing.T) {
|
|
result, err := Substitute("ok ${BAR-def}", defaultMapping)
|
|
assert.NilError(t, err)
|
|
assert.Check(t, is.Equal("ok ", result))
|
|
}
|
|
|
|
func TestNonAlphanumericDefault(t *testing.T) {
|
|
result, err := Substitute("ok ${BAR:-/non:-alphanumeric}", defaultMapping)
|
|
assert.NilError(t, err)
|
|
assert.Check(t, is.Equal("ok /non:-alphanumeric", result))
|
|
}
|
|
|
|
func TestMandatoryVariableErrors(t *testing.T) {
|
|
testCases := []struct {
|
|
template string
|
|
expectedError string
|
|
}{
|
|
{
|
|
template: "not ok ${UNSET_VAR:?Mandatory Variable Unset}",
|
|
expectedError: "required variable UNSET_VAR is missing a value: Mandatory Variable Unset",
|
|
},
|
|
{
|
|
template: "not ok ${BAR:?Mandatory Variable Empty}",
|
|
expectedError: "required variable BAR is missing a value: Mandatory Variable Empty",
|
|
},
|
|
{
|
|
template: "not ok ${UNSET_VAR:?}",
|
|
expectedError: "required variable UNSET_VAR is missing a value",
|
|
},
|
|
{
|
|
template: "not ok ${UNSET_VAR?Mandatory Variable Unset}",
|
|
expectedError: "required variable UNSET_VAR is missing a value: Mandatory Variable Unset",
|
|
},
|
|
{
|
|
template: "not ok ${UNSET_VAR?}",
|
|
expectedError: "required variable UNSET_VAR is missing a value",
|
|
},
|
|
}
|
|
|
|
for _, tc := range testCases {
|
|
_, err := Substitute(tc.template, defaultMapping)
|
|
assert.ErrorContains(t, err, tc.expectedError)
|
|
assert.ErrorType(t, err, reflect.TypeOf(&InvalidTemplateError{}))
|
|
}
|
|
}
|
|
|
|
func TestDefaultsForMandatoryVariables(t *testing.T) {
|
|
testCases := []struct {
|
|
template string
|
|
expected string
|
|
}{
|
|
{
|
|
template: "ok ${FOO:?err}",
|
|
expected: "ok first",
|
|
},
|
|
{
|
|
template: "ok ${FOO?err}",
|
|
expected: "ok first",
|
|
},
|
|
{
|
|
template: "ok ${BAR?err}",
|
|
expected: "ok ",
|
|
},
|
|
}
|
|
|
|
for _, tc := range testCases {
|
|
result, err := Substitute(tc.template, defaultMapping)
|
|
assert.NilError(t, err)
|
|
assert.Check(t, is.Equal(tc.expected, result))
|
|
}
|
|
}
|
|
|
|
func TestSubstituteWithCustomFunc(t *testing.T) {
|
|
errIsMissing := func(substitution string, mapping Mapping) (string, bool, error) {
|
|
value, found := mapping(substitution)
|
|
if !found {
|
|
return "", true, &InvalidTemplateError{
|
|
Template: fmt.Sprintf("required variable %s is missing a value", substitution),
|
|
}
|
|
}
|
|
return value, true, nil
|
|
}
|
|
|
|
result, err := SubstituteWith("ok ${FOO}", defaultMapping, defaultPattern, errIsMissing)
|
|
assert.NilError(t, err)
|
|
assert.Check(t, is.Equal("ok first", result))
|
|
|
|
result, err = SubstituteWith("ok ${BAR}", defaultMapping, defaultPattern, errIsMissing)
|
|
assert.NilError(t, err)
|
|
assert.Check(t, is.Equal("ok ", result))
|
|
|
|
_, err = SubstituteWith("ok ${NOTHERE}", defaultMapping, defaultPattern, errIsMissing)
|
|
assert.Check(t, is.ErrorContains(err, "required variable"))
|
|
}
|
|
|
|
func TestExtractVariables(t *testing.T) {
|
|
testCases := []struct {
|
|
name string
|
|
dict map[string]interface{}
|
|
expected map[string]string
|
|
}{
|
|
{
|
|
name: "empty",
|
|
dict: map[string]interface{}{},
|
|
expected: map[string]string{},
|
|
},
|
|
{
|
|
name: "no-variables",
|
|
dict: map[string]interface{}{
|
|
"foo": "bar",
|
|
},
|
|
expected: map[string]string{},
|
|
},
|
|
{
|
|
name: "variable-without-curly-braces",
|
|
dict: map[string]interface{}{
|
|
"foo": "$bar",
|
|
},
|
|
expected: map[string]string{
|
|
"bar": "",
|
|
},
|
|
},
|
|
{
|
|
name: "variable",
|
|
dict: map[string]interface{}{
|
|
"foo": "${bar}",
|
|
},
|
|
expected: map[string]string{
|
|
"bar": "",
|
|
},
|
|
},
|
|
{
|
|
name: "required-variable",
|
|
dict: map[string]interface{}{
|
|
"foo": "${bar?:foo}",
|
|
},
|
|
expected: map[string]string{
|
|
"bar": "",
|
|
},
|
|
},
|
|
{
|
|
name: "required-variable2",
|
|
dict: map[string]interface{}{
|
|
"foo": "${bar?foo}",
|
|
},
|
|
expected: map[string]string{
|
|
"bar": "",
|
|
},
|
|
},
|
|
{
|
|
name: "default-variable",
|
|
dict: map[string]interface{}{
|
|
"foo": "${bar:-foo}",
|
|
},
|
|
expected: map[string]string{
|
|
"bar": "foo",
|
|
},
|
|
},
|
|
{
|
|
name: "default-variable2",
|
|
dict: map[string]interface{}{
|
|
"foo": "${bar-foo}",
|
|
},
|
|
expected: map[string]string{
|
|
"bar": "foo",
|
|
},
|
|
},
|
|
{
|
|
name: "multiple-values",
|
|
dict: map[string]interface{}{
|
|
"foo": "${bar:-foo}",
|
|
"bar": map[string]interface{}{
|
|
"foo": "${fruit:-banana}",
|
|
"bar": "vegetable",
|
|
},
|
|
"baz": []interface{}{
|
|
"foo",
|
|
"$docker:${project:-cli}",
|
|
"$toto",
|
|
},
|
|
},
|
|
expected: map[string]string{
|
|
"bar": "foo",
|
|
"fruit": "banana",
|
|
"toto": "",
|
|
"docker": "",
|
|
"project": "cli",
|
|
},
|
|
},
|
|
}
|
|
for _, tc := range testCases {
|
|
t.Run(tc.name, func(t *testing.T) {
|
|
actual := ExtractVariables(tc.dict, defaultPattern)
|
|
assert.Check(t, is.DeepEqual(actual, tc.expected))
|
|
})
|
|
}
|
|
}
|