databricks-cli/bundle/schema/schema_test.go

1300 lines
25 KiB
Go
Raw Normal View History

package schema
import (
"encoding/json"
"reflect"
"testing"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
)
2023-01-17 14:34:49 +00:00
func TestIntSchema(t *testing.T) {
2023-01-17 14:08:17 +00:00
var elemInt int
expected :=
`{
"type": "number"
}`
2023-01-19 10:34:55 +00:00
schema, err := NewSchema(reflect.TypeOf(elemInt), nil)
2023-01-17 14:08:17 +00:00
require.NoError(t, err)
2023-01-19 10:34:55 +00:00
jsonSchema, err := json.MarshalIndent(schema, " ", " ")
2023-01-17 14:08:17 +00:00
assert.NoError(t, err)
t.Log("[DEBUG] actual: ", string(jsonSchema))
t.Log("[DEBUG] expected: ", expected)
assert.Equal(t, expected, string(jsonSchema))
}
func TestBooleanSchema(t *testing.T) {
var elem bool
expected :=
`{
"type": "boolean"
}`
2023-01-19 10:34:55 +00:00
schema, err := NewSchema(reflect.TypeOf(elem), nil)
2023-01-17 14:08:17 +00:00
require.NoError(t, err)
2023-01-19 10:34:55 +00:00
jsonSchema, err := json.MarshalIndent(schema, " ", " ")
2023-01-17 14:08:17 +00:00
assert.NoError(t, err)
t.Log("[DEBUG] actual: ", string(jsonSchema))
t.Log("[DEBUG] expected: ", expected)
assert.Equal(t, expected, string(jsonSchema))
}
func TestStringSchema(t *testing.T) {
var elem string
expected :=
`{
"type": "string"
}`
2023-01-19 10:34:55 +00:00
schema, err := NewSchema(reflect.TypeOf(elem), nil)
2023-01-17 14:08:17 +00:00
require.NoError(t, err)
2023-01-19 10:34:55 +00:00
jsonSchema, err := json.MarshalIndent(schema, " ", " ")
2023-01-17 14:08:17 +00:00
assert.NoError(t, err)
t.Log("[DEBUG] actual: ", string(jsonSchema))
t.Log("[DEBUG] expected: ", expected)
assert.Equal(t, expected, string(jsonSchema))
}
2023-01-17 14:34:49 +00:00
func TestStructOfPrimitivesSchema(t *testing.T) {
type Foo struct {
IntVal int `json:"int_val"`
Int8Val int8 `json:"int8_val"`
Int16Val int16 `json:"int16_val"`
Int32Val int32 `json:"int32_val"`
Int64Val int64 `json:"int64_val"`
2023-01-17 14:08:17 +00:00
UIntVal uint `json:"uint_val"`
Uint8Val uint8 `json:"uint8_val"`
Uint16Val uint16 `json:"uint16_val"`
Uint32Val uint32 `json:"uint32_val"`
Uint64Val uint64 `json:"uint64_val"`
2023-01-17 14:08:17 +00:00
Float32Val float32 `json:"float32_val"`
Float64Val float64 `json:"float64_val"`
StringVal string `json:"string_val"`
2023-01-17 14:08:17 +00:00
BoolVal bool `json:"bool_val"`
}
elem := Foo{}
2023-01-18 15:02:38 +00:00
schema, err := NewSchema(reflect.TypeOf(elem), nil)
assert.NoError(t, err)
jsonSchema, err := json.MarshalIndent(schema, " ", " ")
assert.NoError(t, err)
expected :=
`{
"type": "object",
"properties": {
"bool_val": {
2023-01-17 14:08:17 +00:00
"type": "boolean"
},
"float32_val": {
"type": "number"
},
"float64_val": {
"type": "number"
},
"int16_val": {
"type": "number"
},
"int32_val": {
"type": "number"
},
"int64_val": {
"type": "number"
},
"int8_val": {
"type": "number"
},
"int_val": {
"type": "number"
},
"string_val": {
"type": "string"
},
"uint16_val": {
"type": "number"
},
"uint32_val": {
"type": "number"
},
"uint64_val": {
"type": "number"
},
"uint8_val": {
"type": "number"
2023-01-17 14:08:17 +00:00
},
"uint_val": {
"type": "number"
}
},
"additionalProperties": false,
"required": [
"int_val",
"int8_val",
"int16_val",
"int32_val",
"int64_val",
"uint_val",
"uint8_val",
"uint16_val",
"uint32_val",
"uint64_val",
"float32_val",
"float64_val",
"string_val",
"bool_val"
]
}`
2023-01-13 18:25:42 +00:00
t.Log("[DEBUG] actual: ", string(jsonSchema))
t.Log("[DEBUG] expected: ", expected)
assert.Equal(t, expected, string(jsonSchema))
}
2023-01-17 14:34:49 +00:00
func TestStructOfStructsSchema(t *testing.T) {
type Bar struct {
A int `json:"a"`
B string `json:"b,string"`
}
type Foo struct {
Bar Bar `json:"bar"`
}
type MyStruct struct {
Foo Foo `json:"foo"`
}
elem := MyStruct{}
2023-01-18 15:02:38 +00:00
schema, err := NewSchema(reflect.TypeOf(elem), nil)
2023-01-17 14:34:49 +00:00
assert.NoError(t, err)
jsonSchema, err := json.MarshalIndent(schema, " ", " ")
assert.NoError(t, err)
expected :=
`{
"type": "object",
"properties": {
"foo": {
"type": "object",
"properties": {
"bar": {
"type": "object",
"properties": {
"a": {
"type": "number"
},
"b": {
"type": "string"
}
},
"additionalProperties": false,
"required": [
"a",
"b"
]
2023-01-17 14:34:49 +00:00
}
},
"additionalProperties": false,
"required": [
"bar"
]
2023-01-17 14:34:49 +00:00
}
},
"additionalProperties": false,
"required": [
"foo"
]
2023-01-17 14:34:49 +00:00
}`
t.Log("[DEBUG] actual: ", string(jsonSchema))
t.Log("[DEBUG] expected: ", expected)
assert.Equal(t, expected, string(jsonSchema))
}
func TestStructOfMapsSchema(t *testing.T) {
type Bar struct {
MyMap map[string]int `json:"my_map"`
}
type Foo struct {
Bar Bar `json:"bar"`
}
elem := Foo{}
2023-01-18 15:02:38 +00:00
schema, err := NewSchema(reflect.TypeOf(elem), nil)
assert.NoError(t, err)
jsonSchema, err := json.MarshalIndent(schema, " ", " ")
assert.NoError(t, err)
expected :=
`{
"type": "object",
"properties": {
"bar": {
"type": "object",
"properties": {
"my_map": {
"type": "object",
"additionalProperties": {
"type": "number"
}
}
},
"additionalProperties": false,
"required": [
"my_map"
]
}
},
"additionalProperties": false,
"required": [
"bar"
]
}`
t.Log("[DEBUG] actual: ", string(jsonSchema))
t.Log("[DEBUG] expected: ", expected)
assert.Equal(t, expected, string(jsonSchema))
}
func TestStructOfSliceSchema(t *testing.T) {
type Bar struct {
MySlice []string `json:"my_slice"`
}
type Foo struct {
Bar Bar `json:"bar"`
}
elem := Foo{}
2023-01-18 15:02:38 +00:00
schema, err := NewSchema(reflect.TypeOf(elem), nil)
assert.NoError(t, err)
jsonSchema, err := json.MarshalIndent(schema, " ", " ")
assert.NoError(t, err)
expected :=
`{
"type": "object",
"properties": {
"bar": {
"type": "object",
"properties": {
"my_slice": {
"type": "array",
"items": {
"type": "string"
}
}
},
"additionalProperties": false,
"required": [
"my_slice"
]
}
},
"additionalProperties": false,
"required": [
"bar"
]
}`
t.Log("[DEBUG] actual: ", string(jsonSchema))
t.Log("[DEBUG] expected: ", expected)
assert.Equal(t, expected, string(jsonSchema))
}
2023-01-17 14:47:31 +00:00
func TestMapOfPrimitivesSchema(t *testing.T) {
var elem map[string]int
2023-01-18 15:02:38 +00:00
schema, err := NewSchema(reflect.TypeOf(elem), nil)
2023-01-17 14:47:31 +00:00
assert.NoError(t, err)
jsonSchema, err := json.MarshalIndent(schema, " ", " ")
assert.NoError(t, err)
expected :=
`{
"type": "object",
"additionalProperties": {
"type": "number"
}
}`
t.Log("[DEBUG] actual: ", string(jsonSchema))
t.Log("[DEBUG] expected: ", expected)
assert.Equal(t, expected, string(jsonSchema))
}
func TestMapOfStructSchema(t *testing.T) {
type Foo struct {
MyInt int `json:"my_int"`
}
var elem map[string]Foo
2023-01-18 15:02:38 +00:00
schema, err := NewSchema(reflect.TypeOf(elem), nil)
assert.NoError(t, err)
jsonSchema, err := json.MarshalIndent(schema, " ", " ")
assert.NoError(t, err)
expected :=
`{
"type": "object",
"additionalProperties": {
"type": "object",
"properties": {
"my_int": {
"type": "number"
}
},
"additionalProperties": false,
"required": [
"my_int"
]
}
}`
t.Log("[DEBUG] actual: ", string(jsonSchema))
t.Log("[DEBUG] expected: ", expected)
assert.Equal(t, expected, string(jsonSchema))
}
2023-01-17 16:57:48 +00:00
func TestMapOfMapSchema(t *testing.T) {
var elem map[string]map[string]int
2023-01-18 15:02:38 +00:00
schema, err := NewSchema(reflect.TypeOf(elem), nil)
2023-01-17 16:57:48 +00:00
assert.NoError(t, err)
jsonSchema, err := json.MarshalIndent(schema, " ", " ")
assert.NoError(t, err)
expected :=
`{
"type": "object",
"additionalProperties": {
"type": "object",
"additionalProperties": {
"type": "number"
}
}
}`
t.Log("[DEBUG] actual: ", string(jsonSchema))
t.Log("[DEBUG] expected: ", expected)
assert.Equal(t, expected, string(jsonSchema))
}
2023-01-17 16:59:58 +00:00
func TestMapOfSliceSchema(t *testing.T) {
var elem map[string][]string
2023-01-18 15:02:38 +00:00
schema, err := NewSchema(reflect.TypeOf(elem), nil)
2023-01-17 16:59:58 +00:00
assert.NoError(t, err)
jsonSchema, err := json.MarshalIndent(schema, " ", " ")
assert.NoError(t, err)
expected :=
`{
"type": "object",
"additionalProperties": {
"type": "array",
"items": {
"type": "string"
}
}
}`
t.Log("[DEBUG] actual: ", string(jsonSchema))
t.Log("[DEBUG] expected: ", expected)
assert.Equal(t, expected, string(jsonSchema))
}
func TestSliceOfPrimitivesSchema(t *testing.T) {
var elem []float32
2023-01-18 15:02:38 +00:00
schema, err := NewSchema(reflect.TypeOf(elem), nil)
assert.NoError(t, err)
jsonSchema, err := json.MarshalIndent(schema, " ", " ")
assert.NoError(t, err)
expected :=
`{
"type": "array",
"items": {
"type": "number"
}
}`
t.Log("[DEBUG] actual: ", string(jsonSchema))
t.Log("[DEBUG] expected: ", expected)
assert.Equal(t, expected, string(jsonSchema))
}
func TestSliceOfSliceSchema(t *testing.T) {
var elem [][]string
2023-01-18 15:02:38 +00:00
schema, err := NewSchema(reflect.TypeOf(elem), nil)
assert.NoError(t, err)
jsonSchema, err := json.MarshalIndent(schema, " ", " ")
assert.NoError(t, err)
expected :=
`{
"type": "array",
"items": {
"type": "array",
"items": {
"type": "string"
}
}
}`
t.Log("[DEBUG] actual: ", string(jsonSchema))
t.Log("[DEBUG] expected: ", expected)
assert.Equal(t, expected, string(jsonSchema))
}
func TestSliceOfMapSchema(t *testing.T) {
var elem []map[string]int
2023-01-18 15:02:38 +00:00
schema, err := NewSchema(reflect.TypeOf(elem), nil)
assert.NoError(t, err)
jsonSchema, err := json.MarshalIndent(schema, " ", " ")
assert.NoError(t, err)
expected :=
`{
"type": "array",
"items": {
"type": "object",
"additionalProperties": {
"type": "number"
}
}
}`
t.Log("[DEBUG] actual: ", string(jsonSchema))
t.Log("[DEBUG] expected: ", expected)
assert.Equal(t, expected, string(jsonSchema))
}
func TestSliceOfStructSchema(t *testing.T) {
type Foo struct {
MyInt int `json:"my_int"`
}
var elem []Foo
2023-01-18 15:02:38 +00:00
schema, err := NewSchema(reflect.TypeOf(elem), nil)
assert.NoError(t, err)
jsonSchema, err := json.MarshalIndent(schema, " ", " ")
assert.NoError(t, err)
expected :=
`{
"type": "array",
"items": {
"type": "object",
"properties": {
"my_int": {
"type": "number"
}
},
"additionalProperties": false,
"required": [
"my_int"
]
}
}`
t.Log("[DEBUG] actual: ", string(jsonSchema))
t.Log("[DEBUG] expected: ", expected)
assert.Equal(t, expected, string(jsonSchema))
}
2023-01-13 18:26:50 +00:00
func TestEmbeddedStructSchema(t *testing.T) {
2023-01-13 18:25:42 +00:00
type Location struct {
Country string `json:"country"`
2023-01-13 18:26:50 +00:00
State string `json:"state,omitempty"`
2023-01-13 18:25:42 +00:00
}
2023-01-18 09:42:33 +00:00
type Person struct {
Name string `json:"name"`
Age int `json:"age,omitempty"`
Home Location `json:"home"`
}
2023-01-13 18:25:42 +00:00
type Plot struct {
Events map[string]Person `json:"events"`
}
type Story struct {
Plot Plot `json:"plot"`
*Person
Location
}
elem := Story{}
2023-01-18 15:02:38 +00:00
schema, err := NewSchema(reflect.TypeOf(elem), nil)
2023-01-13 18:25:42 +00:00
assert.NoError(t, err)
jsonSchema, err := json.MarshalIndent(schema, " ", " ")
assert.NoError(t, err)
expected :=
`{
"type": "object",
"properties": {
"age": {
"type": "number"
},
"country": {
"type": "string"
},
2023-01-18 09:42:33 +00:00
"home": {
"type": "object",
"properties": {
"country": {
"type": "string"
},
"state": {
"type": "string"
}
},
"additionalProperties": false,
"required": [
"country"
]
2023-01-18 09:42:33 +00:00
},
2023-01-13 18:25:42 +00:00
"name": {
"type": "string"
},
"plot": {
"type": "object",
"properties": {
"events": {
"type": "object",
"additionalProperties": {
"type": "object",
"properties": {
"age": {
"type": "number"
},
2023-01-18 09:42:33 +00:00
"home": {
"type": "object",
"properties": {
"country": {
"type": "string"
},
"state": {
"type": "string"
}
},
"additionalProperties": false,
"required": [
"country"
]
2023-01-18 09:42:33 +00:00
},
2023-01-13 18:25:42 +00:00
"name": {
"type": "string"
}
2023-01-18 09:42:33 +00:00
},
"additionalProperties": false,
"required": [
"name",
"home"
]
2023-01-13 18:25:42 +00:00
}
}
2023-01-18 09:42:33 +00:00
},
"additionalProperties": false,
"required": [
"events"
]
2023-01-13 18:25:42 +00:00
},
"state": {
"type": "string"
}
2023-01-18 09:42:33 +00:00
},
"additionalProperties": false,
"required": [
"plot",
"name",
"home",
"country"
]
2023-01-13 18:25:42 +00:00
}`
t.Log("[DEBUG] actual: ", string(jsonSchema))
t.Log("[DEBUG] expected: ", expected)
2023-01-13 17:01:53 +00:00
assert.Equal(t, expected, string(jsonSchema))
}
2023-01-13 19:03:24 +00:00
2023-01-16 12:18:56 +00:00
func TestErrorWithTrace(t *testing.T) {
tracker := newTracker()
dummyType := reflect.TypeOf(struct{}{})
err := tracker.errWithTrace("with empty trace")
2023-01-16 12:18:56 +00:00
assert.ErrorContains(t, err, "[ERROR] with empty trace. traversal trace: root")
2023-01-20 15:26:13 +00:00
tracker.push(dummyType, "resources")
err = tracker.errWithTrace("with depth = 1")
2023-01-16 12:18:56 +00:00
assert.ErrorContains(t, err, "[ERROR] with depth = 1. traversal trace: root -> resources")
2023-01-20 15:26:13 +00:00
tracker.push(dummyType, "pipelines")
tracker.push(dummyType, "datasets")
err = tracker.errWithTrace("with depth = 4")
2023-01-16 12:18:56 +00:00
assert.ErrorContains(t, err, "[ERROR] with depth = 4. traversal trace: root -> resources -> pipelines -> datasets")
}
func TestNonAnnotatedFieldsAreSkipped(t *testing.T) {
type MyStruct struct {
Foo string
Bar int `json:"bar"`
}
elem := MyStruct{}
2023-01-18 15:02:38 +00:00
schema, err := NewSchema(reflect.TypeOf(elem), nil)
require.NoError(t, err)
jsonSchema, err := json.MarshalIndent(schema, " ", " ")
assert.NoError(t, err)
expectedSchema :=
`{
"type": "object",
"properties": {
"bar": {
"type": "number"
}
},
"additionalProperties": false,
"required": [
"bar"
]
}`
t.Log("[DEBUG] actual: ", string(jsonSchema))
t.Log("[DEBUG] expected: ", expectedSchema)
assert.Equal(t, expectedSchema, string(jsonSchema))
}
func TestDashFieldsAreSkipped(t *testing.T) {
type MyStruct struct {
Foo string `json:"-"`
Bar int `json:"bar"`
}
elem := MyStruct{}
2023-01-18 15:02:38 +00:00
schema, err := NewSchema(reflect.TypeOf(elem), nil)
require.NoError(t, err)
jsonSchema, err := json.MarshalIndent(schema, " ", " ")
assert.NoError(t, err)
expectedSchema :=
`{
"type": "object",
"properties": {
"bar": {
"type": "number"
}
},
"additionalProperties": false,
"required": [
"bar"
]
}`
t.Log("[DEBUG] actual: ", string(jsonSchema))
t.Log("[DEBUG] expected: ", expectedSchema)
assert.Equal(t, expectedSchema, string(jsonSchema))
}
2023-01-18 10:02:25 +00:00
func TestPointerInStructSchema(t *testing.T) {
type Bar struct {
PtrVal2 *int `json:"ptr_val2"`
}
type Foo struct {
PtrInt *int `json:"ptr_int"`
PtrString *string `json:"ptr_string"`
FloatVal float32 `json:"float_val"`
PtrBar *Bar `json:"ptr_bar"`
Bar *Bar `json:"bar"`
}
elem := Foo{}
2023-01-18 15:02:38 +00:00
schema, err := NewSchema(reflect.TypeOf(elem), nil)
2023-01-18 10:02:25 +00:00
require.NoError(t, err)
jsonSchema, err := json.MarshalIndent(schema, " ", " ")
assert.NoError(t, err)
expectedSchema :=
`{
"type": "object",
"properties": {
"bar": {
"type": "object",
"properties": {
"ptr_val2": {
"type": "number"
}
},
"additionalProperties": false,
"required": [
"ptr_val2"
]
2023-01-18 10:02:25 +00:00
},
"float_val": {
"type": "number"
},
"ptr_bar": {
"type": "object",
"properties": {
"ptr_val2": {
"type": "number"
}
},
"additionalProperties": false,
"required": [
"ptr_val2"
]
2023-01-18 10:02:25 +00:00
},
"ptr_int": {
"type": "number"
},
"ptr_string": {
"type": "string"
}
},
"additionalProperties": false,
"required": [
"ptr_int",
"ptr_string",
"float_val",
"ptr_bar",
"bar"
]
2023-01-18 10:02:25 +00:00
}`
t.Log("[DEBUG] actual: ", string(jsonSchema))
t.Log("[DEBUG] expected: ", expectedSchema)
assert.Equal(t, expectedSchema, string(jsonSchema))
}
2023-01-18 17:09:18 +00:00
func TestGenericSchema(t *testing.T) {
2023-01-17 18:32:44 +00:00
type Person struct {
Name string `json:"name"`
Age int `json:"age,omitempty"`
}
type Plot struct {
2023-01-18 17:09:18 +00:00
Stakes []string `json:"stakes"`
Deaths []Person `json:"deaths"`
Murders map[string]Person `json:"murders"`
}
type Wedding struct {
Hidden string `json:","`
Groom Person `json:"groom"`
Bride Person `json:"bride"`
Plots []Plot `json:"plots"`
2023-01-17 18:32:44 +00:00
}
type Story struct {
2023-01-18 17:09:18 +00:00
Hero *Person `json:"hero"`
Villian Person `json:"villian,omitempty"`
Weddings []Wedding `json:"weddings"`
2023-01-17 18:32:44 +00:00
}
elem := Story{}
2023-01-18 15:02:38 +00:00
schema, err := NewSchema(reflect.TypeOf(elem), nil)
2023-01-17 18:32:44 +00:00
assert.NoError(t, err)
jsonSchema, err := json.MarshalIndent(schema, " ", " ")
assert.NoError(t, err)
expected :=
`{
"type": "object",
"properties": {
"hero": {
"type": "object",
"properties": {
"age": {
"type": "number"
},
"name": {
"type": "string"
}
},
"additionalProperties": false,
"required": [
"name"
]
2023-01-17 18:32:44 +00:00
},
"villian": {
"type": "object",
"properties": {
"age": {
"type": "number"
},
"name": {
"type": "string"
}
},
"additionalProperties": false,
"required": [
"name"
]
2023-01-18 17:09:18 +00:00
},
"weddings": {
"type": "array",
"items": {
"type": "object",
"properties": {
"bride": {
"type": "object",
"properties": {
"age": {
"type": "number"
},
"name": {
"type": "string"
}
},
"additionalProperties": false,
"required": [
"name"
]
},
"groom": {
"type": "object",
"properties": {
"age": {
"type": "number"
},
"name": {
"type": "string"
}
},
"additionalProperties": false,
"required": [
"name"
]
},
"plots": {
"type": "array",
"items": {
"type": "object",
"properties": {
"deaths": {
"type": "array",
"items": {
"type": "object",
"properties": {
"age": {
"type": "number"
},
"name": {
"type": "string"
}
},
"additionalProperties": false,
"required": [
"name"
]
}
},
"murders": {
"type": "object",
"additionalProperties": {
"type": "object",
"properties": {
"age": {
"type": "number"
},
"name": {
"type": "string"
}
},
"additionalProperties": false,
"required": [
"name"
]
}
},
"stakes": {
"type": "array",
"items": {
"type": "string"
}
}
},
"additionalProperties": false,
"required": [
"stakes",
"deaths",
"murders"
]
}
}
},
"additionalProperties": false,
"required": [
"groom",
"bride",
"plots"
]
}
2023-01-17 18:32:44 +00:00
}
},
"additionalProperties": false,
"required": [
"hero",
2023-01-18 17:09:18 +00:00
"weddings"
]
2023-01-17 18:32:44 +00:00
}`
t.Log("[DEBUG] actual: ", string(jsonSchema))
t.Log("[DEBUG] expected: ", expected)
assert.Equal(t, expected, string(jsonSchema))
}
2023-01-18 12:12:01 +00:00
func TestFieldsWithoutOmitEmptyAreRequired(t *testing.T) {
type Papaya struct {
A int `json:"a,string,omitempty"`
B string `json:"b"`
}
type MyStruct struct {
Foo string `json:"-,omitempty"`
Bar int `json:"bar"`
Apple int `json:"apple,omitempty"`
Mango int `json:",omitempty"`
Guava int `json:","`
Papaya *Papaya `json:"papaya,"`
}
elem := MyStruct{}
2023-01-18 15:02:38 +00:00
schema, err := NewSchema(reflect.TypeOf(elem), nil)
2023-01-18 12:12:01 +00:00
require.NoError(t, err)
jsonSchema, err := json.MarshalIndent(schema, " ", " ")
assert.NoError(t, err)
expectedSchema :=
`{
"type": "object",
"properties": {
"apple": {
"type": "number"
},
"bar": {
"type": "number"
},
"papaya": {
"type": "object",
"properties": {
"a": {
"type": "number"
},
"b": {
"type": "string"
}
},
"additionalProperties": false,
"required": [
"b"
]
}
},
"additionalProperties": false,
"required": [
"bar",
"papaya"
]
}`
t.Log("[DEBUG] actual: ", string(jsonSchema))
t.Log("[DEBUG] expected: ", expectedSchema)
assert.Equal(t, expectedSchema, string(jsonSchema))
}
2023-01-18 15:02:38 +00:00
func TestDocIngestionInSchema(t *testing.T) {
docs := &Docs{
Documentation: "docs for root",
Children: map[string]Docs{
"my_struct": {
Documentation: "docs for my struct",
},
"my_val": {
Documentation: "docs for my val",
},
"my_slice": {
Documentation: "docs for my slice",
Children: map[string]Docs{
"guava": {
Documentation: "docs for guava",
},
"pineapple": {
Documentation: "docs for pineapple",
},
},
},
"my_map": {
Documentation: "docs for my map",
Children: map[string]Docs{
"apple": {
Documentation: "docs for apple",
},
"mango": {
Documentation: "docs for mango",
},
},
},
},
}
type Foo struct {
Apple int `json:"apple"`
Mango int `json:"mango"`
}
type Bar struct {
Guava int `json:"guava"`
Pineapple int `json:"pineapple"`
}
type MyStruct struct {
A string `json:"a"`
}
type Root struct {
MyStruct *MyStruct `json:"my_struct"`
MyVal int `json:"my_val"`
MySlice []Bar `json:"my_slice"`
MyMap map[string]*Foo `json:"my_map"`
}
elem := Root{}
schema, err := NewSchema(reflect.TypeOf(elem), docs)
require.NoError(t, err)
jsonSchema, err := json.MarshalIndent(schema, " ", " ")
assert.NoError(t, err)
expectedSchema :=
`{
"type": "object",
"description": "docs for root",
"properties": {
"my_map": {
"type": "object",
"description": "docs for my map",
"additionalProperties": {
"type": "object",
"description": "docs for my map",
"properties": {
"apple": {
"type": "number",
"description": "docs for apple"
},
"mango": {
"type": "number",
"description": "docs for mango"
}
},
"additionalProperties": false,
"required": [
"apple",
"mango"
]
}
},
"my_slice": {
"type": "array",
"description": "docs for my slice",
"items": {
"type": "object",
"properties": {
"guava": {
"type": "number",
"description": "docs for guava"
},
"pineapple": {
"type": "number",
"description": "docs for pineapple"
}
},
"additionalProperties": false,
"required": [
"guava",
"pineapple"
]
}
},
"my_struct": {
"type": "object",
"description": "docs for my struct",
"properties": {
"a": {
"type": "string"
}
},
"additionalProperties": false,
"required": [
"a"
]
},
"my_val": {
"type": "number",
"description": "docs for my val"
}
},
"additionalProperties": false,
"required": [
"my_struct",
"my_val",
"my_slice",
"my_map"
]
}`
t.Log("[DEBUG] actual: ", string(jsonSchema))
t.Log("[DEBUG] expected: ", expectedSchema)
assert.Equal(t, expectedSchema, string(jsonSchema))
}
func TestErrorOnMapWithoutStringKey(t *testing.T) {
type Foo struct {
Bar map[int]string `json:"bar"`
}
elem := Foo{}
_, err := NewSchema(reflect.TypeOf(elem), nil)
assert.ErrorContains(t, err, "only strings map keys are valid. key type: int")
}
func TestErrorIfStructRefersToItself(t *testing.T) {
type Foo struct {
MyFoo *Foo `json:"my_foo"`
}
elem := Foo{}
_, err := NewSchema(reflect.TypeOf(elem), nil)
2023-01-18 16:36:50 +00:00
assert.ErrorContains(t, err, "ERROR] cycle detected. traversal trace: root -> my_foo")
}
func TestErrorIfStructHasLoop(t *testing.T) {
type Apple struct {
MyVal int `json:"my_val"`
MyMango struct {
MyGuava struct {
MyPapaya struct {
MyApple *Apple `json:"my_apple"`
} `json:"my_papaya"`
} `json:"my_guava"`
} `json:"my_mango"`
}
elem := Apple{}
_, err := NewSchema(reflect.TypeOf(elem), nil)
2023-01-18 16:36:50 +00:00
assert.ErrorContains(t, err, "[ERROR] cycle detected. traversal trace: root -> my_mango -> my_guava -> my_papaya -> my_apple")
}
func TestInterfaceGeneratesEmptySchema(t *testing.T) {
type Foo struct {
Apple int `json:"apple"`
Mango interface{} `json:"mango"`
}
elem := Foo{}
schema, err := NewSchema(reflect.TypeOf(elem), nil)
assert.NoError(t, err)
jsonSchema, err := json.MarshalIndent(schema, " ", " ")
assert.NoError(t, err)
expected :=
`{
"type": "object",
"properties": {
"apple": {
"type": "number"
},
"mango": {}
},
"additionalProperties": false,
"required": [
"apple",
"mango"
]
}`
t.Log("[DEBUG] actual: ", string(jsonSchema))
t.Log("[DEBUG] expected: ", expected)
assert.Equal(t, expected, string(jsonSchema))
}
// A toy test to generate the schema for bundle. Will be removed once we have a
// command to generate the json schema
2023-01-16 01:30:41 +00:00
// func TestBundleSchema(t *testing.T) {
// elem := config.Root{}
2023-01-13 19:03:24 +00:00
// docs, err := LoadDocs("./bundle_config_docs.yml")
// require.NoError(t, err)
// schema, err := NewSchema(reflect.TypeOf(elem), docs)
2023-01-16 01:30:41 +00:00
// assert.NoError(t, err)
2023-01-13 19:03:24 +00:00
2023-01-16 01:30:41 +00:00
// jsonSchema, err := json.MarshalIndent(schema, " ", " ")
// assert.NoError(t, err)
2023-01-13 19:03:24 +00:00
2023-01-16 01:30:41 +00:00
// t.Log("[DEBUG] actual: ", string(jsonSchema))
// assert.True(t, false)
2023-01-16 01:30:41 +00:00
// }