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.
		
		
		
		
		
			
		
			
				
					
					
						
							528 lines
						
					
					
						
							12 KiB
						
					
					
				
			
		
		
	
	
							528 lines
						
					
					
						
							12 KiB
						
					
					
				| 'use strict'
 | |
| 
 | |
| const { test } = require('tap')
 | |
| const Fastify = require('fastify')
 | |
| const Swagger = require('swagger-parser')
 | |
| const fastifySwagger = require('../../../index')
 | |
| const S = require('fluent-json-schema')
 | |
| 
 | |
| test('support file in json schema', async t => {
 | |
|   const opts7 = {
 | |
|     schema: {
 | |
|       consumes: ['application/x-www-form-urlencoded'],
 | |
|       body: {
 | |
|         type: 'object',
 | |
|         properties: {
 | |
|           hello: {
 | |
|             description: 'hello',
 | |
|             type: 'string',
 | |
|             contentEncoding: 'binary'
 | |
|           }
 | |
|         },
 | |
|         required: ['hello']
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   const fastify = Fastify()
 | |
|   fastify.register(fastifySwagger, {
 | |
|     routePrefix: '/docs',
 | |
|     exposeRoute: true
 | |
|   })
 | |
|   fastify.get('/', opts7, () => {})
 | |
| 
 | |
|   await fastify.ready()
 | |
| 
 | |
|   const swaggerObject = fastify.swagger()
 | |
|   const api = await Swagger.validate(swaggerObject)
 | |
| 
 | |
|   const definedPath = api.paths['/'].get
 | |
|   t.ok(definedPath)
 | |
|   t.same(definedPath.parameters, [{
 | |
|     in: 'formData',
 | |
|     name: 'hello',
 | |
|     description: 'hello',
 | |
|     required: true,
 | |
|     type: 'file'
 | |
|   }])
 | |
| })
 | |
| 
 | |
| test('support response description', async t => {
 | |
|   const opts8 = {
 | |
|     schema: {
 | |
|       response: {
 | |
|         200: {
 | |
|           description: 'Response OK!',
 | |
|           type: 'object'
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   const fastify = Fastify()
 | |
|   fastify.register(fastifySwagger, {
 | |
|     routePrefix: '/docs',
 | |
|     exposeRoute: true
 | |
|   })
 | |
|   fastify.get('/', opts8, () => {})
 | |
| 
 | |
|   await fastify.ready()
 | |
| 
 | |
|   const swaggerObject = fastify.swagger()
 | |
|   const api = await Swagger.validate(swaggerObject)
 | |
| 
 | |
|   const definedPath = api.paths['/'].get
 | |
|   t.same(definedPath.responses['200'].description, 'Response OK!')
 | |
| })
 | |
| 
 | |
| test('response default description', async t => {
 | |
|   const opts9 = {
 | |
|     schema: {
 | |
|       response: {
 | |
|         200: {
 | |
|           type: 'object'
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   const fastify = Fastify()
 | |
|   fastify.register(fastifySwagger, {
 | |
|     routePrefix: '/docs',
 | |
|     exposeRoute: true
 | |
|   })
 | |
|   fastify.get('/', opts9, () => {})
 | |
| 
 | |
|   await fastify.ready()
 | |
| 
 | |
|   const swaggerObject = fastify.swagger()
 | |
|   const api = await Swagger.validate(swaggerObject)
 | |
| 
 | |
|   const definedPath = api.paths['/'].get
 | |
|   t.same(definedPath.responses['200'].description, 'Default Response')
 | |
| })
 | |
| 
 | |
| test('response 2xx', async t => {
 | |
|   const opt = {
 | |
|     schema: {
 | |
|       response: {
 | |
|         '2xx': {
 | |
|           type: 'object'
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   const fastify = Fastify()
 | |
|   fastify.register(fastifySwagger, {
 | |
|     routePrefix: '/docs',
 | |
|     exposeRoute: true
 | |
|   })
 | |
|   fastify.get('/', opt, () => {})
 | |
| 
 | |
|   await fastify.ready()
 | |
| 
 | |
|   const swaggerObject = fastify.swagger()
 | |
|   const api = await Swagger.validate(swaggerObject)
 | |
| 
 | |
|   const definedPath = api.paths['/'].get
 | |
|   t.same(definedPath.responses['200'].description, 'Default Response')
 | |
|   t.notOk(definedPath.responses['2XX'])
 | |
| })
 | |
| 
 | |
| test('response conflict 2xx and 200', async t => {
 | |
|   const opt = {
 | |
|     schema: {
 | |
|       response: {
 | |
|         '2xx': {
 | |
|           type: 'object',
 | |
|           description: '2xx'
 | |
|         },
 | |
|         200: {
 | |
|           type: 'object',
 | |
|           description: '200'
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   const fastify = Fastify()
 | |
|   fastify.register(fastifySwagger, {
 | |
|     routePrefix: '/docs',
 | |
|     exposeRoute: true
 | |
|   })
 | |
|   fastify.get('/', opt, () => {})
 | |
| 
 | |
|   await fastify.ready()
 | |
| 
 | |
|   const swaggerObject = fastify.swagger()
 | |
|   const api = await Swagger.validate(swaggerObject)
 | |
| 
 | |
|   const definedPath = api.paths['/'].get
 | |
|   t.same(definedPath.responses['200'].description, '200')
 | |
|   t.notOk(definedPath.responses['2XX'])
 | |
| })
 | |
| 
 | |
| test('support status code 204', async t => {
 | |
|   const opt = {
 | |
|     schema: {
 | |
|       response: {
 | |
|         204: {
 | |
|           type: 'null',
 | |
|           description: 'No Content'
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   const fastify = Fastify()
 | |
|   fastify.register(fastifySwagger, {
 | |
|     routePrefix: '/docs',
 | |
|     exposeRoute: true
 | |
|   })
 | |
|   fastify.get('/', opt, () => {})
 | |
| 
 | |
|   await fastify.ready()
 | |
| 
 | |
|   const swaggerObject = fastify.swagger()
 | |
|   const api = await Swagger.validate(swaggerObject)
 | |
| 
 | |
|   const definedPath = api.paths['/'].get
 | |
|   t.same(definedPath.responses['204'].description, 'No Content')
 | |
|   t.notOk(definedPath.responses['204'].schema)
 | |
| })
 | |
| 
 | |
| test('support empty response body for different status than 204', async t => {
 | |
|   const opt = {
 | |
|     schema: {
 | |
|       response: {
 | |
|         204: {
 | |
|           type: 'null',
 | |
|           description: 'No Content'
 | |
|         },
 | |
|         503: {
 | |
|           type: 'null',
 | |
|           description: 'Service Unavailable'
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   const fastify = Fastify()
 | |
|   fastify.register(fastifySwagger, {
 | |
|     routePrefix: '/docs',
 | |
|     exposeRoute: true
 | |
|   })
 | |
|   fastify.get('/', opt, () => {})
 | |
| 
 | |
|   await fastify.ready()
 | |
| 
 | |
|   const swaggerObject = fastify.swagger()
 | |
|   const api = await Swagger.validate(swaggerObject)
 | |
| 
 | |
|   const definedPath = api.paths['/'].get
 | |
| 
 | |
|   t.same(definedPath.responses['204'].description, 'No Content')
 | |
|   t.notOk(definedPath.responses['204'].content)
 | |
|   t.notOk(definedPath.responses['503'].type)
 | |
| 
 | |
|   t.same(definedPath.responses['503'].description, 'Service Unavailable')
 | |
|   t.notOk(definedPath.responses['503'].content)
 | |
|   t.notOk(definedPath.responses['503'].type)
 | |
| })
 | |
| 
 | |
| test('support response headers', async t => {
 | |
|   const opt = {
 | |
|     schema: {
 | |
|       response: {
 | |
|         200: {
 | |
|           type: 'object',
 | |
|           properties: {
 | |
|             hello: {
 | |
|               type: 'string'
 | |
|             }
 | |
|           },
 | |
|           headers: {
 | |
|             'X-WORLD': {
 | |
|               type: 'string'
 | |
|             }
 | |
|           }
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   const fastify = Fastify()
 | |
|   fastify.register(fastifySwagger, {
 | |
|     routePrefix: '/docs',
 | |
|     exposeRoute: true
 | |
|   })
 | |
|   fastify.get('/', opt, () => {})
 | |
| 
 | |
|   await fastify.ready()
 | |
| 
 | |
|   const swaggerObject = fastify.swagger()
 | |
|   const api = await Swagger.validate(swaggerObject)
 | |
| 
 | |
|   const definedPath = api.paths['/'].get
 | |
|   t.same(definedPath.responses['200'].headers, opt.schema.response['200'].headers)
 | |
|   t.notOk(definedPath.responses['200'].schema.headers)
 | |
| })
 | |
| 
 | |
| test('response: description and x-response-description', async () => {
 | |
|   const description = 'description - always that of response body, sometimes also that of response as a whole'
 | |
|   const responseDescription = 'description only for the response as a whole'
 | |
| 
 | |
|   test('description without x-response-description doubles as response description', async t => {
 | |
|     // Given a /description endpoint with only a |description| field in its response schema
 | |
|     const fastify = Fastify()
 | |
|     fastify.register(fastifySwagger, {
 | |
|       routePrefix: '/docs',
 | |
|       exposeRoute: true
 | |
|     })
 | |
|     fastify.get('/description', {
 | |
|       schema: {
 | |
|         response: {
 | |
|           200: {
 | |
|             description,
 | |
|             type: 'string'
 | |
|           }
 | |
|         }
 | |
|       }
 | |
|     }, () => {})
 | |
|     await fastify.ready()
 | |
| 
 | |
|     // When the Swagger schema is generated
 | |
|     const swaggerObject = fastify.swagger()
 | |
|     const api = await Swagger.validate(swaggerObject)
 | |
| 
 | |
|     // Then the /description endpoint uses the |description| as both the description of the Response Object as well as of its Schema Object
 | |
|     const responseObject = api.paths['/description'].get.responses['200']
 | |
|     t.ok(responseObject)
 | |
|     t.equal(responseObject.description, description)
 | |
|     t.equal(responseObject.schema.description, description)
 | |
|   })
 | |
| 
 | |
|   test('description alongside x-response-description only describes response body', async t => {
 | |
|     // Given a /responseDescription endpoint that also has a |'x-response-description'| field in its response schema
 | |
|     const fastify = Fastify()
 | |
|     fastify.register(fastifySwagger, {
 | |
|       routePrefix: '/docs',
 | |
|       exposeRoute: true
 | |
|     })
 | |
|     fastify.get('/responseDescription', {
 | |
|       schema: {
 | |
|         response: {
 | |
|           200: {
 | |
|             'x-response-description': responseDescription,
 | |
|             description,
 | |
|             type: 'string'
 | |
|           }
 | |
|         }
 | |
|       }
 | |
|     }, () => {})
 | |
|     await fastify.ready()
 | |
| 
 | |
|     // When the Swagger schema is generated
 | |
|     const swaggerObject = fastify.swagger()
 | |
|     const api = await Swagger.validate(swaggerObject)
 | |
| 
 | |
|     // Then the /responseDescription endpoint uses the |responseDescription| only for the Response Object and the |description| only for the Schema Object
 | |
|     const responseObject = api.paths['/responseDescription'].get.responses['200']
 | |
|     t.ok(responseObject)
 | |
|     t.equal(responseObject.description, responseDescription)
 | |
|     t.equal(responseObject.schema.description, description)
 | |
|     t.equal(responseObject.schema.responseDescription, undefined)
 | |
|   })
 | |
| })
 | |
| 
 | |
| test('support "default" parameter', async t => {
 | |
|   const opt = {
 | |
|     schema: {
 | |
|       response: {
 | |
|         200: {
 | |
|           description: 'Expected Response',
 | |
|           type: 'object',
 | |
|           properties: {
 | |
|             foo: {
 | |
|               type: 'string'
 | |
|             }
 | |
|           }
 | |
|         },
 | |
|         default: {
 | |
|           description: 'Default Response',
 | |
|           type: 'object',
 | |
|           properties: {
 | |
|             bar: {
 | |
|               type: 'string'
 | |
|             }
 | |
|           }
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   const fastify = Fastify()
 | |
|   fastify.register(fastifySwagger, {
 | |
|     routePrefix: '/docs',
 | |
|     exposeRoute: true
 | |
|   })
 | |
|   fastify.get('/', opt, () => {})
 | |
| 
 | |
|   await fastify.ready()
 | |
| 
 | |
|   const swaggerObject = fastify.swagger()
 | |
|   const api = await Swagger.validate(swaggerObject)
 | |
| 
 | |
|   const definedPath = api.paths['/'].get
 | |
| 
 | |
|   t.same(definedPath.responses.default, {
 | |
|     description: 'Default Response',
 | |
|     schema: {
 | |
|       description: 'Default Response',
 | |
|       type: 'object',
 | |
|       properties: {
 | |
|         bar: {
 | |
|           type: 'string'
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   })
 | |
| })
 | |
| 
 | |
| test('fluent-json-schema', async t => {
 | |
|   const opt = {
 | |
|     schema: {
 | |
|       response: {
 | |
|         200: S.object()
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   const fastify = Fastify()
 | |
|   fastify.register(fastifySwagger, {
 | |
|     swagger: true,
 | |
|     routePrefix: '/docs',
 | |
|     exposeRoute: true
 | |
|   })
 | |
|   fastify.get('/', opt, () => {})
 | |
| 
 | |
|   await fastify.ready()
 | |
| 
 | |
|   const swaggerObject = fastify.swagger()
 | |
|   const api = await Swagger.validate(swaggerObject)
 | |
| 
 | |
|   const definedPath = api.paths['/'].get
 | |
|   t.same(definedPath.responses['200'].description, 'Default Response')
 | |
| })
 | |
| 
 | |
| test('support "patternProperties" in json schema', async t => {
 | |
|   const opt = {
 | |
|     schema: {
 | |
|       body: {
 | |
|         type: 'object',
 | |
|         patternProperties: {
 | |
|           '^[a-z]{2,3}-[a-zA-Z]{2}$': {
 | |
|             type: 'string'
 | |
|           }
 | |
|         }
 | |
|       },
 | |
|       response: {
 | |
|         200: {
 | |
|           description: 'Expected Response',
 | |
|           type: 'object',
 | |
|           properties: {
 | |
|             foo: {
 | |
|               type: 'object',
 | |
|               patternProperties: {
 | |
|                 '^[a-z]{2,3}-[a-zA-Z]{2}$': {
 | |
|                   type: 'string'
 | |
|                 }
 | |
|               },
 | |
|               additionalProperties: false
 | |
|             }
 | |
|           }
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   const fastify = Fastify()
 | |
|   fastify.register(fastifySwagger, {
 | |
|     swagger: true,
 | |
|     routePrefix: '/docs',
 | |
|     exposeRoute: true
 | |
|   })
 | |
|   fastify.get('/', opt, () => {})
 | |
| 
 | |
|   await fastify.ready()
 | |
| 
 | |
|   const swaggerObject = fastify.swagger()
 | |
|   const api = await Swagger.validate(swaggerObject)
 | |
| 
 | |
|   const definedPath = api.paths['/'].get
 | |
| 
 | |
|   t.same(definedPath.parameters[0].schema, {
 | |
|     type: 'object',
 | |
|     additionalProperties: { type: 'string' }
 | |
|   })
 | |
| 
 | |
|   t.same(definedPath.responses[200], {
 | |
|     description: 'Expected Response',
 | |
|     schema: {
 | |
|       description: 'Expected Response',
 | |
|       type: 'object',
 | |
|       properties: {
 | |
|         foo: {
 | |
|           type: 'object',
 | |
|           additionalProperties: { type: 'string' }
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   })
 | |
| })
 | |
| 
 | |
| test('support "const" keyword', async t => {
 | |
|   const opt = {
 | |
|     schema: {
 | |
|       body: {
 | |
|         type: 'object',
 | |
|         properties: {
 | |
|           obj: {
 | |
|             type: 'object',
 | |
|             properties: {
 | |
|               constantProp: { const: 'my-const' }
 | |
|             }
 | |
|           }
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   const fastify = Fastify()
 | |
|   fastify.register(fastifySwagger, {
 | |
|     routePrefix: '/docs',
 | |
|     exposeRoute: true
 | |
|   })
 | |
|   fastify.get('/', opt, () => {})
 | |
|   await fastify.ready()
 | |
| 
 | |
|   const swaggerObject = fastify.swagger()
 | |
|   const api = await Swagger.validate(swaggerObject)
 | |
| 
 | |
|   const definedPath = api.paths['/'].get
 | |
|   t.same(definedPath.parameters[0].schema, {
 | |
|     type: 'object',
 | |
|     properties: {
 | |
|       obj: {
 | |
|         type: 'object',
 | |
|         properties: {
 | |
|           constantProp: {
 | |
|             enum: ['my-const']
 | |
|           }
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   })
 | |
| })
 |