node-fastify 5.8.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (354) hide show
  1. package/.borp.yaml +3 -0
  2. package/.markdownlint-cli2.yaml +22 -0
  3. package/.prettierignore +1 -0
  4. package/GOVERNANCE.md +4 -0
  5. package/LICENSE +21 -0
  6. package/PROJECT_CHARTER.md +126 -0
  7. package/README.md +423 -0
  8. package/SECURITY.md +220 -0
  9. package/SPONSORS.md +24 -0
  10. package/build/build-error-serializer.js +35 -0
  11. package/build/build-validation.js +169 -0
  12. package/build/sync-version.js +11 -0
  13. package/docs/Guides/Benchmarking.md +60 -0
  14. package/docs/Guides/Database.md +321 -0
  15. package/docs/Guides/Delay-Accepting-Requests.md +608 -0
  16. package/docs/Guides/Detecting-When-Clients-Abort.md +172 -0
  17. package/docs/Guides/Ecosystem.md +726 -0
  18. package/docs/Guides/Fluent-Schema.md +127 -0
  19. package/docs/Guides/Getting-Started.md +620 -0
  20. package/docs/Guides/Index.md +43 -0
  21. package/docs/Guides/Migration-Guide-V3.md +287 -0
  22. package/docs/Guides/Migration-Guide-V4.md +267 -0
  23. package/docs/Guides/Migration-Guide-V5.md +727 -0
  24. package/docs/Guides/Plugins-Guide.md +520 -0
  25. package/docs/Guides/Prototype-Poisoning.md +383 -0
  26. package/docs/Guides/Recommendations.md +378 -0
  27. package/docs/Guides/Serverless.md +604 -0
  28. package/docs/Guides/Style-Guide.md +246 -0
  29. package/docs/Guides/Testing.md +481 -0
  30. package/docs/Guides/Write-Plugin.md +103 -0
  31. package/docs/Guides/Write-Type-Provider.md +34 -0
  32. package/docs/Reference/ContentTypeParser.md +271 -0
  33. package/docs/Reference/Decorators.md +436 -0
  34. package/docs/Reference/Encapsulation.md +194 -0
  35. package/docs/Reference/Errors.md +377 -0
  36. package/docs/Reference/HTTP2.md +94 -0
  37. package/docs/Reference/Hooks.md +958 -0
  38. package/docs/Reference/Index.md +73 -0
  39. package/docs/Reference/LTS.md +86 -0
  40. package/docs/Reference/Lifecycle.md +99 -0
  41. package/docs/Reference/Logging.md +268 -0
  42. package/docs/Reference/Middleware.md +79 -0
  43. package/docs/Reference/Plugins.md +245 -0
  44. package/docs/Reference/Principles.md +73 -0
  45. package/docs/Reference/Reply.md +1001 -0
  46. package/docs/Reference/Request.md +295 -0
  47. package/docs/Reference/Routes.md +802 -0
  48. package/docs/Reference/Server.md +2389 -0
  49. package/docs/Reference/Type-Providers.md +256 -0
  50. package/docs/Reference/TypeScript.md +1729 -0
  51. package/docs/Reference/Validation-and-Serialization.md +1130 -0
  52. package/docs/Reference/Warnings.md +58 -0
  53. package/docs/index.md +24 -0
  54. package/docs/resources/encapsulation_context.drawio +1 -0
  55. package/docs/resources/encapsulation_context.svg +3 -0
  56. package/eslint.config.js +35 -0
  57. package/examples/asyncawait.js +38 -0
  58. package/examples/benchmark/body.json +3 -0
  59. package/examples/benchmark/hooks-benchmark-async-await.js +44 -0
  60. package/examples/benchmark/hooks-benchmark.js +52 -0
  61. package/examples/benchmark/parser.js +47 -0
  62. package/examples/benchmark/simple.js +30 -0
  63. package/examples/benchmark/webstream.js +27 -0
  64. package/examples/hooks.js +91 -0
  65. package/examples/http2.js +39 -0
  66. package/examples/https.js +38 -0
  67. package/examples/parser.js +53 -0
  68. package/examples/plugin.js +12 -0
  69. package/examples/route-prefix.js +38 -0
  70. package/examples/shared-schema.js +38 -0
  71. package/examples/simple-stream.js +20 -0
  72. package/examples/simple.js +32 -0
  73. package/examples/simple.mjs +27 -0
  74. package/examples/typescript-server.ts +79 -0
  75. package/examples/use-plugin.js +29 -0
  76. package/fastify.d.ts +253 -0
  77. package/fastify.js +985 -0
  78. package/integration/server.js +29 -0
  79. package/integration/test.sh +23 -0
  80. package/lib/config-validator.js +1266 -0
  81. package/lib/content-type-parser.js +413 -0
  82. package/lib/content-type.js +160 -0
  83. package/lib/context.js +98 -0
  84. package/lib/decorate.js +152 -0
  85. package/lib/error-handler.js +173 -0
  86. package/lib/error-serializer.js +134 -0
  87. package/lib/error-status.js +14 -0
  88. package/lib/errors.js +516 -0
  89. package/lib/four-oh-four.js +190 -0
  90. package/lib/handle-request.js +195 -0
  91. package/lib/head-route.js +45 -0
  92. package/lib/hooks.js +429 -0
  93. package/lib/initial-config-validation.js +37 -0
  94. package/lib/logger-factory.js +136 -0
  95. package/lib/logger-pino.js +68 -0
  96. package/lib/noop-set.js +10 -0
  97. package/lib/plugin-override.js +90 -0
  98. package/lib/plugin-utils.js +169 -0
  99. package/lib/promise.js +23 -0
  100. package/lib/reply.js +1030 -0
  101. package/lib/req-id-gen-factory.js +52 -0
  102. package/lib/request.js +391 -0
  103. package/lib/route.js +686 -0
  104. package/lib/schema-controller.js +164 -0
  105. package/lib/schemas.js +207 -0
  106. package/lib/server.js +441 -0
  107. package/lib/symbols.js +71 -0
  108. package/lib/validation.js +280 -0
  109. package/lib/warnings.js +57 -0
  110. package/lib/wrap-thenable.js +84 -0
  111. package/package.json +225 -0
  112. package/scripts/validate-ecosystem-links.js +179 -0
  113. package/test/404s.test.js +2035 -0
  114. package/test/500s.test.js +422 -0
  115. package/test/allow-unsafe-regex.test.js +92 -0
  116. package/test/als.test.js +65 -0
  117. package/test/async-await.test.js +705 -0
  118. package/test/async-dispose.test.js +20 -0
  119. package/test/async_hooks.test.js +52 -0
  120. package/test/body-limit.test.js +224 -0
  121. package/test/buffer.test.js +74 -0
  122. package/test/build/error-serializer.test.js +36 -0
  123. package/test/build/version.test.js +14 -0
  124. package/test/build-certificate.js +109 -0
  125. package/test/bundler/README.md +29 -0
  126. package/test/bundler/esbuild/bundler-test.js +32 -0
  127. package/test/bundler/esbuild/package.json +10 -0
  128. package/test/bundler/esbuild/src/fail-plugin-version.js +14 -0
  129. package/test/bundler/esbuild/src/index.js +9 -0
  130. package/test/bundler/webpack/bundler-test.js +32 -0
  131. package/test/bundler/webpack/package.json +11 -0
  132. package/test/bundler/webpack/src/fail-plugin-version.js +14 -0
  133. package/test/bundler/webpack/src/index.js +9 -0
  134. package/test/bundler/webpack/webpack.config.js +15 -0
  135. package/test/case-insensitive.test.js +102 -0
  136. package/test/chainable.test.js +40 -0
  137. package/test/child-logger-factory.test.js +128 -0
  138. package/test/client-timeout.test.js +38 -0
  139. package/test/close-pipelining.test.js +78 -0
  140. package/test/close.test.js +706 -0
  141. package/test/conditional-pino.test.js +47 -0
  142. package/test/connection-timeout.test.js +42 -0
  143. package/test/constrained-routes.test.js +1138 -0
  144. package/test/content-length.test.js +174 -0
  145. package/test/content-parser.test.js +739 -0
  146. package/test/content-type.test.js +181 -0
  147. package/test/context-config.test.js +164 -0
  148. package/test/custom-http-server.test.js +118 -0
  149. package/test/custom-parser-async.test.js +59 -0
  150. package/test/custom-parser.0.test.js +701 -0
  151. package/test/custom-parser.1.test.js +266 -0
  152. package/test/custom-parser.2.test.js +91 -0
  153. package/test/custom-parser.3.test.js +208 -0
  154. package/test/custom-parser.4.test.js +218 -0
  155. package/test/custom-parser.5.test.js +130 -0
  156. package/test/custom-querystring-parser.test.js +129 -0
  157. package/test/decorator.test.js +1330 -0
  158. package/test/delete.test.js +344 -0
  159. package/test/diagnostics-channel/404.test.js +49 -0
  160. package/test/diagnostics-channel/async-delay-request.test.js +65 -0
  161. package/test/diagnostics-channel/async-request.test.js +64 -0
  162. package/test/diagnostics-channel/error-before-handler.test.js +35 -0
  163. package/test/diagnostics-channel/error-request.test.js +53 -0
  164. package/test/diagnostics-channel/error-status.test.js +123 -0
  165. package/test/diagnostics-channel/init.test.js +50 -0
  166. package/test/diagnostics-channel/sync-delay-request.test.js +49 -0
  167. package/test/diagnostics-channel/sync-request-reply.test.js +51 -0
  168. package/test/diagnostics-channel/sync-request.test.js +54 -0
  169. package/test/encapsulated-child-logger-factory.test.js +69 -0
  170. package/test/encapsulated-error-handler.test.js +237 -0
  171. package/test/esm/errorCodes.test.mjs +10 -0
  172. package/test/esm/esm.test.mjs +13 -0
  173. package/test/esm/index.test.js +8 -0
  174. package/test/esm/named-exports.mjs +14 -0
  175. package/test/esm/other.mjs +8 -0
  176. package/test/esm/plugin.mjs +8 -0
  177. package/test/fastify-instance.test.js +300 -0
  178. package/test/find-route.test.js +152 -0
  179. package/test/fluent-schema.test.js +209 -0
  180. package/test/genReqId.test.js +426 -0
  181. package/test/handler-context.test.js +45 -0
  182. package/test/handler-timeout.test.js +367 -0
  183. package/test/has-route.test.js +88 -0
  184. package/test/header-overflow.test.js +55 -0
  185. package/test/helper.js +496 -0
  186. package/test/hooks-async.test.js +1099 -0
  187. package/test/hooks.on-listen.test.js +1162 -0
  188. package/test/hooks.on-ready.test.js +421 -0
  189. package/test/hooks.test.js +3578 -0
  190. package/test/http-methods/copy.test.js +35 -0
  191. package/test/http-methods/custom-http-methods.test.js +114 -0
  192. package/test/http-methods/get.test.js +412 -0
  193. package/test/http-methods/head.test.js +263 -0
  194. package/test/http-methods/lock.test.js +108 -0
  195. package/test/http-methods/mkcalendar.test.js +143 -0
  196. package/test/http-methods/mkcol.test.js +35 -0
  197. package/test/http-methods/move.test.js +42 -0
  198. package/test/http-methods/propfind.test.js +136 -0
  199. package/test/http-methods/proppatch.test.js +105 -0
  200. package/test/http-methods/report.test.js +142 -0
  201. package/test/http-methods/search.test.js +233 -0
  202. package/test/http-methods/trace.test.js +21 -0
  203. package/test/http-methods/unlock.test.js +38 -0
  204. package/test/http2/closing.test.js +270 -0
  205. package/test/http2/constraint.test.js +109 -0
  206. package/test/http2/head.test.js +34 -0
  207. package/test/http2/plain.test.js +68 -0
  208. package/test/http2/secure-with-fallback.test.js +113 -0
  209. package/test/http2/secure.test.js +67 -0
  210. package/test/http2/unknown-http-method.test.js +34 -0
  211. package/test/https/custom-https-server.test.js +58 -0
  212. package/test/https/https.test.js +136 -0
  213. package/test/imports.test.js +17 -0
  214. package/test/inject.test.js +502 -0
  215. package/test/input-validation.js +335 -0
  216. package/test/internals/all.test.js +38 -0
  217. package/test/internals/content-type-parser.test.js +111 -0
  218. package/test/internals/context.test.js +31 -0
  219. package/test/internals/decorator.test.js +156 -0
  220. package/test/internals/errors.test.js +982 -0
  221. package/test/internals/handle-request.test.js +270 -0
  222. package/test/internals/hook-runner.test.js +449 -0
  223. package/test/internals/hooks.test.js +96 -0
  224. package/test/internals/initial-config.test.js +383 -0
  225. package/test/internals/logger.test.js +163 -0
  226. package/test/internals/plugin.test.js +170 -0
  227. package/test/internals/promise.test.js +63 -0
  228. package/test/internals/reply-serialize.test.js +714 -0
  229. package/test/internals/reply.test.js +1920 -0
  230. package/test/internals/req-id-gen-factory.test.js +133 -0
  231. package/test/internals/request-validate.test.js +1402 -0
  232. package/test/internals/request.test.js +506 -0
  233. package/test/internals/schema-controller-perf.test.js +40 -0
  234. package/test/internals/server.test.js +91 -0
  235. package/test/internals/validation.test.js +352 -0
  236. package/test/issue-4959.test.js +118 -0
  237. package/test/keep-alive-timeout.test.js +42 -0
  238. package/test/listen.1.test.js +154 -0
  239. package/test/listen.2.test.js +113 -0
  240. package/test/listen.3.test.js +83 -0
  241. package/test/listen.4.test.js +168 -0
  242. package/test/listen.5.test.js +122 -0
  243. package/test/logger/instantiation.test.js +341 -0
  244. package/test/logger/logger-test-utils.js +47 -0
  245. package/test/logger/logging.test.js +460 -0
  246. package/test/logger/options.test.js +579 -0
  247. package/test/logger/request.test.js +292 -0
  248. package/test/logger/response.test.js +183 -0
  249. package/test/logger/tap-parallel-not-ok +0 -0
  250. package/test/max-requests-per-socket.test.js +113 -0
  251. package/test/middleware.test.js +37 -0
  252. package/test/noop-set.test.js +19 -0
  253. package/test/nullable-validation.test.js +187 -0
  254. package/test/options.error-handler.test.js +5 -0
  255. package/test/options.test.js +5 -0
  256. package/test/output-validation.test.js +140 -0
  257. package/test/patch.error-handler.test.js +5 -0
  258. package/test/patch.test.js +5 -0
  259. package/test/plugin.1.test.js +230 -0
  260. package/test/plugin.2.test.js +314 -0
  261. package/test/plugin.3.test.js +287 -0
  262. package/test/plugin.4.test.js +504 -0
  263. package/test/plugin.helper.js +8 -0
  264. package/test/plugin.name.display.js +10 -0
  265. package/test/post-empty-body.test.js +38 -0
  266. package/test/pretty-print.test.js +366 -0
  267. package/test/promises.test.js +125 -0
  268. package/test/proto-poisoning.test.js +145 -0
  269. package/test/put.error-handler.test.js +5 -0
  270. package/test/put.test.js +5 -0
  271. package/test/register.test.js +184 -0
  272. package/test/reply-code.test.js +148 -0
  273. package/test/reply-early-hints.test.js +100 -0
  274. package/test/reply-error.test.js +815 -0
  275. package/test/reply-trailers.test.js +445 -0
  276. package/test/reply-web-stream-locked.test.js +37 -0
  277. package/test/request-error.test.js +624 -0
  278. package/test/request-header-host.test.js +339 -0
  279. package/test/request-id.test.js +118 -0
  280. package/test/request-timeout.test.js +53 -0
  281. package/test/route-hooks.test.js +635 -0
  282. package/test/route-prefix.test.js +904 -0
  283. package/test/route-shorthand.test.js +48 -0
  284. package/test/route.1.test.js +259 -0
  285. package/test/route.2.test.js +100 -0
  286. package/test/route.3.test.js +213 -0
  287. package/test/route.4.test.js +127 -0
  288. package/test/route.5.test.js +211 -0
  289. package/test/route.6.test.js +306 -0
  290. package/test/route.7.test.js +406 -0
  291. package/test/route.8.test.js +225 -0
  292. package/test/router-options.test.js +1108 -0
  293. package/test/same-shape.test.js +124 -0
  294. package/test/schema-examples.test.js +661 -0
  295. package/test/schema-feature.test.js +2198 -0
  296. package/test/schema-serialization.test.js +1171 -0
  297. package/test/schema-special-usage.test.js +1348 -0
  298. package/test/schema-validation.test.js +1572 -0
  299. package/test/scripts/validate-ecosystem-links.test.js +339 -0
  300. package/test/serialize-response.test.js +186 -0
  301. package/test/server.test.js +347 -0
  302. package/test/set-error-handler.test.js +69 -0
  303. package/test/skip-reply-send.test.js +317 -0
  304. package/test/stream-serializers.test.js +40 -0
  305. package/test/stream.1.test.js +94 -0
  306. package/test/stream.2.test.js +129 -0
  307. package/test/stream.3.test.js +198 -0
  308. package/test/stream.4.test.js +176 -0
  309. package/test/stream.5.test.js +188 -0
  310. package/test/sync-routes.test.js +32 -0
  311. package/test/throw.test.js +359 -0
  312. package/test/toolkit.js +63 -0
  313. package/test/trust-proxy.test.js +162 -0
  314. package/test/type-provider.test.js +22 -0
  315. package/test/types/content-type-parser.test-d.ts +72 -0
  316. package/test/types/decorate-request-reply.test-d.ts +18 -0
  317. package/test/types/dummy-plugin.ts +9 -0
  318. package/test/types/errors.test-d.ts +90 -0
  319. package/test/types/fastify.test-d.ts +352 -0
  320. package/test/types/hooks.test-d.ts +550 -0
  321. package/test/types/import.ts +2 -0
  322. package/test/types/instance.test-d.ts +588 -0
  323. package/test/types/logger.test-d.ts +277 -0
  324. package/test/types/plugin.test-d.ts +97 -0
  325. package/test/types/register.test-d.ts +237 -0
  326. package/test/types/reply.test-d.ts +254 -0
  327. package/test/types/request.test-d.ts +188 -0
  328. package/test/types/route.test-d.ts +553 -0
  329. package/test/types/schema.test-d.ts +135 -0
  330. package/test/types/serverFactory.test-d.ts +37 -0
  331. package/test/types/type-provider.test-d.ts +1213 -0
  332. package/test/types/using.test-d.ts +17 -0
  333. package/test/upgrade.test.js +52 -0
  334. package/test/url-rewriting.test.js +122 -0
  335. package/test/use-semicolon-delimiter.test.js +168 -0
  336. package/test/validation-error-handling.test.js +900 -0
  337. package/test/versioned-routes.test.js +603 -0
  338. package/test/web-api.test.js +616 -0
  339. package/test/wrap-thenable.test.js +30 -0
  340. package/types/content-type-parser.d.ts +75 -0
  341. package/types/context.d.ts +22 -0
  342. package/types/errors.d.ts +92 -0
  343. package/types/hooks.d.ts +875 -0
  344. package/types/instance.d.ts +609 -0
  345. package/types/logger.d.ts +107 -0
  346. package/types/plugin.d.ts +44 -0
  347. package/types/register.d.ts +42 -0
  348. package/types/reply.d.ts +81 -0
  349. package/types/request.d.ts +95 -0
  350. package/types/route.d.ts +199 -0
  351. package/types/schema.d.ts +61 -0
  352. package/types/server-factory.d.ts +19 -0
  353. package/types/type-provider.d.ts +130 -0
  354. package/types/utils.d.ts +98 -0
