mirror of https://github.com/databricks/cli.git
1438 lines
30 KiB
Go
1438 lines
30 KiB
Go
package schema
|
|
|
|
import (
|
|
"encoding/json"
|
|
"reflect"
|
|
"testing"
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
// TODO: Add a test that checks the primitive overrides for reference regexs wor.
|
|
|
|
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{}
|
|
|
|
schema, err := New(reflect.TypeOf(elem), nil)
|
|
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": {
|
|
"anyOf": [
|
|
{
|
|
"type": "number"
|
|
},
|
|
{
|
|
"type": "string",
|
|
"pattern": "\\$\\{([a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\.[a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\[[0-9]+\\])*)*(\\[[0-9]+\\])*)\\}"
|
|
}
|
|
]
|
|
},
|
|
"b": {
|
|
"type": "string"
|
|
}
|
|
},
|
|
"additionalProperties": false,
|
|
"required": [
|
|
"a",
|
|
"b"
|
|
]
|
|
}
|
|
},
|
|
"additionalProperties": false,
|
|
"required": [
|
|
"bar"
|
|
]
|
|
}
|
|
},
|
|
"additionalProperties": false,
|
|
"required": [
|
|
"foo"
|
|
]
|
|
}`
|
|
|
|
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{}
|
|
|
|
schema, err := New(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": {
|
|
"anyOf": [
|
|
{
|
|
"type": "number"
|
|
},
|
|
{
|
|
"type": "string",
|
|
"pattern": "\\$\\{([a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\.[a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\[[0-9]+\\])*)*(\\[[0-9]+\\])*)\\}"
|
|
}
|
|
]
|
|
}
|
|
}
|
|
},
|
|
"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{}
|
|
|
|
schema, err := New(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))
|
|
}
|
|
|
|
func TestMapOfStructSchema(t *testing.T) {
|
|
type Foo struct {
|
|
MyInt int `json:"my_int"`
|
|
}
|
|
|
|
var elem map[string]Foo
|
|
|
|
schema, err := New(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": {
|
|
"anyOf": [
|
|
{
|
|
"type": "number"
|
|
},
|
|
{
|
|
"type": "string",
|
|
"pattern": "\\$\\{([a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\.[a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\[[0-9]+\\])*)*(\\[[0-9]+\\])*)\\}"
|
|
}
|
|
]
|
|
}
|
|
},
|
|
"additionalProperties": false,
|
|
"required": [
|
|
"my_int"
|
|
]
|
|
}
|
|
}`
|
|
|
|
t.Log("[DEBUG] actual: ", string(jsonSchema))
|
|
t.Log("[DEBUG] expected: ", expected)
|
|
assert.Equal(t, expected, string(jsonSchema))
|
|
}
|
|
|
|
func TestMapOfMapSchema(t *testing.T) {
|
|
var elem map[string]map[string]int
|
|
|
|
schema, err := New(reflect.TypeOf(elem), nil)
|
|
assert.NoError(t, err)
|
|
|
|
jsonSchema, err := json.MarshalIndent(schema, " ", " ")
|
|
assert.NoError(t, err)
|
|
|
|
expected :=
|
|
`{
|
|
"type": "object",
|
|
"additionalProperties": {
|
|
"type": "object",
|
|
"additionalProperties": {
|
|
"anyOf": [
|
|
{
|
|
"type": "number"
|
|
},
|
|
{
|
|
"type": "string",
|
|
"pattern": "\\$\\{([a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\.[a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\[[0-9]+\\])*)*(\\[[0-9]+\\])*)\\}"
|
|
}
|
|
]
|
|
}
|
|
}
|
|
}`
|
|
|
|
t.Log("[DEBUG] actual: ", string(jsonSchema))
|
|
t.Log("[DEBUG] expected: ", expected)
|
|
assert.Equal(t, expected, string(jsonSchema))
|
|
}
|
|
|
|
func TestMapOfSliceSchema(t *testing.T) {
|
|
var elem map[string][]string
|
|
|
|
schema, err := New(reflect.TypeOf(elem), nil)
|
|
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 TestSliceOfSliceSchema(t *testing.T) {
|
|
var elem [][]string
|
|
|
|
schema, err := New(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
|
|
|
|
schema, err := New(reflect.TypeOf(elem), nil)
|
|
assert.NoError(t, err)
|
|
|
|
jsonSchema, err := json.MarshalIndent(schema, " ", " ")
|
|
assert.NoError(t, err)
|
|
|
|
expected :=
|
|
`{
|
|
"type": "array",
|
|
"items": {
|
|
"type": "object",
|
|
"additionalProperties": {
|
|
"anyOf": [
|
|
{
|
|
"type": "number"
|
|
},
|
|
{
|
|
"type": "string",
|
|
"pattern": "\\$\\{([a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\.[a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\[[0-9]+\\])*)*(\\[[0-9]+\\])*)\\}"
|
|
}
|
|
]
|
|
}
|
|
}
|
|
}`
|
|
|
|
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
|
|
|
|
schema, err := New(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": {
|
|
"anyOf": [
|
|
{
|
|
"type": "number"
|
|
},
|
|
{
|
|
"type": "string",
|
|
"pattern": "\\$\\{([a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\.[a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\[[0-9]+\\])*)*(\\[[0-9]+\\])*)\\}"
|
|
}
|
|
]
|
|
}
|
|
},
|
|
"additionalProperties": false,
|
|
"required": [
|
|
"my_int"
|
|
]
|
|
}
|
|
}`
|
|
|
|
t.Log("[DEBUG] actual: ", string(jsonSchema))
|
|
t.Log("[DEBUG] expected: ", expected)
|
|
assert.Equal(t, expected, string(jsonSchema))
|
|
}
|
|
|
|
func TestErrorWithTrace(t *testing.T) {
|
|
tracker := newTracker()
|
|
dummyType := reflect.TypeOf(struct{}{})
|
|
err := tracker.errWithTrace("with empty trace", "root")
|
|
assert.ErrorContains(t, err, "with empty trace. traversal trace: root")
|
|
|
|
tracker.push(dummyType, "resources")
|
|
err = tracker.errWithTrace("with depth = 1", "root")
|
|
assert.ErrorContains(t, err, "with depth = 1. traversal trace: root -> resources")
|
|
|
|
tracker.push(dummyType, "pipelines")
|
|
tracker.push(dummyType, "datasets")
|
|
err = tracker.errWithTrace("with depth = 4", "root")
|
|
assert.ErrorContains(t, err, "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{}
|
|
|
|
schema, err := New(reflect.TypeOf(elem), nil)
|
|
require.NoError(t, err)
|
|
|
|
jsonSchema, err := json.MarshalIndent(schema, " ", " ")
|
|
assert.NoError(t, err)
|
|
|
|
expectedSchema :=
|
|
`{
|
|
"type": "object",
|
|
"properties": {
|
|
"bar": {
|
|
"anyOf": [
|
|
{
|
|
"type": "number"
|
|
},
|
|
{
|
|
"type": "string",
|
|
"pattern": "\\$\\{([a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\.[a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\[[0-9]+\\])*)*(\\[[0-9]+\\])*)\\}"
|
|
}
|
|
]
|
|
}
|
|
},
|
|
"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{}
|
|
|
|
schema, err := New(reflect.TypeOf(elem), nil)
|
|
require.NoError(t, err)
|
|
|
|
jsonSchema, err := json.MarshalIndent(schema, " ", " ")
|
|
assert.NoError(t, err)
|
|
|
|
expectedSchema :=
|
|
`{
|
|
"type": "object",
|
|
"properties": {
|
|
"bar": {
|
|
"anyOf": [
|
|
{
|
|
"type": "number"
|
|
},
|
|
{
|
|
"type": "string",
|
|
"pattern": "\\$\\{([a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\.[a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\[[0-9]+\\])*)*(\\[[0-9]+\\])*)\\}"
|
|
}
|
|
]
|
|
}
|
|
},
|
|
"additionalProperties": false,
|
|
"required": [
|
|
"bar"
|
|
]
|
|
}`
|
|
|
|
t.Log("[DEBUG] actual: ", string(jsonSchema))
|
|
t.Log("[DEBUG] expected: ", expectedSchema)
|
|
|
|
assert.Equal(t, expectedSchema, string(jsonSchema))
|
|
}
|
|
|
|
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{}
|
|
|
|
schema, err := New(reflect.TypeOf(elem), nil)
|
|
require.NoError(t, err)
|
|
|
|
jsonSchema, err := json.MarshalIndent(schema, " ", " ")
|
|
assert.NoError(t, err)
|
|
|
|
expectedSchema :=
|
|
`{
|
|
"type": "object",
|
|
"properties": {
|
|
"bar": {
|
|
"type": "object",
|
|
"properties": {
|
|
"ptr_val2": {
|
|
"anyOf": [
|
|
{
|
|
"type": "number"
|
|
},
|
|
{
|
|
"type": "string",
|
|
"pattern": "\\$\\{([a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\.[a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\[[0-9]+\\])*)*(\\[[0-9]+\\])*)\\}"
|
|
}
|
|
]
|
|
}
|
|
},
|
|
"additionalProperties": false,
|
|
"required": [
|
|
"ptr_val2"
|
|
]
|
|
},
|
|
"float_val": {
|
|
"anyOf": [
|
|
{
|
|
"type": "number"
|
|
},
|
|
{
|
|
"type": "string",
|
|
"pattern": "\\$\\{([a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\.[a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\[[0-9]+\\])*)*(\\[[0-9]+\\])*)\\}"
|
|
}
|
|
]
|
|
},
|
|
"ptr_bar": {
|
|
"type": "object",
|
|
"properties": {
|
|
"ptr_val2": {
|
|
"anyOf": [
|
|
{
|
|
"type": "number"
|
|
},
|
|
{
|
|
"type": "string",
|
|
"pattern": "\\$\\{([a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\.[a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\[[0-9]+\\])*)*(\\[[0-9]+\\])*)\\}"
|
|
}
|
|
]
|
|
}
|
|
},
|
|
"additionalProperties": false,
|
|
"required": [
|
|
"ptr_val2"
|
|
]
|
|
},
|
|
"ptr_int": {
|
|
"anyOf": [
|
|
{
|
|
"type": "number"
|
|
},
|
|
{
|
|
"type": "string",
|
|
"pattern": "\\$\\{([a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\.[a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\[[0-9]+\\])*)*(\\[[0-9]+\\])*)\\}"
|
|
}
|
|
]
|
|
},
|
|
"ptr_string": {
|
|
"type": "string"
|
|
}
|
|
},
|
|
"additionalProperties": false,
|
|
"required": [
|
|
"ptr_int",
|
|
"ptr_string",
|
|
"float_val",
|
|
"ptr_bar",
|
|
"bar"
|
|
]
|
|
}`
|
|
|
|
t.Log("[DEBUG] actual: ", string(jsonSchema))
|
|
t.Log("[DEBUG] expected: ", expectedSchema)
|
|
|
|
assert.Equal(t, expectedSchema, string(jsonSchema))
|
|
}
|
|
|
|
func TestGenericSchema(t *testing.T) {
|
|
type Person struct {
|
|
Name string `json:"name"`
|
|
Age int `json:"age,omitempty"`
|
|
}
|
|
|
|
type Plot struct {
|
|
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"`
|
|
}
|
|
|
|
type Story struct {
|
|
Hero *Person `json:"hero"`
|
|
Villian Person `json:"villian,omitempty"`
|
|
Weddings []Wedding `json:"weddings"`
|
|
}
|
|
|
|
elem := Story{}
|
|
|
|
schema, err := New(reflect.TypeOf(elem), nil)
|
|
assert.NoError(t, err)
|
|
|
|
jsonSchema, err := json.MarshalIndent(schema, " ", " ")
|
|
assert.NoError(t, err)
|
|
|
|
expected :=
|
|
`{
|
|
"type": "object",
|
|
"properties": {
|
|
"hero": {
|
|
"type": "object",
|
|
"properties": {
|
|
"age": {
|
|
"anyOf": [
|
|
{
|
|
"type": "number"
|
|
},
|
|
{
|
|
"type": "string",
|
|
"pattern": "\\$\\{([a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\.[a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\[[0-9]+\\])*)*(\\[[0-9]+\\])*)\\}"
|
|
}
|
|
]
|
|
},
|
|
"name": {
|
|
"type": "string"
|
|
}
|
|
},
|
|
"additionalProperties": false,
|
|
"required": [
|
|
"name"
|
|
]
|
|
},
|
|
"villian": {
|
|
"type": "object",
|
|
"properties": {
|
|
"age": {
|
|
"anyOf": [
|
|
{
|
|
"type": "number"
|
|
},
|
|
{
|
|
"type": "string",
|
|
"pattern": "\\$\\{([a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\.[a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\[[0-9]+\\])*)*(\\[[0-9]+\\])*)\\}"
|
|
}
|
|
]
|
|
},
|
|
"name": {
|
|
"type": "string"
|
|
}
|
|
},
|
|
"additionalProperties": false,
|
|
"required": [
|
|
"name"
|
|
]
|
|
},
|
|
"weddings": {
|
|
"type": "array",
|
|
"items": {
|
|
"type": "object",
|
|
"properties": {
|
|
"bride": {
|
|
"type": "object",
|
|
"properties": {
|
|
"age": {
|
|
"anyOf": [
|
|
{
|
|
"type": "number"
|
|
},
|
|
{
|
|
"type": "string",
|
|
"pattern": "\\$\\{([a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\.[a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\[[0-9]+\\])*)*(\\[[0-9]+\\])*)\\}"
|
|
}
|
|
]
|
|
},
|
|
"name": {
|
|
"type": "string"
|
|
}
|
|
},
|
|
"additionalProperties": false,
|
|
"required": [
|
|
"name"
|
|
]
|
|
},
|
|
"groom": {
|
|
"type": "object",
|
|
"properties": {
|
|
"age": {
|
|
"anyOf": [
|
|
{
|
|
"type": "number"
|
|
},
|
|
{
|
|
"type": "string",
|
|
"pattern": "\\$\\{([a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\.[a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\[[0-9]+\\])*)*(\\[[0-9]+\\])*)\\}"
|
|
}
|
|
]
|
|
},
|
|
"name": {
|
|
"type": "string"
|
|
}
|
|
},
|
|
"additionalProperties": false,
|
|
"required": [
|
|
"name"
|
|
]
|
|
},
|
|
"plots": {
|
|
"type": "array",
|
|
"items": {
|
|
"type": "object",
|
|
"properties": {
|
|
"deaths": {
|
|
"type": "array",
|
|
"items": {
|
|
"type": "object",
|
|
"properties": {
|
|
"age": {
|
|
"anyOf": [
|
|
{
|
|
"type": "number"
|
|
},
|
|
{
|
|
"type": "string",
|
|
"pattern": "\\$\\{([a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\.[a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\[[0-9]+\\])*)*(\\[[0-9]+\\])*)\\}"
|
|
}
|
|
]
|
|
},
|
|
"name": {
|
|
"type": "string"
|
|
}
|
|
},
|
|
"additionalProperties": false,
|
|
"required": [
|
|
"name"
|
|
]
|
|
}
|
|
},
|
|
"murders": {
|
|
"type": "object",
|
|
"additionalProperties": {
|
|
"type": "object",
|
|
"properties": {
|
|
"age": {
|
|
"anyOf": [
|
|
{
|
|
"type": "number"
|
|
},
|
|
{
|
|
"type": "string",
|
|
"pattern": "\\$\\{([a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\.[a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\[[0-9]+\\])*)*(\\[[0-9]+\\])*)\\}"
|
|
}
|
|
]
|
|
},
|
|
"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"
|
|
]
|
|
}
|
|
}
|
|
},
|
|
"additionalProperties": false,
|
|
"required": [
|
|
"hero",
|
|
"weddings"
|
|
]
|
|
}`
|
|
|
|
t.Log("[DEBUG] actual: ", string(jsonSchema))
|
|
t.Log("[DEBUG] expected: ", expected)
|
|
assert.Equal(t, expected, string(jsonSchema))
|
|
}
|
|
|
|
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{}
|
|
|
|
schema, err := New(reflect.TypeOf(elem), nil)
|
|
require.NoError(t, err)
|
|
|
|
jsonSchema, err := json.MarshalIndent(schema, " ", " ")
|
|
assert.NoError(t, err)
|
|
|
|
expectedSchema :=
|
|
`{
|
|
"type": "object",
|
|
"properties": {
|
|
"apple": {
|
|
"anyOf": [
|
|
{
|
|
"type": "number"
|
|
},
|
|
{
|
|
"type": "string",
|
|
"pattern": "\\$\\{([a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\.[a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\[[0-9]+\\])*)*(\\[[0-9]+\\])*)\\}"
|
|
}
|
|
]
|
|
},
|
|
"bar": {
|
|
"anyOf": [
|
|
{
|
|
"type": "number"
|
|
},
|
|
{
|
|
"type": "string",
|
|
"pattern": "\\$\\{([a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\.[a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\[[0-9]+\\])*)*(\\[[0-9]+\\])*)\\}"
|
|
}
|
|
]
|
|
},
|
|
"papaya": {
|
|
"type": "object",
|
|
"properties": {
|
|
"a": {
|
|
"anyOf": [
|
|
{
|
|
"type": "number"
|
|
},
|
|
{
|
|
"type": "string",
|
|
"pattern": "\\$\\{([a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\.[a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\[[0-9]+\\])*)*(\\[[0-9]+\\])*)\\}"
|
|
}
|
|
]
|
|
},
|
|
"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))
|
|
}
|
|
|
|
func TestDocIngestionForObject(t *testing.T) {
|
|
docs := &Docs{
|
|
Description: "docs for root",
|
|
Properties: map[string]*Docs{
|
|
"my_struct": {
|
|
Description: "docs for my struct",
|
|
Properties: map[string]*Docs{
|
|
"a": {
|
|
Description: "docs for a",
|
|
},
|
|
"c": {
|
|
Description: "docs for c which does not exist on my_struct",
|
|
},
|
|
},
|
|
},
|
|
},
|
|
}
|
|
|
|
type MyStruct struct {
|
|
A string `json:"a"`
|
|
B int `json:"b"`
|
|
}
|
|
|
|
type Root struct {
|
|
MyStruct *MyStruct `json:"my_struct"`
|
|
}
|
|
|
|
elem := Root{}
|
|
|
|
schema, err := New(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_struct": {
|
|
"type": "object",
|
|
"description": "docs for my struct",
|
|
"properties": {
|
|
"a": {
|
|
"type": "string",
|
|
"description": "docs for a"
|
|
},
|
|
"b": {
|
|
"anyOf": [
|
|
{
|
|
"type": "number"
|
|
},
|
|
{
|
|
"type": "string",
|
|
"pattern": "\\$\\{([a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\.[a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\[[0-9]+\\])*)*(\\[[0-9]+\\])*)\\}"
|
|
}
|
|
]
|
|
}
|
|
},
|
|
"additionalProperties": false,
|
|
"required": [
|
|
"a",
|
|
"b"
|
|
]
|
|
}
|
|
},
|
|
"additionalProperties": false,
|
|
"required": [
|
|
"my_struct"
|
|
]
|
|
}`
|
|
|
|
t.Log("[DEBUG] actual: ", string(jsonSchema))
|
|
t.Log("[DEBUG] expected: ", expectedSchema)
|
|
|
|
assert.Equal(t, expectedSchema, string(jsonSchema))
|
|
}
|
|
|
|
func TestDocIngestionForSlice(t *testing.T) {
|
|
docs := &Docs{
|
|
Description: "docs for root",
|
|
Properties: map[string]*Docs{
|
|
"my_slice": {
|
|
Description: "docs for my slice",
|
|
Items: &Docs{
|
|
Properties: map[string]*Docs{
|
|
"guava": {
|
|
Description: "docs for guava",
|
|
},
|
|
"pineapple": {
|
|
Description: "docs for pineapple",
|
|
},
|
|
"watermelon": {
|
|
Description: "docs for watermelon which does not exist in schema",
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
}
|
|
|
|
type Bar struct {
|
|
Guava int `json:"guava"`
|
|
Pineapple int `json:"pineapple"`
|
|
}
|
|
|
|
type Root struct {
|
|
MySlice []Bar `json:"my_slice"`
|
|
}
|
|
|
|
elem := Root{}
|
|
|
|
schema, err := New(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_slice": {
|
|
"type": "array",
|
|
"description": "docs for my slice",
|
|
"items": {
|
|
"type": "object",
|
|
"properties": {
|
|
"guava": {
|
|
"description": "docs for guava",
|
|
"anyOf": [
|
|
{
|
|
"type": "number"
|
|
},
|
|
{
|
|
"type": "string",
|
|
"pattern": "\\$\\{([a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\.[a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\[[0-9]+\\])*)*(\\[[0-9]+\\])*)\\}"
|
|
}
|
|
]
|
|
},
|
|
"pineapple": {
|
|
"description": "docs for pineapple",
|
|
"anyOf": [
|
|
{
|
|
"type": "number"
|
|
},
|
|
{
|
|
"type": "string",
|
|
"pattern": "\\$\\{([a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\.[a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\[[0-9]+\\])*)*(\\[[0-9]+\\])*)\\}"
|
|
}
|
|
]
|
|
}
|
|
},
|
|
"additionalProperties": false,
|
|
"required": [
|
|
"guava",
|
|
"pineapple"
|
|
]
|
|
}
|
|
}
|
|
},
|
|
"additionalProperties": false,
|
|
"required": [
|
|
"my_slice"
|
|
]
|
|
}`
|
|
|
|
t.Log("[DEBUG] actual: ", string(jsonSchema))
|
|
t.Log("[DEBUG] expected: ", expectedSchema)
|
|
|
|
assert.Equal(t, expectedSchema, string(jsonSchema))
|
|
}
|
|
|
|
func TestDocIngestionForMap(t *testing.T) {
|
|
docs := &Docs{
|
|
Description: "docs for root",
|
|
Properties: map[string]*Docs{
|
|
"my_map": {
|
|
Description: "docs for my map",
|
|
AdditionalProperties: &Docs{
|
|
Properties: map[string]*Docs{
|
|
"apple": {
|
|
Description: "docs for apple",
|
|
},
|
|
"mango": {
|
|
Description: "docs for mango",
|
|
},
|
|
"watermelon": {
|
|
Description: "docs for watermelon which does not exist in schema",
|
|
},
|
|
"papaya": {
|
|
Description: "docs for papaya which does not exist in schema",
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
}
|
|
|
|
type Foo struct {
|
|
Apple int `json:"apple"`
|
|
Mango int `json:"mango"`
|
|
}
|
|
|
|
type Root struct {
|
|
MyMap map[string]*Foo `json:"my_map"`
|
|
}
|
|
|
|
elem := Root{}
|
|
|
|
schema, err := New(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",
|
|
"properties": {
|
|
"apple": {
|
|
"description": "docs for apple",
|
|
"anyOf": [
|
|
{
|
|
"type": "number"
|
|
},
|
|
{
|
|
"type": "string",
|
|
"pattern": "\\$\\{([a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\.[a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\[[0-9]+\\])*)*(\\[[0-9]+\\])*)\\}"
|
|
}
|
|
]
|
|
},
|
|
"mango": {
|
|
"description": "docs for mango",
|
|
"anyOf": [
|
|
{
|
|
"type": "number"
|
|
},
|
|
{
|
|
"type": "string",
|
|
"pattern": "\\$\\{([a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\.[a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\[[0-9]+\\])*)*(\\[[0-9]+\\])*)\\}"
|
|
}
|
|
]
|
|
}
|
|
},
|
|
"additionalProperties": false,
|
|
"required": [
|
|
"apple",
|
|
"mango"
|
|
]
|
|
}
|
|
}
|
|
},
|
|
"additionalProperties": false,
|
|
"required": [
|
|
"my_map"
|
|
]
|
|
}`
|
|
|
|
t.Log("[DEBUG] actual: ", string(jsonSchema))
|
|
t.Log("[DEBUG] expected: ", expectedSchema)
|
|
|
|
assert.Equal(t, expectedSchema, string(jsonSchema))
|
|
}
|
|
|
|
func TestDocIngestionForTopLevelPrimitive(t *testing.T) {
|
|
docs := &Docs{
|
|
Description: "docs for root",
|
|
Properties: map[string]*Docs{
|
|
"my_val": {
|
|
Description: "docs for my val",
|
|
},
|
|
},
|
|
}
|
|
|
|
type Root struct {
|
|
MyVal int `json:"my_val"`
|
|
}
|
|
|
|
elem := Root{}
|
|
|
|
schema, err := New(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_val": {
|
|
"description": "docs for my val",
|
|
"anyOf": [
|
|
{
|
|
"type": "number"
|
|
},
|
|
{
|
|
"type": "string",
|
|
"pattern": "\\$\\{([a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\.[a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\[[0-9]+\\])*)*(\\[[0-9]+\\])*)\\}"
|
|
}
|
|
]
|
|
}
|
|
},
|
|
"additionalProperties": false,
|
|
"required": [
|
|
"my_val"
|
|
]
|
|
}`
|
|
|
|
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 := New(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 := New(reflect.TypeOf(elem), nil)
|
|
assert.ErrorContains(t, err, "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 := New(reflect.TypeOf(elem), nil)
|
|
assert.ErrorContains(t, err, "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 := New(reflect.TypeOf(elem), nil)
|
|
assert.NoError(t, err)
|
|
|
|
jsonSchema, err := json.MarshalIndent(schema, " ", " ")
|
|
assert.NoError(t, err)
|
|
|
|
expected :=
|
|
`{
|
|
"type": "object",
|
|
"properties": {
|
|
"apple": {
|
|
"anyOf": [
|
|
{
|
|
"type": "number"
|
|
},
|
|
{
|
|
"type": "string",
|
|
"pattern": "\\$\\{([a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\.[a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\[[0-9]+\\])*)*(\\[[0-9]+\\])*)\\}"
|
|
}
|
|
]
|
|
},
|
|
"mango": {}
|
|
},
|
|
"additionalProperties": false,
|
|
"required": [
|
|
"apple",
|
|
"mango"
|
|
]
|
|
}`
|
|
|
|
t.Log("[DEBUG] actual: ", string(jsonSchema))
|
|
t.Log("[DEBUG] expected: ", expected)
|
|
assert.Equal(t, expected, string(jsonSchema))
|
|
}
|
|
|
|
func TestBundleReadOnlytag(t *testing.T) {
|
|
type Pokemon struct {
|
|
Pikachu string `json:"pikachu" bundle:"readonly"`
|
|
Raichu string `json:"raichu"`
|
|
}
|
|
|
|
type Foo struct {
|
|
Pokemon *Pokemon `json:"pokemon"`
|
|
Apple int `json:"apple"`
|
|
Mango string `json:"mango" bundle:"readonly"`
|
|
}
|
|
|
|
elem := Foo{}
|
|
|
|
schema, err := New(reflect.TypeOf(elem), nil)
|
|
assert.NoError(t, err)
|
|
|
|
jsonSchema, err := json.MarshalIndent(schema, " ", " ")
|
|
assert.NoError(t, err)
|
|
|
|
expected :=
|
|
`{
|
|
"type": "object",
|
|
"properties": {
|
|
"apple": {
|
|
"anyOf": [
|
|
{
|
|
"type": "number"
|
|
},
|
|
{
|
|
"type": "string",
|
|
"pattern": "\\$\\{([a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\.[a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\[[0-9]+\\])*)*(\\[[0-9]+\\])*)\\}"
|
|
}
|
|
]
|
|
},
|
|
"pokemon": {
|
|
"type": "object",
|
|
"properties": {
|
|
"raichu": {
|
|
"type": "string"
|
|
}
|
|
},
|
|
"additionalProperties": false,
|
|
"required": [
|
|
"raichu"
|
|
]
|
|
}
|
|
},
|
|
"additionalProperties": false,
|
|
"required": [
|
|
"pokemon",
|
|
"apple"
|
|
]
|
|
}`
|
|
|
|
t.Log("[DEBUG] actual: ", string(jsonSchema))
|
|
t.Log("[DEBUG] expected: ", expected)
|
|
assert.Equal(t, expected, string(jsonSchema))
|
|
}
|
|
|
|
func TestBundleInternalTag(t *testing.T) {
|
|
type Pokemon struct {
|
|
Pikachu string `json:"pikachu" bundle:"internal"`
|
|
Raichu string `json:"raichu"`
|
|
}
|
|
|
|
type Foo struct {
|
|
Pokemon *Pokemon `json:"pokemon"`
|
|
Apple int `json:"apple"`
|
|
Mango string `json:"mango" bundle:"internal"`
|
|
}
|
|
|
|
elem := Foo{}
|
|
|
|
schema, err := New(reflect.TypeOf(elem), nil)
|
|
assert.NoError(t, err)
|
|
|
|
jsonSchema, err := json.MarshalIndent(schema, " ", " ")
|
|
assert.NoError(t, err)
|
|
|
|
expected :=
|
|
`{
|
|
"type": "object",
|
|
"properties": {
|
|
"apple": {
|
|
"anyOf": [
|
|
{
|
|
"type": "number"
|
|
},
|
|
{
|
|
"type": "string",
|
|
"pattern": "\\$\\{([a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\.[a-zA-Z]+([-_]?[a-zA-Z0-9]+)*(\\[[0-9]+\\])*)*(\\[[0-9]+\\])*)\\}"
|
|
}
|
|
]
|
|
},
|
|
"pokemon": {
|
|
"type": "object",
|
|
"properties": {
|
|
"raichu": {
|
|
"type": "string"
|
|
}
|
|
},
|
|
"additionalProperties": false,
|
|
"required": [
|
|
"raichu"
|
|
]
|
|
}
|
|
},
|
|
"additionalProperties": false,
|
|
"required": [
|
|
"pokemon",
|
|
"apple"
|
|
]
|
|
}`
|
|
|
|
t.Log("[DEBUG] actual: ", string(jsonSchema))
|
|
t.Log("[DEBUG] expected: ", expected)
|
|
assert.Equal(t, expected, string(jsonSchema))
|
|
}
|