mirror of https://github.com/databricks/cli.git
606 lines
16 KiB
Go
606 lines
16 KiB
Go
package template
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"os"
|
|
"path"
|
|
"path/filepath"
|
|
"testing"
|
|
"text/template"
|
|
|
|
"github.com/databricks/cli/libs/jsonschema"
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
func TestTemplateConfigAssignValuesFromFile(t *testing.T) {
|
|
testDir := "./testdata/config-assign-from-file"
|
|
|
|
ctx := context.Background()
|
|
c, err := newConfig(ctx, os.DirFS(testDir), "schema.json")
|
|
require.NoError(t, err)
|
|
|
|
err = c.assignValuesFromFile(filepath.Join(testDir, "config.json"))
|
|
if assert.NoError(t, err) {
|
|
assert.Equal(t, int64(1), c.values["int_val"])
|
|
assert.InDelta(t, float64(2), c.values["float_val"].(float64), 0.0001)
|
|
assert.Equal(t, true, c.values["bool_val"])
|
|
assert.Equal(t, "hello", c.values["string_val"])
|
|
}
|
|
}
|
|
|
|
func TestTemplateConfigAssignValuesFromFileDoesNotOverwriteExistingConfigs(t *testing.T) {
|
|
testDir := "./testdata/config-assign-from-file"
|
|
|
|
ctx := context.Background()
|
|
c, err := newConfig(ctx, os.DirFS(testDir), "schema.json")
|
|
require.NoError(t, err)
|
|
|
|
c.values = map[string]any{
|
|
"string_val": "this-is-not-overwritten",
|
|
}
|
|
|
|
err = c.assignValuesFromFile(filepath.Join(testDir, "config.json"))
|
|
if assert.NoError(t, err) {
|
|
assert.Equal(t, int64(1), c.values["int_val"])
|
|
assert.InDelta(t, float64(2), c.values["float_val"].(float64), 0.0001)
|
|
assert.Equal(t, true, c.values["bool_val"])
|
|
assert.Equal(t, "this-is-not-overwritten", c.values["string_val"])
|
|
}
|
|
}
|
|
|
|
func TestTemplateConfigAssignValuesFromFileForInvalidIntegerValue(t *testing.T) {
|
|
testDir := "./testdata/config-assign-from-file-invalid-int"
|
|
|
|
ctx := context.Background()
|
|
c, err := newConfig(ctx, os.DirFS(testDir), "schema.json")
|
|
require.NoError(t, err)
|
|
|
|
err = c.assignValuesFromFile(filepath.Join(testDir, "config.json"))
|
|
assert.EqualError(t, err, fmt.Sprintf("failed to load config from file %s: failed to parse property int_val: cannot convert \"abc\" to an integer", filepath.Join(testDir, "config.json")))
|
|
}
|
|
|
|
func TestTemplateConfigAssignValuesFromFileFiltersPropertiesNotInTheSchema(t *testing.T) {
|
|
testDir := "./testdata/config-assign-from-file-unknown-property"
|
|
|
|
ctx := context.Background()
|
|
c, err := newConfig(ctx, os.DirFS(testDir), "schema.json")
|
|
require.NoError(t, err)
|
|
|
|
err = c.assignValuesFromFile(filepath.Join(testDir, "config.json"))
|
|
assert.NoError(t, err)
|
|
|
|
// assert only the known property is loaded
|
|
assert.Len(t, c.values, 1)
|
|
assert.Equal(t, "i am a known property", c.values["string_val"])
|
|
}
|
|
|
|
func TestTemplateConfigAssignValuesFromDefaultValues(t *testing.T) {
|
|
testDir := "./testdata/config-assign-from-default-value"
|
|
|
|
ctx := context.Background()
|
|
c, err := newConfig(ctx, os.DirFS(testDir), "schema.json")
|
|
require.NoError(t, err)
|
|
|
|
r, err := newRenderer(ctx, nil, nil, os.DirFS("."), "./testdata/empty/template", "./testdata/empty/library")
|
|
require.NoError(t, err)
|
|
|
|
err = c.assignDefaultValues(r)
|
|
if assert.NoError(t, err) {
|
|
assert.Equal(t, int64(123), c.values["int_val"])
|
|
assert.InDelta(t, float64(123), c.values["float_val"].(float64), 0.0001)
|
|
assert.Equal(t, true, c.values["bool_val"])
|
|
assert.Equal(t, "hello", c.values["string_val"])
|
|
}
|
|
}
|
|
|
|
func TestTemplateConfigAssignValuesFromTemplatedDefaultValues(t *testing.T) {
|
|
testDir := "./testdata/config-assign-from-templated-default-value"
|
|
|
|
ctx := context.Background()
|
|
c, err := newConfig(ctx, os.DirFS(testDir), "schema.json")
|
|
require.NoError(t, err)
|
|
|
|
r, err := newRenderer(ctx, nil, nil, os.DirFS("."), path.Join(testDir, "template/template"), path.Join(testDir, "template/library"))
|
|
require.NoError(t, err)
|
|
|
|
// Note: only the string value is templated.
|
|
// The JSON schema package doesn't allow using a string default for integer types.
|
|
err = c.assignDefaultValues(r)
|
|
if assert.NoError(t, err) {
|
|
assert.Equal(t, int64(123), c.values["int_val"])
|
|
assert.InDelta(t, float64(123), c.values["float_val"].(float64), 0.0001)
|
|
assert.Equal(t, true, c.values["bool_val"])
|
|
assert.Equal(t, "world", c.values["string_val"])
|
|
}
|
|
}
|
|
|
|
func TestTemplateConfigValidateValuesDefined(t *testing.T) {
|
|
ctx := context.Background()
|
|
c, err := newConfig(ctx, os.DirFS("testdata/config-test-schema"), "test-schema.json")
|
|
require.NoError(t, err)
|
|
|
|
c.values = map[string]any{
|
|
"int_val": 1,
|
|
"float_val": 1.0,
|
|
"bool_val": false,
|
|
}
|
|
|
|
err = c.validate()
|
|
assert.EqualError(t, err, "validation for template input parameters failed. no value provided for required property string_val")
|
|
}
|
|
|
|
func TestTemplateConfigValidateTypeForValidConfig(t *testing.T) {
|
|
ctx := context.Background()
|
|
c, err := newConfig(ctx, os.DirFS("testdata/config-test-schema"), "test-schema.json")
|
|
require.NoError(t, err)
|
|
|
|
c.values = map[string]any{
|
|
"int_val": 1,
|
|
"float_val": 1.1,
|
|
"bool_val": true,
|
|
"string_val": "abcd",
|
|
}
|
|
|
|
err = c.validate()
|
|
assert.NoError(t, err)
|
|
}
|
|
|
|
func TestTemplateConfigValidateTypeForUnknownField(t *testing.T) {
|
|
ctx := context.Background()
|
|
c, err := newConfig(ctx, os.DirFS("testdata/config-test-schema"), "test-schema.json")
|
|
require.NoError(t, err)
|
|
|
|
c.values = map[string]any{
|
|
"unknown_prop": 1,
|
|
"int_val": 1,
|
|
"float_val": 1.1,
|
|
"bool_val": true,
|
|
"string_val": "abcd",
|
|
}
|
|
|
|
err = c.validate()
|
|
assert.EqualError(t, err, "validation for template input parameters failed. property unknown_prop is not defined in the schema")
|
|
}
|
|
|
|
func TestTemplateConfigValidateTypeForInvalidType(t *testing.T) {
|
|
ctx := context.Background()
|
|
c, err := newConfig(ctx, os.DirFS("testdata/config-test-schema"), "test-schema.json")
|
|
require.NoError(t, err)
|
|
|
|
c.values = map[string]any{
|
|
"int_val": "this-should-be-an-int",
|
|
"float_val": 1.1,
|
|
"bool_val": true,
|
|
"string_val": "abcd",
|
|
}
|
|
|
|
err = c.validate()
|
|
assert.EqualError(t, err, "validation for template input parameters failed. incorrect type for property int_val: expected type integer, but value is \"this-should-be-an-int\"")
|
|
}
|
|
|
|
func TestTemplateValidateSchema(t *testing.T) {
|
|
var err error
|
|
toSchema := func(s string) *jsonschema.Schema {
|
|
return &jsonschema.Schema{
|
|
Properties: map[string]*jsonschema.Schema{
|
|
"foo": {
|
|
Type: jsonschema.Type(s),
|
|
},
|
|
},
|
|
}
|
|
}
|
|
|
|
err = validateSchema(toSchema("string"))
|
|
assert.NoError(t, err)
|
|
|
|
err = validateSchema(toSchema("boolean"))
|
|
assert.NoError(t, err)
|
|
|
|
err = validateSchema(toSchema("number"))
|
|
assert.NoError(t, err)
|
|
|
|
err = validateSchema(toSchema("integer"))
|
|
assert.NoError(t, err)
|
|
|
|
err = validateSchema(toSchema("object"))
|
|
assert.EqualError(t, err, "property type object is not supported by bundle templates")
|
|
|
|
err = validateSchema(toSchema("array"))
|
|
assert.EqualError(t, err, "property type array is not supported by bundle templates")
|
|
}
|
|
|
|
func TestTemplateValidateSchemaVersion(t *testing.T) {
|
|
version := latestSchemaVersion
|
|
schema := jsonschema.Schema{
|
|
Extension: jsonschema.Extension{
|
|
Version: &version,
|
|
},
|
|
}
|
|
assert.NoError(t, validateSchema(&schema))
|
|
|
|
version = latestSchemaVersion + 1
|
|
schema = jsonschema.Schema{
|
|
Extension: jsonschema.Extension{
|
|
Version: &version,
|
|
},
|
|
}
|
|
assert.EqualError(t, validateSchema(&schema), fmt.Sprintf("template schema version %d is not supported by this version of the CLI. Please upgrade your CLI to the latest version", version))
|
|
|
|
version = 5000
|
|
schema = jsonschema.Schema{
|
|
Extension: jsonschema.Extension{
|
|
Version: &version,
|
|
},
|
|
}
|
|
assert.EqualError(t, validateSchema(&schema), "template schema version 5000 is not supported by this version of the CLI. Please upgrade your CLI to the latest version")
|
|
|
|
version = 0
|
|
schema = jsonschema.Schema{
|
|
Extension: jsonschema.Extension{
|
|
Version: &version,
|
|
},
|
|
}
|
|
assert.NoError(t, validateSchema(&schema))
|
|
}
|
|
|
|
func TestTemplateEnumValidation(t *testing.T) {
|
|
schema := jsonschema.Schema{
|
|
Properties: map[string]*jsonschema.Schema{
|
|
"abc": {
|
|
Type: "integer",
|
|
Enum: []any{1, 2, 3, 4},
|
|
},
|
|
},
|
|
}
|
|
|
|
c := &config{
|
|
schema: &schema,
|
|
values: map[string]any{
|
|
"abc": 5,
|
|
},
|
|
}
|
|
assert.EqualError(t, c.validate(), "validation for template input parameters failed. expected value of property abc to be one of [1 2 3 4]. Found: 5")
|
|
|
|
c = &config{
|
|
schema: &schema,
|
|
values: map[string]any{
|
|
"abc": 4,
|
|
},
|
|
}
|
|
assert.NoError(t, c.validate())
|
|
}
|
|
|
|
func TestTemplateSchemaErrorsWithEmptyDescription(t *testing.T) {
|
|
ctx := context.Background()
|
|
_, err := newConfig(ctx, os.DirFS("./testdata/config-test-schema"), "invalid-test-schema.json")
|
|
assert.EqualError(t, err, "template property property-without-description is missing a description")
|
|
}
|
|
|
|
func testRenderer() *renderer {
|
|
return &renderer{
|
|
config: map[string]any{
|
|
"fruit": "apples",
|
|
},
|
|
baseTemplate: template.New(""),
|
|
}
|
|
}
|
|
|
|
func TestPromptIsSkippedWhenEmpty(t *testing.T) {
|
|
c := config{
|
|
ctx: context.Background(),
|
|
values: make(map[string]any),
|
|
schema: &jsonschema.Schema{
|
|
Properties: map[string]*jsonschema.Schema{
|
|
"always-skip": {
|
|
Type: "string",
|
|
Default: "I like {{.fruit}}",
|
|
Extension: jsonschema.Extension{
|
|
SkipPromptIf: &jsonschema.Schema{},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
}
|
|
|
|
// We should always skip the prompt here. An empty JSON schema by definition
|
|
// matches all possible configurations.
|
|
skip, err := c.skipPrompt(jsonschema.Property{
|
|
Name: "always-skip",
|
|
Schema: c.schema.Properties["always-skip"],
|
|
}, testRenderer())
|
|
assert.NoError(t, err)
|
|
assert.True(t, skip)
|
|
assert.Equal(t, "I like apples", c.values["always-skip"])
|
|
}
|
|
|
|
func TestPromptSkipErrorsWithEmptyDefault(t *testing.T) {
|
|
c := config{
|
|
ctx: context.Background(),
|
|
values: make(map[string]any),
|
|
schema: &jsonschema.Schema{
|
|
Properties: map[string]*jsonschema.Schema{
|
|
"no-default": {
|
|
Type: "string",
|
|
Extension: jsonschema.Extension{
|
|
SkipPromptIf: &jsonschema.Schema{},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
}
|
|
|
|
_, err := c.skipPrompt(jsonschema.Property{
|
|
Name: "no-default",
|
|
Schema: c.schema.Properties["no-default"],
|
|
}, testRenderer())
|
|
assert.EqualError(t, err, "property no-default has skip_prompt_if set but no default value")
|
|
}
|
|
|
|
func TestPromptIsSkippedIfValueIsAssigned(t *testing.T) {
|
|
c := config{
|
|
ctx: context.Background(),
|
|
values: make(map[string]any),
|
|
schema: &jsonschema.Schema{
|
|
Properties: map[string]*jsonschema.Schema{
|
|
"already-assigned": {
|
|
Type: "string",
|
|
Default: "some-default-value",
|
|
},
|
|
},
|
|
},
|
|
}
|
|
|
|
c.values["already-assigned"] = "some-value"
|
|
skip, err := c.skipPrompt(jsonschema.Property{
|
|
Name: "already-assigned",
|
|
Schema: c.schema.Properties["already-assigned"],
|
|
}, testRenderer())
|
|
assert.NoError(t, err)
|
|
assert.True(t, skip)
|
|
assert.Equal(t, "some-value", c.values["already-assigned"])
|
|
}
|
|
|
|
func TestPromptIsSkipped(t *testing.T) {
|
|
c := config{
|
|
ctx: context.Background(),
|
|
values: make(map[string]any),
|
|
schema: &jsonschema.Schema{
|
|
Properties: map[string]*jsonschema.Schema{
|
|
"abc": {
|
|
Type: "string",
|
|
},
|
|
"def": {
|
|
Type: "integer",
|
|
},
|
|
"xyz": {
|
|
Type: "string",
|
|
Default: "hello-world",
|
|
Extension: jsonschema.Extension{
|
|
SkipPromptIf: &jsonschema.Schema{
|
|
Properties: map[string]*jsonschema.Schema{
|
|
"abc": {
|
|
Const: "foobar",
|
|
},
|
|
"def": {
|
|
Const: 123,
|
|
},
|
|
},
|
|
Required: []string{"abc", "def"},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
}
|
|
|
|
// No skip condition defined. Prompt should not be skipped.
|
|
skip, err := c.skipPrompt(jsonschema.Property{
|
|
Name: "abc",
|
|
Schema: c.schema.Properties["abc"],
|
|
}, testRenderer())
|
|
assert.NoError(t, err)
|
|
assert.False(t, skip)
|
|
|
|
// No values assigned to config. Prompt should not be skipped.
|
|
skip, err = c.skipPrompt(jsonschema.Property{
|
|
Name: "xyz",
|
|
Schema: c.schema.Properties["xyz"],
|
|
}, testRenderer())
|
|
assert.NoError(t, err)
|
|
assert.False(t, skip)
|
|
assert.NotContains(t, c.values, "xyz")
|
|
|
|
// Values do not match skip condition. Prompt should not be skipped.
|
|
c.values["abc"] = "foo"
|
|
c.values["def"] = 123
|
|
skip, err = c.skipPrompt(jsonschema.Property{
|
|
Name: "xyz",
|
|
Schema: c.schema.Properties["xyz"],
|
|
}, testRenderer())
|
|
assert.NoError(t, err)
|
|
assert.False(t, skip)
|
|
assert.NotContains(t, c.values, "xyz")
|
|
|
|
// Values do not match skip condition. Prompt should not be skipped.
|
|
c.values["abc"] = "foobar"
|
|
c.values["def"] = 1234
|
|
skip, err = c.skipPrompt(jsonschema.Property{
|
|
Name: "xyz",
|
|
Schema: c.schema.Properties["xyz"],
|
|
}, testRenderer())
|
|
assert.NoError(t, err)
|
|
assert.False(t, skip)
|
|
assert.NotContains(t, c.values, "xyz")
|
|
|
|
// Values match skip condition. Prompt should be skipped. Default value should
|
|
// be assigned to "xyz".
|
|
c.values["abc"] = "foobar"
|
|
c.values["def"] = 123
|
|
skip, err = c.skipPrompt(jsonschema.Property{
|
|
Name: "xyz",
|
|
Schema: c.schema.Properties["xyz"],
|
|
}, testRenderer())
|
|
assert.NoError(t, err)
|
|
assert.True(t, skip)
|
|
assert.Equal(t, "hello-world", c.values["xyz"])
|
|
}
|
|
|
|
func TestPromptIsSkippedAnyOf(t *testing.T) {
|
|
c := config{
|
|
ctx: context.Background(),
|
|
values: make(map[string]any),
|
|
schema: &jsonschema.Schema{
|
|
Properties: map[string]*jsonschema.Schema{
|
|
"abc": {
|
|
Type: "string",
|
|
},
|
|
"def": {
|
|
Type: "integer",
|
|
},
|
|
"xyz": {
|
|
Type: "string",
|
|
Default: "hello-world",
|
|
Extension: jsonschema.Extension{
|
|
SkipPromptIf: &jsonschema.Schema{
|
|
AnyOf: []jsonschema.Schema{
|
|
{
|
|
Properties: map[string]*jsonschema.Schema{
|
|
"abc": {
|
|
Const: "foobar",
|
|
},
|
|
"def": {
|
|
Const: 123,
|
|
},
|
|
},
|
|
Required: []string{"abc", "def"},
|
|
},
|
|
{
|
|
Properties: map[string]*jsonschema.Schema{
|
|
"abc": {
|
|
Const: "barfoo",
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
}
|
|
|
|
// No skip condition defined. Prompt should not be skipped.
|
|
skip, err := c.skipPrompt(jsonschema.Property{
|
|
Name: "abc",
|
|
Schema: c.schema.Properties["abc"],
|
|
}, testRenderer())
|
|
assert.NoError(t, err)
|
|
assert.False(t, skip)
|
|
|
|
// Values do not match skip condition. Prompt should not be skipped.
|
|
c.values = map[string]any{
|
|
"abc": "foobar",
|
|
"def": 1234,
|
|
}
|
|
skip, err = c.skipPrompt(jsonschema.Property{
|
|
Name: "xyz",
|
|
Schema: c.schema.Properties["xyz"],
|
|
}, testRenderer())
|
|
assert.NoError(t, err)
|
|
assert.False(t, skip)
|
|
assert.NotContains(t, c.values, "xyz")
|
|
|
|
// def is missing value. Prompt should not be skipped.
|
|
c.values = map[string]any{
|
|
"abc": "foobar",
|
|
}
|
|
_, err = c.skipPrompt(jsonschema.Property{
|
|
Name: "xyz",
|
|
Schema: c.schema.Properties["xyz"],
|
|
}, testRenderer())
|
|
assert.NoError(t, err)
|
|
assert.False(t, skip)
|
|
assert.NotContains(t, c.values, "xyz")
|
|
|
|
// abc is missing value. Prompt should be skipped because abc is optional
|
|
// in second condition.
|
|
c.values = map[string]any{
|
|
"def": 123,
|
|
}
|
|
skip, err = c.skipPrompt(jsonschema.Property{
|
|
Name: "xyz",
|
|
Schema: c.schema.Properties["xyz"],
|
|
}, testRenderer())
|
|
assert.NoError(t, err)
|
|
assert.True(t, skip)
|
|
assert.Equal(t, "hello-world", c.values["xyz"])
|
|
|
|
// Values match skip condition. Prompt should be skipped. Default value should
|
|
// be assigned to "xyz".
|
|
c.values = map[string]any{
|
|
"abc": "foobar",
|
|
"def": 123,
|
|
}
|
|
skip, err = c.skipPrompt(jsonschema.Property{
|
|
Name: "xyz",
|
|
Schema: c.schema.Properties["xyz"],
|
|
}, testRenderer())
|
|
assert.NoError(t, err)
|
|
assert.True(t, skip)
|
|
assert.Equal(t, "hello-world", c.values["xyz"])
|
|
|
|
// Values match skip condition. Prompt should be skipped. Default value should
|
|
// be assigned to "xyz".
|
|
c.values = map[string]any{
|
|
"abc": "barfoo",
|
|
}
|
|
skip, err = c.skipPrompt(jsonschema.Property{
|
|
Name: "xyz",
|
|
Schema: c.schema.Properties["xyz"],
|
|
}, testRenderer())
|
|
assert.NoError(t, err)
|
|
assert.True(t, skip)
|
|
assert.Equal(t, "hello-world", c.values["xyz"])
|
|
}
|
|
|
|
func TestConfigEnumValues(t *testing.T) {
|
|
c := &config{
|
|
schema: &jsonschema.Schema{
|
|
Properties: map[string]*jsonschema.Schema{
|
|
"a": {
|
|
Type: jsonschema.StringType,
|
|
},
|
|
"b": {
|
|
Type: jsonschema.BooleanType,
|
|
},
|
|
"c": {
|
|
Type: jsonschema.StringType,
|
|
Enum: []any{"v1", "v2"},
|
|
},
|
|
"d": {
|
|
Type: jsonschema.StringType,
|
|
Enum: []any{"v3", "v4"},
|
|
},
|
|
"e": {
|
|
Type: jsonschema.StringType,
|
|
Enum: []any{"v5", "v6"},
|
|
},
|
|
},
|
|
},
|
|
values: map[string]any{
|
|
"a": "w1",
|
|
"b": false,
|
|
"c": "v1",
|
|
"d": "v3",
|
|
"e": "v7",
|
|
},
|
|
}
|
|
|
|
assert.Equal(t, map[string]string{
|
|
"c": "v1",
|
|
"d": "v3",
|
|
}, c.enumValues())
|
|
}
|