From e0f72dcb873306f7ff4815dfdf769213f16a1e9b Mon Sep 17 00:00:00 2001 From: Roman Hotsiy Date: Wed, 27 Apr 2022 12:33:06 +0800 Subject: [PATCH] fix: support circular oneOf/anyOf local props --- src/traverse.js | 30 ++++++++++++++++++++++++++---- src/utils.js | 2 +- test/integration.spec.js | 34 ++++++++++++++++++++++++++++++++++ 3 files changed, 61 insertions(+), 5 deletions(-) diff --git a/src/traverse.js b/src/traverse.js index bc9d2a8..b8d220d 100644 --- a/src/traverse.js +++ b/src/traverse.js @@ -112,14 +112,19 @@ export function traverse(schema, options, spec, context) { writeOnly: schema.writeOnly }, schema.oneOf[0]); - return ( - tryInferExample(schema) || traverse(firstOneOf, options, spec, context) - ); + return traverseOneOrAnyOf(schema, firstOneOf) } if (schema.anyOf && schema.anyOf.length) { popSchemaStack(seenSchemasStack, context); - return tryInferExample(schema) || traverse(schema.anyOf[0], options, spec, context); + + // Make sure to pass down readOnly and writeOnly annotations from the parent + const firstAnyOf = Object.assign({ + readOnly: schema.readOnly, + writeOnly: schema.writeOnly + }, schema.anyOf[0]); + + return traverseOneOrAnyOf(schema, firstAnyOf) } if (schema.if && schema.then) { @@ -151,4 +156,21 @@ export function traverse(schema, options, spec, context) { writeOnly: schema.writeOnly, type: type }; + + function traverseOneOrAnyOf(schema, selectedSubSchema) { + const inferred = tryInferExample(schema); + if (inferred !== undefined) { + return inferred; + } + + const localExample = traverse({...schema, oneOf: undefined, anyOf: undefined }, options, spec, context); + const subSchemaExample = traverse(selectedSubSchema, options, spec, context); + + if (typeof localExample.value === 'object' && typeof subSchemaExample.value === 'object') { + const mergedExample = mergeDeep(localExample.value, subSchemaExample.value); + return {...subSchemaExample, value: mergedExample }; + } + + return subSchemaExample; + } } diff --git a/src/utils.js b/src/utils.js index 9a68fc7..04f90aa 100644 --- a/src/utils.js +++ b/src/utils.js @@ -32,7 +32,7 @@ export function mergeDeep(...objects) { const isObject = obj => obj && typeof obj === 'object'; return objects.reduce((prev, obj) => { - Object.keys(obj).forEach(key => { + Object.keys(obj || {}).forEach(key => { const pVal = prev[key]; const oVal = obj[key]; diff --git a/test/integration.spec.js b/test/integration.spec.js index 3a3b94d..a87febd 100644 --- a/test/integration.spec.js +++ b/test/integration.spec.js @@ -522,6 +522,40 @@ describe('Integration', function() { expected = 0; expect(result).to.equal(expected); }); + + it('should work with nested circular oneOf', function () { + result = OpenAPISampler.sample({ $ref: '#/definitions/A' }, {}, { + definitions: { + A: { + properties: { + a: { type: 'string' } + }, + oneOf: [ + { $ref: '#/definitions/A' } + ] + } + } + }); + expected = { a: 'string' } + expect(result).to.deep.equal(expected); + }); + + it('should work with nested circular anyOf', function () { + result = OpenAPISampler.sample({ $ref: '#/definitions/A' }, {}, { + definitions: { + A: { + properties: { + a: { type: 'string' } + }, + anyOf: [ + { $ref: '#/definitions/A' } + ] + } + } + }); + expected = { a: 'string' } + expect(result).to.deep.equal(expected); + }); }); describe('inferring type from root schema', function() {