1 | 'use strict'
|
2 |
|
3 | const {
|
4 | kSchemaHeaders: headersSchema,
|
5 | kSchemaParams: paramsSchema,
|
6 | kSchemaQuerystring: querystringSchema,
|
7 | kSchemaBody: bodySchema,
|
8 | kSchemaResponse: responseSchema
|
9 | } = require('./symbols')
|
10 | const scChecker = /^[1-5]{1}[0-9]{2}$|^[1-5]xx$|^default$/
|
11 |
|
12 | const {
|
13 | FST_ERR_SCH_RESPONSE_SCHEMA_NOT_NESTED_2XX
|
14 | } = require('./errors')
|
15 |
|
16 | const { FSTWRN001 } = require('./warnings')
|
17 |
|
18 | function compileSchemasForSerialization (context, compile) {
|
19 | if (!context.schema || !context.schema.response) {
|
20 | return
|
21 | }
|
22 | const { method, url } = context.config || {}
|
23 | context[responseSchema] = Object.keys(context.schema.response)
|
24 | .reduce(function (acc, statusCode) {
|
25 | const schema = context.schema.response[statusCode]
|
26 | statusCode = statusCode.toLowerCase()
|
27 | if (!scChecker.exec(statusCode)) {
|
28 | throw new FST_ERR_SCH_RESPONSE_SCHEMA_NOT_NESTED_2XX()
|
29 | }
|
30 |
|
31 | if (schema.content) {
|
32 | const contentTypesSchemas = {}
|
33 | for (const mediaName of Object.keys(schema.content)) {
|
34 | const contentSchema = schema.content[mediaName].schema
|
35 | contentTypesSchemas[mediaName] = compile({
|
36 | schema: contentSchema,
|
37 | url,
|
38 | method,
|
39 | httpStatus: statusCode,
|
40 | contentType: mediaName
|
41 | })
|
42 | }
|
43 | acc[statusCode] = contentTypesSchemas
|
44 | } else {
|
45 | acc[statusCode] = compile({
|
46 | schema,
|
47 | url,
|
48 | method,
|
49 | httpStatus: statusCode
|
50 | })
|
51 | }
|
52 |
|
53 | return acc
|
54 | }, {})
|
55 | }
|
56 |
|
57 | function compileSchemasForValidation (context, compile, isCustom) {
|
58 | const { schema } = context
|
59 | if (!schema) {
|
60 | return
|
61 | }
|
62 |
|
63 | const { method, url } = context.config || {}
|
64 |
|
65 | const headers = schema.headers
|
66 |
|
67 | if (headers && (isCustom || Object.getPrototypeOf(headers) !== Object.prototype)) {
|
68 |
|
69 | context[headersSchema] = compile({ schema: headers, method, url, httpPart: 'headers' })
|
70 | } else if (headers) {
|
71 |
|
72 |
|
73 | const headersSchemaLowerCase = {}
|
74 | Object.keys(headers).forEach(k => { headersSchemaLowerCase[k] = headers[k] })
|
75 | if (headersSchemaLowerCase.required instanceof Array) {
|
76 | headersSchemaLowerCase.required = headersSchemaLowerCase.required.map(h => h.toLowerCase())
|
77 | }
|
78 | if (headers.properties) {
|
79 | headersSchemaLowerCase.properties = {}
|
80 | Object.keys(headers.properties).forEach(k => {
|
81 | headersSchemaLowerCase.properties[k.toLowerCase()] = headers.properties[k]
|
82 | })
|
83 | }
|
84 | context[headersSchema] = compile({ schema: headersSchemaLowerCase, method, url, httpPart: 'headers' })
|
85 | } else if (Object.prototype.hasOwnProperty.call(schema, 'headers')) {
|
86 | FSTWRN001('headers', method, url)
|
87 | }
|
88 |
|
89 | if (schema.body) {
|
90 | context[bodySchema] = compile({ schema: schema.body, method, url, httpPart: 'body' })
|
91 | } else if (Object.prototype.hasOwnProperty.call(schema, 'body')) {
|
92 | FSTWRN001('body', method, url)
|
93 | }
|
94 |
|
95 | if (schema.querystring) {
|
96 | context[querystringSchema] = compile({ schema: schema.querystring, method, url, httpPart: 'querystring' })
|
97 | } else if (Object.prototype.hasOwnProperty.call(schema, 'querystring')) {
|
98 | FSTWRN001('querystring', method, url)
|
99 | }
|
100 |
|
101 | if (schema.params) {
|
102 | context[paramsSchema] = compile({ schema: schema.params, method, url, httpPart: 'params' })
|
103 | } else if (Object.prototype.hasOwnProperty.call(schema, 'params')) {
|
104 | FSTWRN001('params', method, url)
|
105 | }
|
106 | }
|
107 |
|
108 | function validateParam (validatorFunction, request, paramName) {
|
109 | const isUndefined = request[paramName] === undefined
|
110 | const ret = validatorFunction && validatorFunction(isUndefined ? null : request[paramName])
|
111 |
|
112 | if (ret?.then) {
|
113 | return ret
|
114 | .then((res) => { return answer(res) })
|
115 | .catch(err => { return err })
|
116 | }
|
117 |
|
118 | return answer(ret)
|
119 |
|
120 | function answer (ret) {
|
121 | if (ret === false) return validatorFunction.errors
|
122 | if (ret && ret.error) return ret.error
|
123 | if (ret && ret.value) request[paramName] = ret.value
|
124 | return false
|
125 | }
|
126 | }
|
127 |
|
128 | function validate (context, request, execution) {
|
129 | const runExecution = execution === undefined
|
130 |
|
131 | if (runExecution || !execution.skipParams) {
|
132 | const params = validateParam(context[paramsSchema], request, 'params')
|
133 | if (params) {
|
134 | if (typeof params.then !== 'function') {
|
135 | return wrapValidationError(params, 'params', context.schemaErrorFormatter)
|
136 | } else {
|
137 | return validateAsyncParams(params, context, request)
|
138 | }
|
139 | }
|
140 | }
|
141 |
|
142 | if (runExecution || !execution.skipBody) {
|
143 | const body = validateParam(context[bodySchema], request, 'body')
|
144 | if (body) {
|
145 | if (typeof body.then !== 'function') {
|
146 | return wrapValidationError(body, 'body', context.schemaErrorFormatter)
|
147 | } else {
|
148 | return validateAsyncBody(body, context, request)
|
149 | }
|
150 | }
|
151 | }
|
152 |
|
153 | if (runExecution || !execution.skipQuery) {
|
154 | const query = validateParam(context[querystringSchema], request, 'query')
|
155 | if (query) {
|
156 | if (typeof query.then !== 'function') {
|
157 | return wrapValidationError(query, 'querystring', context.schemaErrorFormatter)
|
158 | } else {
|
159 | return validateAsyncQuery(query, context, request)
|
160 | }
|
161 | }
|
162 | }
|
163 |
|
164 | const headers = validateParam(context[headersSchema], request, 'headers')
|
165 | if (headers) {
|
166 | if (typeof headers.then !== 'function') {
|
167 | return wrapValidationError(headers, 'headers', context.schemaErrorFormatter)
|
168 | } else {
|
169 | return validateAsyncHeaders(headers, context, request)
|
170 | }
|
171 | }
|
172 |
|
173 | return false
|
174 | }
|
175 |
|
176 | function validateAsyncParams (validatePromise, context, request) {
|
177 | return validatePromise
|
178 | .then((paramsResult) => {
|
179 | if (paramsResult) {
|
180 | return wrapValidationError(paramsResult, 'params', context.schemaErrorFormatter)
|
181 | }
|
182 |
|
183 | return validate(context, request, { skipParams: true })
|
184 | })
|
185 | }
|
186 |
|
187 | function validateAsyncBody (validatePromise, context, request) {
|
188 | return validatePromise
|
189 | .then((bodyResult) => {
|
190 | if (bodyResult) {
|
191 | return wrapValidationError(bodyResult, 'body', context.schemaErrorFormatter)
|
192 | }
|
193 |
|
194 | return validate(context, request, { skipParams: true, skipBody: true })
|
195 | })
|
196 | }
|
197 |
|
198 | function validateAsyncQuery (validatePromise, context, request) {
|
199 | return validatePromise
|
200 | .then((queryResult) => {
|
201 | if (queryResult) {
|
202 | return wrapValidationError(queryResult, 'querystring', context.schemaErrorFormatter)
|
203 | }
|
204 |
|
205 | return validate(context, request, { skipParams: true, skipBody: true, skipQuery: true })
|
206 | })
|
207 | }
|
208 |
|
209 | function validateAsyncHeaders (validatePromise, context, request) {
|
210 | return validatePromise
|
211 | .then((headersResult) => {
|
212 | if (headersResult) {
|
213 | return wrapValidationError(headersResult, 'headers', context.schemaErrorFormatter)
|
214 | }
|
215 |
|
216 | return false
|
217 | })
|
218 | }
|
219 |
|
220 | function wrapValidationError (result, dataVar, schemaErrorFormatter) {
|
221 | if (result instanceof Error) {
|
222 | result.statusCode = result.statusCode || 400
|
223 | result.code = result.code || 'FST_ERR_VALIDATION'
|
224 | result.validationContext = result.validationContext || dataVar
|
225 | return result
|
226 | }
|
227 |
|
228 | const error = schemaErrorFormatter(result, dataVar)
|
229 | error.statusCode = error.statusCode || 400
|
230 | error.code = error.code || 'FST_ERR_VALIDATION'
|
231 | error.validation = result
|
232 | error.validationContext = dataVar
|
233 | return error
|
234 | }
|
235 |
|
236 | module.exports = {
|
237 | symbols: { bodySchema, querystringSchema, responseSchema, paramsSchema, headersSchema },
|
238 | compileSchemasForValidation,
|
239 | compileSchemasForSerialization,
|
240 | validate
|
241 | }
|