From fe74f69912d86732f8cd5cee800163f7095c2f99 Mon Sep 17 00:00:00 2001 From: Peter Mouland Date: Sun, 26 Sep 2021 11:18:47 +0100 Subject: [PATCH] fix: nested refs or swagger spec --- lib/spec/openapi/utils.js | 14 +------------- lib/spec/swagger/index.js | 7 +++++-- lib/spec/swagger/utils.js | 29 ++++++++++++++++++++++++----- lib/util/common.js | 15 ++++++++++++++- test/spec/swagger/refs.js | 35 +++++++++++++++++++++++++++++++++++ 5 files changed, 79 insertions(+), 21 deletions(-) diff --git a/lib/spec/openapi/utils.js b/lib/spec/openapi/utils.js index 79761295..892adfa4 100644 --- a/lib/spec/openapi/utils.js +++ b/lib/spec/openapi/utils.js @@ -1,6 +1,6 @@ 'use strict' -const { readPackageJson, formatParamUrl, resolveLocalRef } = require('../../util/common') +const { readPackageJson, formatParamUrl, resolveLocalRef, convertExamplesArrayToObject } = require('../../util/common') const { xResponseDescription, xConsume } = require('../../constants') const { rawRequired } = require('../../symbols') @@ -96,18 +96,6 @@ function transformDefsToComponents (jsonSchema) { return jsonSchema } -function convertExamplesArrayToObject (examples) { - return examples.reduce((examplesObject, example, index) => { - if (typeof example === 'object') { - examplesObject['example' + (index + 1)] = { value: example } - } else { - examplesObject[example] = { value: example } - } - - return examplesObject - }, {}) -} - // For supported keys read: // https://swagger.io/docs/specification/describing-parameters/ function plainJsonObjectToOpenapi3 (container, jsonSchema, externalSchemas) { diff --git a/lib/spec/swagger/index.js b/lib/spec/swagger/index.js index 485a58dc..0f29c5b5 100644 --- a/lib/spec/swagger/index.js +++ b/lib/spec/swagger/index.js @@ -2,7 +2,7 @@ const yaml = require('js-yaml') const { shouldRouteHide } = require('../../util/common') -const { prepareDefaultOptions, prepareSwaggerObject, prepareSwaggerMethod, normalizeUrl } = require('./utils') +const { prepareDefaultOptions, prepareSwaggerObject, prepareSwaggerMethod, normalizeUrl, transformDefs } = require('./utils') module.exports = function (opts, cache, routes, Ref, done) { let ref @@ -28,7 +28,10 @@ module.exports = function (opts, cache, routes, Ref, done) { // The $ids are needed by Ref() to check the URI so we need // to remove them at the end of the process Object.values(swaggerObject.definitions) - .forEach(_ => { delete _.$id }) + .forEach(_ => { + delete _.$id + if (_.properties) transformDefs(_.properties) + }) swaggerObject.paths = {} for (const route of routes) { diff --git a/lib/spec/swagger/utils.js b/lib/spec/swagger/utils.js index 2a465d6a..b560dd36 100644 --- a/lib/spec/swagger/utils.js +++ b/lib/spec/swagger/utils.js @@ -1,6 +1,6 @@ 'use strict' -const { readPackageJson, formatParamUrl, resolveLocalRef } = require('../../util/common') +const { readPackageJson, formatParamUrl, resolveLocalRef, convertExamplesArrayToObject } = require('../../util/common') const { xResponseDescription, xConsume } = require('../../constants') function prepareDefaultOptions (opts) { @@ -157,9 +157,9 @@ function isConsumesFormOnly (schema) { const consumes = schema.consumes return ( consumes && - consumes.length === 1 && - (consumes[0] === 'application/x-www-form-urlencoded' || - consumes[0] === 'multipart/form-data') + consumes.length === 1 && + (consumes[0] === 'application/x-www-form-urlencoded' || + consumes[0] === 'multipart/form-data') ) } @@ -266,9 +266,28 @@ function prepareSwaggerMethod (schema, ref, swaggerObject) { return swaggerMethod } +function transformDefs (jsonSchema) { + if (typeof jsonSchema === 'object' && jsonSchema !== null) { + Object.keys(jsonSchema).forEach(function (key) { + if (key === '$ref') { + jsonSchema[key] = `#/definitions/${jsonSchema[key]}` + } else if (key === 'examples' && Array.isArray(jsonSchema[key]) && (jsonSchema[key].length > 1)) { + jsonSchema.examples = convertExamplesArrayToObject(jsonSchema.examples) + } else if (key === 'examples' && Array.isArray(jsonSchema[key]) && (jsonSchema[key].length === 1)) { + jsonSchema.example = jsonSchema[key][0] + delete jsonSchema[key] + } else { + jsonSchema[key] = transformDefs(jsonSchema[key]) + } + }) + } + return jsonSchema +} + module.exports = { prepareDefaultOptions, prepareSwaggerObject, prepareSwaggerMethod, - normalizeUrl + normalizeUrl, + transformDefs } diff --git a/lib/util/common.js b/lib/util/common.js index e0ba2337..84f73aa0 100644 --- a/lib/util/common.js +++ b/lib/util/common.js @@ -129,11 +129,24 @@ function resolveSwaggerFunction (opts, cache, routes, Ref, done) { } } +function convertExamplesArrayToObject (examples) { + return examples.reduce((examplesObject, example, index) => { + if (typeof example === 'object') { + examplesObject['example' + (index + 1)] = { value: example } + } else { + examplesObject[example] = { value: example } + } + + return examplesObject + }, {}) +} + module.exports = { addHook, shouldRouteHide, readPackageJson, formatParamUrl, resolveLocalRef, - resolveSwaggerFunction + resolveSwaggerFunction, + convertExamplesArrayToObject } diff --git a/test/spec/swagger/refs.js b/test/spec/swagger/refs.js index 9acc349d..1e95a5f7 100644 --- a/test/spec/swagger/refs.js +++ b/test/spec/swagger/refs.js @@ -64,3 +64,38 @@ test('support $ref schema', async t => { await Swagger.validate(res.json()) t.pass('valid swagger object') }) + +test('support nested $ref schema : complex case', async (t) => { + const fastify = Fastify() + fastify.register(fastifySwagger, { + refResolver: { + buildLocalReference (json) { + return json.$id + } + } + }) + fastify.register(async (instance) => { + instance.addSchema({ $id: 'schemaA', type: 'object', properties: { id: { type: 'integer' } } }) + instance.addSchema({ $id: 'schemaB', type: 'object', properties: { id: { type: 'string' } } }) + instance.addSchema({ $id: 'schemaC', type: 'object', properties: { a: { type: 'array', items: { $ref: 'schemaA' } } } }) + instance.addSchema({ $id: 'schemaD', type: 'object', properties: { b: { $ref: 'schemaB' }, c: { $ref: 'schemaC' } } }) + instance.post('/url1', { schema: { body: { $ref: 'schemaD' }, response: { 200: { $ref: 'schemaB' } } } }, () => {}) + instance.post('/url2', { schema: { body: { $ref: 'schemaC' }, response: { 200: { $ref: 'schemaA' } } } }, () => {}) + }) + + await fastify.ready() + + const swaggerObject = fastify.swagger() + t.equal(typeof swaggerObject, 'object') + + const definitions = swaggerObject.definitions + + t.match(Object.keys(definitions), ['schemaA', 'schemaB', 'schemaC', 'schemaD']) + + // ref must be prefixed by '#/definitions/' + t.equal(definitions.schemaC.properties.a.items.$ref, '#/definitions/schemaA') + t.equal(definitions.schemaD.properties.b.$ref, '#/definitions/schemaB') + t.equal(definitions.schemaD.properties.c.$ref, '#/definitions/schemaC') + + await Swagger.validate(swaggerObject) +})