@@ -0,0 +1,661 @@
1
+ 'use strict'
2
+
3
+ const { test } = require('node:test')
4
+ const localize = require('ajv-i18n')
5
+ const Fastify = require('..')
6
+
7
+ test('Example - URI $id', (t, done) => {
8
+ t.plan(1)
9
+ const fastify = Fastify()
10
+ fastify.addSchema({
11
+ $id: 'http://fastify.test/',
12
+ type: 'object',
13
+ properties: {
14
+ hello: { type: 'string' }
15
+ }
16
+ })
17
+
18
+ fastify.post('/', {
19
+ handler () { },
20
+ schema: {
21
+ body: {
22
+ type: 'array',
23
+ items: { $ref: 'http://fastify.test#/properties/hello' }
24
+ }
25
+ }
26
+ })
27
+
28
+ fastify.ready(err => {
29
+ t.assert.ifError(err)
30
+ done()
31
+ })
32
+ })
33
+
34
+ test('Example - string $id', (t, done) => {
35
+ t.plan(1)
36
+ const fastify = Fastify()
37
+ fastify.addSchema({
38
+ $id: 'commonSchema',
39
+ type: 'object',
40
+ properties: {
41
+ hello: { type: 'string' }
42
+ }
43
+ })
44
+
45
+ fastify.post('/', {
46
+ handler () { },
47
+ schema: {
48
+ body: { $ref: 'commonSchema#' },
49
+ headers: { $ref: 'commonSchema#' }
50
+ }
51
+ })
52
+
53
+ fastify.ready(err => {
54
+ t.assert.ifError(err)
55
+ done()
56
+ })
57
+ })
58
+
59
+ test('Example - get schema', (t, done) => {
60
+ t.plan(1)
61
+ const fastify = Fastify()
62
+ fastify.addSchema({
63
+ $id: 'schemaId',
64
+ type: 'object',
65
+ properties: {
66
+ hello: { type: 'string' }
67
+ }
68
+ })
69
+
70
+ const mySchemas = fastify.getSchemas()
71
+ const mySchema = fastify.getSchema('schemaId')
72
+ t.assert.deepStrictEqual(mySchemas.schemaId, mySchema)
73
+ done()
74
+ })
75
+
76
+ test('Example - get schema encapsulated', async t => {
77
+ const fastify = Fastify()
78
+
79
+ fastify.addSchema({ $id: 'one', my: 'hello' })
80
+ // will return only `one` schema
81
+ fastify.get('/', (request, reply) => { reply.send(fastify.getSchemas()) })
82
+
83
+ fastify.register((instance, opts, done) => {
84
+ instance.addSchema({ $id: 'two', my: 'ciao' })
85
+ // will return `one` and `two` schemas
86
+ instance.get('/sub', (request, reply) => { reply.send(instance.getSchemas()) })
87
+
88
+ instance.register((subinstance, opts, done) => {
89
+ subinstance.addSchema({ $id: 'three', my: 'hola' })
90
+ // will return `one`, `two` and `three`
91
+ subinstance.get('/deep', (request, reply) => { reply.send(subinstance.getSchemas()) })
92
+ done()
93
+ })
94
+ done()
95
+ })
96
+
97
+ const r1 = await fastify.inject('/')
98
+ const r2 = await fastify.inject('/sub')
99
+ const r3 = await fastify.inject('/deep')
100
+
101
+ t.assert.deepStrictEqual(Object.keys(r1.json()), ['one'])
102
+ t.assert.deepStrictEqual(Object.keys(r2.json()), ['one', 'two'])
103
+ t.assert.deepStrictEqual(Object.keys(r3.json()), ['one', 'two', 'three'])
104
+ })
105
+
106
+ test('Example - validation', (t, done) => {
107
+ t.plan(1)
108
+ const fastify = Fastify({
109
+ ajv: {
110
+ customOptions: {
111
+ allowUnionTypes: true
112
+ }
113
+ }
114
+ })
115
+ const handler = () => { }
116
+
117
+ const bodyJsonSchema = {
118
+ type: 'object',
119
+ required: ['requiredKey'],
120
+ properties: {
121
+ someKey: { type: 'string' },
122
+ someOtherKey: { type: 'number' },
123
+ requiredKey: {
124
+ type: 'array',
125
+ maxItems: 3,
126
+ items: { type: 'integer' }
127
+ },
128
+ nullableKey: { type: ['number', 'null'] }, // or { type: 'number', nullable: true }
129
+ multipleTypesKey: { type: ['boolean', 'number'] },
130
+ multipleRestrictedTypesKey: {
131
+ oneOf: [
132
+ { type: 'string', maxLength: 5 },
133
+ { type: 'number', minimum: 10 }
134
+ ]
135
+ },
136
+ enumKey: {
137
+ type: 'string',
138
+ enum: ['John', 'Foo']
139
+ },
140
+ notTypeKey: {
141
+ not: { type: 'array' }
142
+ }
143
+ }
144
+ }
145
+
146
+ const queryStringJsonSchema = {
147
+ type: 'object',
148
+ properties: {
149
+ name: { type: 'string' },
150
+ excitement: { type: 'integer' }
151
+ }
152
+ }
153
+
154
+ const paramsJsonSchema = {
155
+ type: 'object',
156
+ properties: {
157
+ par1: { type: 'string' },
158
+ par2: { type: 'number' }
159
+ }
160
+ }
161
+
162
+ const headersJsonSchema = {
163
+ type: 'object',
164
+ properties: {
165
+ 'x-foo': { type: 'string' }
166
+ },
167
+ required: ['x-foo']
168
+ }
169
+
170
+ const schema = {
171
+ body: bodyJsonSchema,
172
+ querystring: queryStringJsonSchema,
173
+ params: paramsJsonSchema,
174
+ headers: headersJsonSchema
175
+ }
176
+
177
+ fastify.post('/the/url', { schema }, handler)
178
+ fastify.ready(err => {
179
+ t.assert.ifError(err)
180
+ done()
181
+ })
182
+ })
183
+
184
+ test('Example - ajv config', (t, done) => {
185
+ t.plan(1)
186
+
187
+ const fastify = Fastify({
188
+ ajv: {
189
+ plugins: [
190
+ require('ajv-merge-patch')
191
+ ]
192
+ }
193
+ })
194
+
195
+ fastify.post('/', {
196
+ handler (req, reply) { reply.send({ ok: 1 }) },
197
+ schema: {
198
+ body: {
199
+ $patch: {
200
+ source: {
201
+ type: 'object',
202
+ properties: {
203
+ q: {
204
+ type: 'string'
205
+ }
206
+ }
207
+ },
208
+ with: [
209
+ {
210
+ op: 'add',
211
+ path: '/properties/q',
212
+ value: { type: 'number' }
213
+ }
214
+ ]
215
+ }
216
+ }
217
+ }
218
+ })
219
+
220
+ fastify.post('/foo', {
221
+ handler (req, reply) { reply.send({ ok: 1 }) },
222
+ schema: {
223
+ body: {
224
+ $merge: {
225
+ source: {
226
+ type: 'object',
227
+ properties: {
228
+ q: {
229
+ type: 'string'
230
+ }
231
+ }
232
+ },
233
+ with: {
234
+ required: ['q']
235
+ }
236
+ }
237
+ }
238
+ }
239
+ })
240
+
241
+ fastify.ready(err => {
242
+ t.assert.ifError(err)
243
+ done()
244
+ })
245
+ })
246
+
247
+ test('Example Joi', (t, done) => {
248
+ t.plan(1)
249
+ const fastify = Fastify()
250
+ const handler = () => { }
251
+
252
+ const Joi = require('joi')
253
+ fastify.post('/the/url', {
254
+ schema: {
255
+ body: Joi.object().keys({
256
+ hello: Joi.string().required()
257
+ }).required()
258
+ },
259
+ validatorCompiler: ({ schema, method, url, httpPart }) => {
260
+ return data => schema.validate(data)
261
+ }
262
+ }, handler)
263
+
264
+ fastify.ready(err => {
265
+ t.assert.ifError(err)
266
+ done()
267
+ })
268
+ })
269
+
270
+ test('Example yup', (t, done) => {
271
+ t.plan(1)
272
+ const fastify = Fastify()
273
+ const handler = () => { }
274
+
275
+ const yup = require('yup')
276
+ // Validation options to match ajv's baseline options used in Fastify
277
+ const yupOptions = {
278
+ strict: false,
279
+ abortEarly: false, // return all errors
280
+ stripUnknown: true, // remove additional properties
281
+ recursive: true
282
+ }
283
+
284
+ fastify.post('/the/url', {
285
+ schema: {
286
+ body: yup.object({
287
+ age: yup.number().integer().required(),
288
+ sub: yup.object().shape({
289
+ name: yup.string().required()
290
+ }).required()
291
+ })
292
+ },
293
+ validatorCompiler: ({ schema, method, url, httpPart }) => {
294
+ return function (data) {
295
+ // with option strict = false, yup `validateSync` function returns the coerced value if validation was successful, or throws if validation failed
296
+ try {
297
+ const result = schema.validateSync(data, yupOptions)
298
+ return { value: result }
299
+ } catch (e) {
300
+ return { error: e }
301
+ }
302
+ }
303
+ }
304
+ }, handler)
305
+
306
+ fastify.ready(err => {
307
+ t.assert.ifError(err)
308
+ done()
309
+ })
310
+ })
311
+
312
+ test('Example - serialization', (t, done) => {
313
+ t.plan(1)
314
+ const fastify = Fastify()
315
+ const handler = () => { }
316
+
317
+ const schema = {
318
+ response: {
319
+ 200: {
320
+ type: 'object',
321
+ properties: {
322
+ value: { type: 'string' },
323
+ otherValue: { type: 'boolean' }
324
+ }
325
+ }
326
+ }
327
+ }
328
+
329
+ fastify.post('/the/url', { schema }, handler)
330
+ fastify.ready(err => {
331
+ t.assert.ifError(err)
332
+ done()
333
+ })
334
+ })
335
+
336
+ test('Example - serialization 2', (t, done) => {
337
+ t.plan(1)
338
+ const fastify = Fastify()
339
+ const handler = () => { }
340
+
341
+ const schema = {
342
+ response: {
343
+ '2xx': {
344
+ type: 'object',
345
+ properties: {
346
+ value: { type: 'string' },
347
+ otherValue: { type: 'boolean' }
348
+ }
349
+ },
350
+ 201: {
351
+ // the contract syntax
352
+ value: { type: 'string' }
353
+ }
354
+ }
355
+ }
356
+
357
+ fastify.post('/the/url', { schema }, handler)
358
+ fastify.ready(err => {
359
+ t.assert.ifError(err)
360
+ done()
361
+ })
362
+ })
363
+
364
+ test('Example - serializator', (t, done) => {
365
+ t.plan(1)
366
+ const fastify = Fastify()
367
+
368
+ fastify.setSerializerCompiler(({ schema, method, url, httpStatus }) => {
369
+ return data => JSON.stringify(data)
370
+ })
371
+
372
+ fastify.get('/user', {
373
+ handler (req, reply) {
374
+ reply.send({ id: 1, name: 'Foo', image: 'BIG IMAGE' })
375
+ },
376
+ schema: {
377
+ response: {
378
+ '2xx': {
379
+ type: 'object',
380
+ properties: {
381
+ id: { type: 'number' },
382
+ name: { type: 'string' }
383
+ }
384
+ }
385
+ }
386
+ }
387
+ })
388
+
389
+ fastify.ready(err => {
390
+ t.assert.ifError(err)
391
+ done()
392
+ })
393
+ })
394
+
395
+ test('Example - schemas examples', (t, done) => {
396
+ t.plan(1)
397
+ const fastify = Fastify()
398
+ const handler = () => { }
399
+
400
+ fastify.addSchema({
401
+ $id: 'http://foo/common.json',
402
+ type: 'object',
403
+ definitions: {
404
+ foo: {
405
+ $id: '#address',
406
+ type: 'object',
407
+ properties: {
408
+ city: { type: 'string' }
409
+ }
410
+ }
411
+ }
412
+ })
413
+
414
+ fastify.addSchema({
415
+ $id: 'http://foo/shared.json',
416
+ type: 'object',
417
+ definitions: {
418
+ foo: {
419
+ type: 'object',
420
+ properties: {
421
+ city: { type: 'string' }
422
+ }
423
+ }
424
+ }
425
+ })
426
+
427
+ const refToId = {
428
+ type: 'object',
429
+ definitions: {
430
+ foo: {
431
+ $id: '#address',
432
+ type: 'object',
433
+ properties: {
434
+ city: { type: 'string' }
435
+ }
436
+ }
437
+ },
438
+ properties: {
439
+ home: { $ref: '#address' },
440
+ work: { $ref: '#address' }
441
+ }
442
+ }
443
+
444
+ const refToDefinitions = {
445
+ type: 'object',
446
+ definitions: {
447
+ foo: {
448
+ $id: '#address',
449
+ type: 'object',
450
+ properties: {
451
+ city: { type: 'string' }
452
+ }
453
+ }
454
+ },
455
+ properties: {
456
+ home: { $ref: '#/definitions/foo' },
457
+ work: { $ref: '#/definitions/foo' }
458
+ }
459
+ }
460
+
461
+ const refToSharedSchemaId = {
462
+ type: 'object',
463
+ properties: {
464
+ home: { $ref: 'http://foo/common.json#address' },
465
+ work: { $ref: 'http://foo/common.json#address' }
466
+ }
467
+ }
468
+
469
+ const refToSharedSchemaDefinitions = {
470
+ type: 'object',
471
+ properties: {
472
+ home: { $ref: 'http://foo/shared.json#/definitions/foo' },
473
+ work: { $ref: 'http://foo/shared.json#/definitions/foo' }
474
+ }
475
+ }
476
+
477
+ fastify.post('/', {
478
+ handler,
479
+ schema: {
480
+ body: refToId,
481
+ headers: refToDefinitions,
482
+ params: refToSharedSchemaId,
483
+ query: refToSharedSchemaDefinitions
484
+ }
485
+
486
+ })
487
+
488
+ fastify.ready(err => {
489
+ t.assert.ifError(err)
490
+ done()
491
+ })
492
+ })
493
+
494
+ test('should return custom error messages with ajv-errors', (t, done) => {
495
+ t.plan(3)
496
+
497
+ const fastify = Fastify({
498
+ ajv: {
499
+ customOptions: { allErrors: true },
500
+ plugins: [
501
+ require('ajv-errors')
502
+ ]
503
+ }
504
+ })
505
+
506
+ const schema = {
507
+ body: {
508
+ type: 'object',
509
+ properties: {
510
+ name: { type: 'string' },
511
+ work: { type: 'string' },
512
+ age: {
513
+ type: 'number',
514
+ errorMessage: {
515
+ type: 'bad age - should be num'
516
+ }
517
+ }
518
+ },
519
+ required: ['name', 'work'],
520
+ errorMessage: {
521
+ required: {
522
+ name: 'name please',
523
+ work: 'work please',
524
+ age: 'age please'
525
+ }
526
+ }
527
+ }
528
+ }
529
+
530
+ fastify.post('/', { schema }, function (req, reply) {
531
+ reply.code(200).send(req.body.name)
532
+ })
533
+
534
+ fastify.inject({
535
+ method: 'POST',
536
+ payload: {
537
+ hello: 'salman',
538
+ age: 'bad'
539
+ },
540
+ url: '/'
541
+ }, (err, res) => {
542
+ t.assert.ifError(err)
543
+ t.assert.deepStrictEqual(JSON.parse(res.payload), {
544
+ statusCode: 400,
545
+ code: 'FST_ERR_VALIDATION',
546
+ error: 'Bad Request',
547
+ message: 'body/age bad age - should be num, body name please, body work please'
548
+ })
549
+ t.assert.strictEqual(res.statusCode, 400)
550
+ done()
551
+ })
552
+ })
553
+
554
+ test('should be able to handle formats of ajv-formats when added by plugins option', (t, done) => {
555
+ t.plan(3)
556
+
557
+ const fastify = Fastify({
558
+ ajv: {
559
+ plugins: [
560
+ require('ajv-formats')
561
+ ]
562
+ }
563
+ })
564
+
565
+ const schema = {
566
+ body: {
567
+ type: 'object',
568
+ properties: {
569
+ id: { type: 'string', format: 'uuid' },
570
+ email: { type: 'string', format: 'email' }
571
+ },
572
+ required: ['id', 'email']
573
+ }
574
+ }
575
+
576
+ fastify.post('/', { schema }, function (req, reply) {
577
+ reply.code(200).send(req.body.id)
578
+ })
579
+
580
+ fastify.inject({
581
+ method: 'POST',
582
+ payload: {
583
+ id: '254381a5-888c-4b41-8116-e3b1a54980bd',
584
+ email: 'info@fastify.dev'
585
+ },
586
+ url: '/'
587
+ }, (_err, res) => {
588
+ t.assert.strictEqual(res.body, '254381a5-888c-4b41-8116-e3b1a54980bd')
589
+ t.assert.strictEqual(res.statusCode, 200)
590
+ })
591
+
592
+ fastify.inject({
593
+ method: 'POST',
594
+ payload: {
595
+ id: 'invalid',
596
+ email: 'info@fastify.dev'
597
+ },
598
+ url: '/'
599
+ }, (_err, res) => {
600
+ t.assert.deepStrictEqual(JSON.parse(res.payload), {
601
+ statusCode: 400,
602
+ code: 'FST_ERR_VALIDATION',
603
+ error: 'Bad Request',
604
+ message: 'body/id must match format "uuid"'
605
+ })
606
+ done()
607
+ })
608
+ })
609
+
610
+ test('should return localized error messages with ajv-i18n', (t, done) => {
611
+ t.plan(3)
612
+
613
+ const schema = {
614
+ body: {
615
+ type: 'object',
616
+ properties: {
617
+ name: { type: 'string' },
618
+ work: { type: 'string' }
619
+ },
620
+ required: ['name', 'work']
621
+ }
622
+ }
623
+
624
+ const fastify = Fastify({
625
+ ajv: {
626
+ customOptions: { allErrors: true }
627
+ }
628
+ })
629
+
630
+ fastify.setErrorHandler(function (error, request, reply) {
631
+ if (error.validation) {
632
+ localize.ru(error.validation)
633
+ reply.status(400).send(error.validation)
634
+ return
635
+ }
636
+ reply.send(error)
637
+ })
638
+
639
+ fastify.post('/', { schema }, function (req, reply) {
640
+ reply.code(200).send(req.body.name)
641
+ })
642
+
643
+ fastify.inject({
644
+ method: 'POST',
645
+ payload: {
646
+ name: 'salman'
647
+ },
648
+ url: '/'
649
+ }, (err, res) => {
650
+ t.assert.ifError(err)
651
+ t.assert.deepStrictEqual(JSON.parse(res.payload), [{
652
+ instancePath: '',
653
+ keyword: 'required',
654
+ message: 'должно иметь обязательное поле work',
655
+ params: { missingProperty: 'work' },
656
+ schemaPath: '#/required'
657
+ }])
658
+ t.assert.strictEqual(res.statusCode, 400)
659
+ done()
660
+ })
661
+ })