2023-08-07 13:14:25 +00:00
package template
import (
2023-09-07 14:36:06 +00:00
"context"
2023-11-30 14:28:51 +00:00
"fmt"
2024-11-20 09:28:35 +00:00
"os"
"path"
2024-08-12 14:20:04 +00:00
"path/filepath"
2023-08-07 13:14:25 +00:00
"testing"
2023-11-30 16:07:45 +00:00
"text/template"
2023-08-07 13:14:25 +00:00
"github.com/databricks/cli/libs/jsonschema"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
)
func TestTemplateConfigAssignValuesFromFile ( t * testing . T ) {
2024-08-12 14:20:04 +00:00
testDir := "./testdata/config-assign-from-file"
2023-08-07 13:14:25 +00:00
2024-08-12 14:20:04 +00:00
ctx := context . Background ( )
2024-11-20 09:28:35 +00:00
c , err := newConfig ( ctx , os . DirFS ( testDir ) , "schema.json" )
2024-08-12 14:20:04 +00:00
require . NoError ( t , err )
2023-08-07 13:14:25 +00:00
2024-08-12 14:20:04 +00:00
err = c . assignValuesFromFile ( filepath . Join ( testDir , "config.json" ) )
if assert . NoError ( t , err ) {
assert . Equal ( t , int64 ( 1 ) , c . values [ "int_val" ] )
assert . Equal ( t , float64 ( 2 ) , c . values [ "float_val" ] )
assert . Equal ( t , true , c . values [ "bool_val" ] )
assert . Equal ( t , "hello" , c . values [ "string_val" ] )
}
2023-08-07 13:14:25 +00:00
}
2024-08-12 14:20:04 +00:00
func TestTemplateConfigAssignValuesFromFileDoesNotOverwriteExistingConfigs ( t * testing . T ) {
testDir := "./testdata/config-assign-from-file"
2023-08-07 13:14:25 +00:00
2024-08-12 14:20:04 +00:00
ctx := context . Background ( )
2024-11-20 09:28:35 +00:00
c , err := newConfig ( ctx , os . DirFS ( testDir ) , "schema.json" )
2024-08-12 14:20:04 +00:00
require . NoError ( t , err )
2023-08-07 13:14:25 +00:00
2023-09-07 14:36:06 +00:00
c . values = map [ string ] any {
"string_val" : "this-is-not-overwritten" ,
2023-08-07 13:14:25 +00:00
}
2024-08-12 14:20:04 +00:00
err = c . assignValuesFromFile ( filepath . Join ( testDir , "config.json" ) )
if assert . NoError ( t , err ) {
assert . Equal ( t , int64 ( 1 ) , c . values [ "int_val" ] )
assert . Equal ( t , float64 ( 2 ) , c . values [ "float_val" ] )
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 ( )
2024-11-20 09:28:35 +00:00
c , err := newConfig ( ctx , os . DirFS ( testDir ) , "schema.json" )
2024-08-12 14:20:04 +00:00
require . NoError ( t , err )
2023-08-07 13:14:25 +00:00
2024-08-12 14:20:04 +00:00
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" ) ) )
2023-08-07 13:14:25 +00:00
}
2024-03-26 13:02:09 +00:00
func TestTemplateConfigAssignValuesFromFileFiltersPropertiesNotInTheSchema ( t * testing . T ) {
2024-08-12 14:20:04 +00:00
testDir := "./testdata/config-assign-from-file-unknown-property"
2024-03-26 13:02:09 +00:00
2024-08-12 14:20:04 +00:00
ctx := context . Background ( )
2024-11-20 09:28:35 +00:00
c , err := newConfig ( ctx , os . DirFS ( testDir ) , "schema.json" )
2024-08-12 14:20:04 +00:00
require . NoError ( t , err )
err = c . assignValuesFromFile ( filepath . Join ( testDir , "config.json" ) )
2024-03-26 13:02:09 +00:00
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" ] )
}
2024-08-12 14:20:04 +00:00
func TestTemplateConfigAssignValuesFromDefaultValues ( t * testing . T ) {
testDir := "./testdata/config-assign-from-default-value"
2023-08-07 13:14:25 +00:00
2023-10-19 07:08:36 +00:00
ctx := context . Background ( )
2024-11-20 09:28:35 +00:00
c , err := newConfig ( ctx , os . DirFS ( testDir ) , "schema.json" )
2024-08-12 14:20:04 +00:00
require . NoError ( t , err )
2024-11-20 10:11:31 +00:00
r , err := newRenderer ( ctx , nil , nil , os . DirFS ( "." ) , "./testdata/empty/template" , "./testdata/empty/library" )
2023-10-19 07:08:36 +00:00
require . NoError ( t , err )
err = c . assignDefaultValues ( r )
2024-08-12 14:20:04 +00:00
if assert . NoError ( t , err ) {
assert . Equal ( t , int64 ( 123 ) , c . values [ "int_val" ] )
assert . Equal ( t , float64 ( 123 ) , c . values [ "float_val" ] )
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 ( )
2024-11-20 09:28:35 +00:00
c , err := newConfig ( ctx , os . DirFS ( testDir ) , "schema.json" )
2024-08-12 14:20:04 +00:00
require . NoError ( t , err )
2024-11-20 10:11:31 +00:00
r , err := newRenderer ( ctx , nil , nil , os . DirFS ( "." ) , path . Join ( testDir , "template/template" ) , path . Join ( testDir , "template/library" ) )
2024-08-12 14:20:04 +00:00
require . NoError ( t , err )
2023-08-07 13:14:25 +00:00
2024-08-12 14:20:04 +00:00
// 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 . Equal ( t , float64 ( 123 ) , c . values [ "float_val" ] )
assert . Equal ( t , true , c . values [ "bool_val" ] )
assert . Equal ( t , "world" , c . values [ "string_val" ] )
}
2023-08-07 13:14:25 +00:00
}
func TestTemplateConfigValidateValuesDefined ( t * testing . T ) {
2024-08-12 14:20:04 +00:00
ctx := context . Background ( )
2024-11-20 09:28:35 +00:00
c , err := newConfig ( ctx , os . DirFS ( "testdata/config-test-schema" ) , "test-schema.json" )
2024-08-12 14:20:04 +00:00
require . NoError ( t , err )
2023-09-07 14:36:06 +00:00
c . values = map [ string ] any {
"int_val" : 1 ,
"float_val" : 1.0 ,
"bool_val" : false ,
2023-08-07 13:14:25 +00:00
}
2024-08-12 14:20:04 +00:00
err = c . validate ( )
2023-09-07 14:36:06 +00:00
assert . EqualError ( t , err , "validation for template input parameters failed. no value provided for required property string_val" )
2023-08-07 13:14:25 +00:00
}
func TestTemplateConfigValidateTypeForValidConfig ( t * testing . T ) {
2024-08-12 14:20:04 +00:00
ctx := context . Background ( )
2024-11-20 09:28:35 +00:00
c , err := newConfig ( ctx , os . DirFS ( "testdata/config-test-schema" ) , "test-schema.json" )
2024-08-12 14:20:04 +00:00
require . NoError ( t , err )
2023-09-07 14:36:06 +00:00
c . values = map [ string ] any {
"int_val" : 1 ,
"float_val" : 1.1 ,
"bool_val" : true ,
"string_val" : "abcd" ,
2023-08-07 13:14:25 +00:00
}
2024-08-12 14:20:04 +00:00
err = c . validate ( )
2023-08-07 13:14:25 +00:00
assert . NoError ( t , err )
}
func TestTemplateConfigValidateTypeForUnknownField ( t * testing . T ) {
2024-08-12 14:20:04 +00:00
ctx := context . Background ( )
2024-11-20 09:28:35 +00:00
c , err := newConfig ( ctx , os . DirFS ( "testdata/config-test-schema" ) , "test-schema.json" )
2024-08-12 14:20:04 +00:00
require . NoError ( t , err )
2023-09-07 14:36:06 +00:00
c . values = map [ string ] any {
"unknown_prop" : 1 ,
"int_val" : 1 ,
"float_val" : 1.1 ,
"bool_val" : true ,
"string_val" : "abcd" ,
2023-08-07 13:14:25 +00:00
}
2024-08-12 14:20:04 +00:00
err = c . validate ( )
2023-09-07 14:36:06 +00:00
assert . EqualError ( t , err , "validation for template input parameters failed. property unknown_prop is not defined in the schema" )
2023-08-07 13:14:25 +00:00
}
func TestTemplateConfigValidateTypeForInvalidType ( t * testing . T ) {
2024-08-12 14:20:04 +00:00
ctx := context . Background ( )
2024-11-20 09:28:35 +00:00
c , err := newConfig ( ctx , os . DirFS ( "testdata/config-test-schema" ) , "test-schema.json" )
2024-08-12 14:20:04 +00:00
require . NoError ( t , err )
2023-09-07 14:36:06 +00:00
c . values = map [ string ] any {
"int_val" : "this-should-be-an-int" ,
"float_val" : 1.1 ,
"bool_val" : true ,
"string_val" : "abcd" ,
2023-08-07 13:14:25 +00:00
}
2024-08-12 14:20:04 +00:00
err = c . validate ( )
2023-09-07 14:36:06 +00:00
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\"" )
2023-08-07 13:14:25 +00:00
}
2023-08-15 14:28:04 +00:00
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" )
}
2023-09-08 12:07:22 +00:00
2023-11-30 14:28:51 +00:00
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 ) )
}
2023-09-08 12:07:22 +00:00
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 ( ) )
}
2023-10-19 07:08:36 +00:00
2023-11-08 16:48:37 +00:00
func TestTemplateSchemaErrorsWithEmptyDescription ( t * testing . T ) {
2024-11-20 09:28:35 +00:00
ctx := context . Background ( )
_ , err := newConfig ( ctx , os . DirFS ( "./testdata/config-test-schema" ) , "invalid-test-schema.json" )
2023-11-08 16:48:37 +00:00
assert . EqualError ( t , err , "template property property-without-description is missing a description" )
}
2023-11-30 16:07:45 +00:00
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 ,
} ,
} ,
2024-01-25 10:09:42 +00:00
Required : [ ] string { "abc" , "def" } ,
2023-11-30 16:07:45 +00:00
} ,
} ,
} ,
} ,
} ,
}
// 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" ] )
}
2024-01-25 10:09:42 +00:00
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 {
2024-09-10 13:55:18 +00:00
AnyOf : [ ] jsonschema . Schema {
2024-01-25 10:09:42 +00:00
{
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" ] )
}
2024-12-27 06:49:42 +00:00
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 ( ) )
}