You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
506 lines
9.1 KiB
506 lines
9.1 KiB
'use strict'
|
|
|
|
const { test } = require('tap')
|
|
const build = require('..')
|
|
|
|
test('object with multiple types field', (t) => {
|
|
t.plan(2)
|
|
|
|
const schema = {
|
|
title: 'object with multiple types field',
|
|
type: 'object',
|
|
properties: {
|
|
str: {
|
|
oneOf: [{
|
|
type: 'string'
|
|
}, {
|
|
type: 'boolean'
|
|
}]
|
|
}
|
|
}
|
|
}
|
|
const stringify = build(schema)
|
|
|
|
t.equal(stringify({ str: 'string' }), '{"str":"string"}')
|
|
t.equal(stringify({ str: true }), '{"str":true}')
|
|
})
|
|
|
|
test('object with field of type object or null', (t) => {
|
|
t.plan(2)
|
|
|
|
const schema = {
|
|
title: 'object with field of type object or null',
|
|
type: 'object',
|
|
properties: {
|
|
prop: {
|
|
oneOf: [{
|
|
type: 'object',
|
|
properties: {
|
|
str: {
|
|
type: 'string'
|
|
}
|
|
}
|
|
}, {
|
|
type: 'null'
|
|
}]
|
|
}
|
|
}
|
|
}
|
|
const stringify = build(schema)
|
|
|
|
t.equal(stringify({ prop: null }), '{"prop":null}')
|
|
|
|
t.equal(stringify({
|
|
prop: {
|
|
str: 'string', remove: 'this'
|
|
}
|
|
}), '{"prop":{"str":"string"}}')
|
|
})
|
|
|
|
test('object with field of type object or array', (t) => {
|
|
t.plan(2)
|
|
|
|
const schema = {
|
|
title: 'object with field of type object or array',
|
|
type: 'object',
|
|
properties: {
|
|
prop: {
|
|
oneOf: [{
|
|
type: 'object',
|
|
properties: {},
|
|
additionalProperties: true
|
|
}, {
|
|
type: 'array',
|
|
items: {
|
|
type: 'string'
|
|
}
|
|
}]
|
|
}
|
|
}
|
|
}
|
|
const stringify = build(schema)
|
|
|
|
t.equal(stringify({
|
|
prop: { str: 'string' }
|
|
}), '{"prop":{"str":"string"}}')
|
|
|
|
t.equal(stringify({
|
|
prop: ['string']
|
|
}), '{"prop":["string"]}')
|
|
})
|
|
|
|
test('object with field of type string and coercion disable ', (t) => {
|
|
t.plan(1)
|
|
|
|
const schema = {
|
|
title: 'object with field of type string',
|
|
type: 'object',
|
|
properties: {
|
|
str: {
|
|
oneOf: [{
|
|
type: 'string'
|
|
}]
|
|
}
|
|
}
|
|
}
|
|
const stringify = build(schema)
|
|
|
|
const value = stringify({
|
|
str: 1
|
|
})
|
|
t.equal(value, '{"str":null}')
|
|
})
|
|
|
|
test('object with field of type string and coercion enable ', (t) => {
|
|
t.plan(1)
|
|
|
|
const schema = {
|
|
title: 'object with field of type string',
|
|
type: 'object',
|
|
properties: {
|
|
str: {
|
|
oneOf: [{
|
|
type: 'string'
|
|
}]
|
|
}
|
|
}
|
|
}
|
|
|
|
const options = {
|
|
ajv: {
|
|
coerceTypes: true
|
|
}
|
|
}
|
|
const stringify = build(schema, options)
|
|
|
|
const value = stringify({
|
|
str: 1
|
|
})
|
|
t.equal(value, '{"str":"1"}')
|
|
})
|
|
|
|
test('object with field with type union of multiple objects', (t) => {
|
|
t.plan(2)
|
|
|
|
const schema = {
|
|
title: 'object with oneOf property value containing objects',
|
|
type: 'object',
|
|
properties: {
|
|
oneOfSchema: {
|
|
oneOf: [
|
|
{
|
|
type: 'object',
|
|
properties: {
|
|
baz: { type: 'number' }
|
|
},
|
|
required: ['baz']
|
|
},
|
|
{
|
|
type: 'object',
|
|
properties: {
|
|
bar: { type: 'string' }
|
|
},
|
|
required: ['bar']
|
|
}
|
|
]
|
|
}
|
|
},
|
|
required: ['oneOfSchema']
|
|
}
|
|
|
|
const stringify = build(schema)
|
|
|
|
t.equal(stringify({ oneOfSchema: { baz: 5 } }), '{"oneOfSchema":{"baz":5}}')
|
|
|
|
t.equal(stringify({ oneOfSchema: { bar: 'foo' } }), '{"oneOfSchema":{"bar":"foo"}}')
|
|
})
|
|
|
|
test('null value in schema', (t) => {
|
|
t.plan(0)
|
|
|
|
const schema = {
|
|
title: 'schema with null child',
|
|
type: 'string',
|
|
nullable: true,
|
|
enum: [null]
|
|
}
|
|
|
|
build(schema)
|
|
})
|
|
|
|
test('oneOf and $ref together', (t) => {
|
|
t.plan(2)
|
|
|
|
const schema = {
|
|
type: 'object',
|
|
properties: {
|
|
cs: {
|
|
oneOf: [
|
|
{
|
|
$ref: '#/definitions/Option'
|
|
},
|
|
{
|
|
type: 'boolean'
|
|
}
|
|
]
|
|
}
|
|
},
|
|
definitions: {
|
|
Option: {
|
|
type: 'string'
|
|
}
|
|
}
|
|
}
|
|
|
|
const stringify = build(schema)
|
|
|
|
t.equal(stringify({ cs: 'franco' }), '{"cs":"franco"}')
|
|
|
|
t.equal(stringify({ cs: true }), '{"cs":true}')
|
|
})
|
|
|
|
test('oneOf and $ref: 2 levels are fine', (t) => {
|
|
t.plan(1)
|
|
|
|
const schema = {
|
|
type: 'object',
|
|
properties: {
|
|
cs: {
|
|
oneOf: [
|
|
{
|
|
$ref: '#/definitions/Option'
|
|
},
|
|
{
|
|
type: 'boolean'
|
|
}
|
|
]
|
|
}
|
|
},
|
|
definitions: {
|
|
Option: {
|
|
oneOf: [
|
|
{
|
|
type: 'number'
|
|
},
|
|
{
|
|
type: 'boolean'
|
|
}
|
|
]
|
|
}
|
|
}
|
|
}
|
|
|
|
const stringify = build(schema)
|
|
const value = stringify({
|
|
cs: 3
|
|
})
|
|
t.equal(value, '{"cs":3}')
|
|
})
|
|
|
|
test('oneOf and $ref: multiple levels should throw at build.', (t) => {
|
|
t.plan(3)
|
|
|
|
const schema = {
|
|
type: 'object',
|
|
properties: {
|
|
cs: {
|
|
oneOf: [
|
|
{
|
|
$ref: '#/definitions/Option'
|
|
},
|
|
{
|
|
type: 'boolean'
|
|
}
|
|
]
|
|
}
|
|
},
|
|
definitions: {
|
|
Option: {
|
|
oneOf: [
|
|
{
|
|
$ref: '#/definitions/Option2'
|
|
},
|
|
{
|
|
type: 'string'
|
|
}
|
|
]
|
|
},
|
|
Option2: {
|
|
type: 'number'
|
|
}
|
|
}
|
|
}
|
|
|
|
const stringify = build(schema)
|
|
|
|
t.equal(stringify({ cs: 3 }), '{"cs":3}')
|
|
t.equal(stringify({ cs: true }), '{"cs":true}')
|
|
t.equal(stringify({ cs: 'pippo' }), '{"cs":"pippo"}')
|
|
})
|
|
|
|
test('oneOf and $ref - multiple external $ref', (t) => {
|
|
t.plan(2)
|
|
|
|
const externalSchema = {
|
|
external: {
|
|
definitions: {
|
|
def: {
|
|
type: 'object',
|
|
properties: {
|
|
prop: { oneOf: [{ $ref: 'external2#/definitions/other' }] }
|
|
}
|
|
}
|
|
}
|
|
},
|
|
external2: {
|
|
definitions: {
|
|
internal: {
|
|
type: 'string'
|
|
},
|
|
other: {
|
|
type: 'object',
|
|
properties: {
|
|
prop2: { $ref: '#/definitions/internal' }
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
const schema = {
|
|
title: 'object with $ref',
|
|
type: 'object',
|
|
properties: {
|
|
obj: {
|
|
$ref: 'external#/definitions/def'
|
|
}
|
|
}
|
|
}
|
|
|
|
const object = {
|
|
obj: {
|
|
prop: {
|
|
prop2: 'test'
|
|
}
|
|
}
|
|
}
|
|
|
|
const stringify = build(schema, { schema: externalSchema })
|
|
const output = stringify(object)
|
|
|
|
JSON.parse(output)
|
|
t.pass()
|
|
|
|
t.equal(output, '{"obj":{"prop":{"prop2":"test"}}}')
|
|
})
|
|
|
|
test('oneOf with enum with more than 100 entries', (t) => {
|
|
t.plan(1)
|
|
|
|
const schema = {
|
|
title: 'type array that may have one of declared items',
|
|
type: 'array',
|
|
items: {
|
|
oneOf: [
|
|
{
|
|
type: 'string',
|
|
enum: ['EUR', 'USD', ...(new Set([...new Array(200)].map(() => Math.random().toString(36).substr(2, 3)))).values()]
|
|
},
|
|
{ type: 'null' }
|
|
]
|
|
}
|
|
}
|
|
const stringify = build(schema)
|
|
|
|
const value = stringify(['EUR', 'USD', null])
|
|
t.equal(value, '["EUR","USD",null]')
|
|
})
|
|
|
|
test('oneOf object with field of type string with format or null', (t) => {
|
|
t.plan(1)
|
|
|
|
const toStringify = new Date()
|
|
|
|
const withOneOfSchema = {
|
|
type: 'object',
|
|
properties: {
|
|
prop: {
|
|
oneOf: [{
|
|
type: 'string',
|
|
format: 'date-time'
|
|
}, {
|
|
type: 'null'
|
|
}]
|
|
}
|
|
}
|
|
}
|
|
|
|
const withOneOfStringify = build(withOneOfSchema)
|
|
|
|
t.equal(withOneOfStringify({
|
|
prop: toStringify
|
|
}), `{"prop":"${toStringify.toISOString()}"}`)
|
|
})
|
|
|
|
test('one array item match oneOf types', (t) => {
|
|
t.plan(1)
|
|
|
|
const schema = {
|
|
type: 'object',
|
|
additionalProperties: false,
|
|
required: ['data'],
|
|
properties: {
|
|
data: {
|
|
type: 'array',
|
|
minItems: 1,
|
|
items: {
|
|
oneOf: [
|
|
{
|
|
type: 'string'
|
|
},
|
|
{
|
|
type: 'number'
|
|
}
|
|
]
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
const stringify = build(schema)
|
|
|
|
const responseWithMappedType = stringify({
|
|
data: [false, 'foo']
|
|
})
|
|
|
|
t.equal('{"data":["foo"]}', responseWithMappedType)
|
|
})
|
|
|
|
test('some array items match oneOf types', (t) => {
|
|
t.plan(1)
|
|
|
|
const schema = {
|
|
type: 'object',
|
|
additionalProperties: false,
|
|
required: ['data'],
|
|
properties: {
|
|
data: {
|
|
type: 'array',
|
|
minItems: 1,
|
|
items: {
|
|
oneOf: [
|
|
{
|
|
type: 'string'
|
|
},
|
|
{
|
|
type: 'number'
|
|
}
|
|
]
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
const stringify = build(schema)
|
|
|
|
const responseWithMappedTypes = stringify({
|
|
data: [false, 'foo', true, 5]
|
|
})
|
|
|
|
t.equal('{"data":["foo",5]}', responseWithMappedTypes)
|
|
})
|
|
|
|
test('all array items does not match oneOf types', (t) => {
|
|
t.plan(1)
|
|
|
|
const schema = {
|
|
type: 'object',
|
|
additionalProperties: false,
|
|
required: ['data'],
|
|
properties: {
|
|
data: {
|
|
type: 'array',
|
|
minItems: 1,
|
|
items: {
|
|
oneOf: [
|
|
{
|
|
type: 'string'
|
|
},
|
|
{
|
|
type: 'number'
|
|
}
|
|
]
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
const stringify = build(schema)
|
|
|
|
const emptyResponse = stringify({
|
|
data: [null, false, true, undefined, [], {}]
|
|
})
|
|
|
|
t.equal('{"data":[]}', emptyResponse)
|
|
})
|