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.
- package/.borp.yaml +3 -0
- package/.markdownlint-cli2.yaml +22 -0
- package/.prettierignore +1 -0
- package/GOVERNANCE.md +4 -0
- package/LICENSE +21 -0
- package/PROJECT_CHARTER.md +126 -0
- package/README.md +423 -0
- package/SECURITY.md +220 -0
- package/SPONSORS.md +24 -0
- package/build/build-error-serializer.js +35 -0
- package/build/build-validation.js +169 -0
- package/build/sync-version.js +11 -0
- package/docs/Guides/Benchmarking.md +60 -0
- package/docs/Guides/Database.md +321 -0
- package/docs/Guides/Delay-Accepting-Requests.md +608 -0
- package/docs/Guides/Detecting-When-Clients-Abort.md +172 -0
- package/docs/Guides/Ecosystem.md +726 -0
- package/docs/Guides/Fluent-Schema.md +127 -0
- package/docs/Guides/Getting-Started.md +620 -0
- package/docs/Guides/Index.md +43 -0
- package/docs/Guides/Migration-Guide-V3.md +287 -0
- package/docs/Guides/Migration-Guide-V4.md +267 -0
- package/docs/Guides/Migration-Guide-V5.md +727 -0
- package/docs/Guides/Plugins-Guide.md +520 -0
- package/docs/Guides/Prototype-Poisoning.md +383 -0
- package/docs/Guides/Recommendations.md +378 -0
- package/docs/Guides/Serverless.md +604 -0
- package/docs/Guides/Style-Guide.md +246 -0
- package/docs/Guides/Testing.md +481 -0
- package/docs/Guides/Write-Plugin.md +103 -0
- package/docs/Guides/Write-Type-Provider.md +34 -0
- package/docs/Reference/ContentTypeParser.md +271 -0
- package/docs/Reference/Decorators.md +436 -0
- package/docs/Reference/Encapsulation.md +194 -0
- package/docs/Reference/Errors.md +377 -0
- package/docs/Reference/HTTP2.md +94 -0
- package/docs/Reference/Hooks.md +958 -0
- package/docs/Reference/Index.md +73 -0
- package/docs/Reference/LTS.md +86 -0
- package/docs/Reference/Lifecycle.md +99 -0
- package/docs/Reference/Logging.md +268 -0
- package/docs/Reference/Middleware.md +79 -0
- package/docs/Reference/Plugins.md +245 -0
- package/docs/Reference/Principles.md +73 -0
- package/docs/Reference/Reply.md +1001 -0
- package/docs/Reference/Request.md +295 -0
- package/docs/Reference/Routes.md +802 -0
- package/docs/Reference/Server.md +2389 -0
- package/docs/Reference/Type-Providers.md +256 -0
- package/docs/Reference/TypeScript.md +1729 -0
- package/docs/Reference/Validation-and-Serialization.md +1130 -0
- package/docs/Reference/Warnings.md +58 -0
- package/docs/index.md +24 -0
- package/docs/resources/encapsulation_context.drawio +1 -0
- package/docs/resources/encapsulation_context.svg +3 -0
- package/eslint.config.js +35 -0
- package/examples/asyncawait.js +38 -0
- package/examples/benchmark/body.json +3 -0
- package/examples/benchmark/hooks-benchmark-async-await.js +44 -0
- package/examples/benchmark/hooks-benchmark.js +52 -0
- package/examples/benchmark/parser.js +47 -0
- package/examples/benchmark/simple.js +30 -0
- package/examples/benchmark/webstream.js +27 -0
- package/examples/hooks.js +91 -0
- package/examples/http2.js +39 -0
- package/examples/https.js +38 -0
- package/examples/parser.js +53 -0
- package/examples/plugin.js +12 -0
- package/examples/route-prefix.js +38 -0
- package/examples/shared-schema.js +38 -0
- package/examples/simple-stream.js +20 -0
- package/examples/simple.js +32 -0
- package/examples/simple.mjs +27 -0
- package/examples/typescript-server.ts +79 -0
- package/examples/use-plugin.js +29 -0
- package/fastify.d.ts +253 -0
- package/fastify.js +985 -0
- package/integration/server.js +29 -0
- package/integration/test.sh +23 -0
- package/lib/config-validator.js +1266 -0
- package/lib/content-type-parser.js +413 -0
- package/lib/content-type.js +160 -0
- package/lib/context.js +98 -0
- package/lib/decorate.js +152 -0
- package/lib/error-handler.js +173 -0
- package/lib/error-serializer.js +134 -0
- package/lib/error-status.js +14 -0
- package/lib/errors.js +516 -0
- package/lib/four-oh-four.js +190 -0
- package/lib/handle-request.js +195 -0
- package/lib/head-route.js +45 -0
- package/lib/hooks.js +429 -0
- package/lib/initial-config-validation.js +37 -0
- package/lib/logger-factory.js +136 -0
- package/lib/logger-pino.js +68 -0
- package/lib/noop-set.js +10 -0
- package/lib/plugin-override.js +90 -0
- package/lib/plugin-utils.js +169 -0
- package/lib/promise.js +23 -0
- package/lib/reply.js +1030 -0
- package/lib/req-id-gen-factory.js +52 -0
- package/lib/request.js +391 -0
- package/lib/route.js +686 -0
- package/lib/schema-controller.js +164 -0
- package/lib/schemas.js +207 -0
- package/lib/server.js +441 -0
- package/lib/symbols.js +71 -0
- package/lib/validation.js +280 -0
- package/lib/warnings.js +57 -0
- package/lib/wrap-thenable.js +84 -0
- package/package.json +225 -0
- package/scripts/validate-ecosystem-links.js +179 -0
- package/test/404s.test.js +2035 -0
- package/test/500s.test.js +422 -0
- package/test/allow-unsafe-regex.test.js +92 -0
- package/test/als.test.js +65 -0
- package/test/async-await.test.js +705 -0
- package/test/async-dispose.test.js +20 -0
- package/test/async_hooks.test.js +52 -0
- package/test/body-limit.test.js +224 -0
- package/test/buffer.test.js +74 -0
- package/test/build/error-serializer.test.js +36 -0
- package/test/build/version.test.js +14 -0
- package/test/build-certificate.js +109 -0
- package/test/bundler/README.md +29 -0
- package/test/bundler/esbuild/bundler-test.js +32 -0
- package/test/bundler/esbuild/package.json +10 -0
- package/test/bundler/esbuild/src/fail-plugin-version.js +14 -0
- package/test/bundler/esbuild/src/index.js +9 -0
- package/test/bundler/webpack/bundler-test.js +32 -0
- package/test/bundler/webpack/package.json +11 -0
- package/test/bundler/webpack/src/fail-plugin-version.js +14 -0
- package/test/bundler/webpack/src/index.js +9 -0
- package/test/bundler/webpack/webpack.config.js +15 -0
- package/test/case-insensitive.test.js +102 -0
- package/test/chainable.test.js +40 -0
- package/test/child-logger-factory.test.js +128 -0
- package/test/client-timeout.test.js +38 -0
- package/test/close-pipelining.test.js +78 -0
- package/test/close.test.js +706 -0
- package/test/conditional-pino.test.js +47 -0
- package/test/connection-timeout.test.js +42 -0
- package/test/constrained-routes.test.js +1138 -0
- package/test/content-length.test.js +174 -0
- package/test/content-parser.test.js +739 -0
- package/test/content-type.test.js +181 -0
- package/test/context-config.test.js +164 -0
- package/test/custom-http-server.test.js +118 -0
- package/test/custom-parser-async.test.js +59 -0
- package/test/custom-parser.0.test.js +701 -0
- package/test/custom-parser.1.test.js +266 -0
- package/test/custom-parser.2.test.js +91 -0
- package/test/custom-parser.3.test.js +208 -0
- package/test/custom-parser.4.test.js +218 -0
- package/test/custom-parser.5.test.js +130 -0
- package/test/custom-querystring-parser.test.js +129 -0
- package/test/decorator.test.js +1330 -0
- package/test/delete.test.js +344 -0
- package/test/diagnostics-channel/404.test.js +49 -0
- package/test/diagnostics-channel/async-delay-request.test.js +65 -0
- package/test/diagnostics-channel/async-request.test.js +64 -0
- package/test/diagnostics-channel/error-before-handler.test.js +35 -0
- package/test/diagnostics-channel/error-request.test.js +53 -0
- package/test/diagnostics-channel/error-status.test.js +123 -0
- package/test/diagnostics-channel/init.test.js +50 -0
- package/test/diagnostics-channel/sync-delay-request.test.js +49 -0
- package/test/diagnostics-channel/sync-request-reply.test.js +51 -0
- package/test/diagnostics-channel/sync-request.test.js +54 -0
- package/test/encapsulated-child-logger-factory.test.js +69 -0
- package/test/encapsulated-error-handler.test.js +237 -0
- package/test/esm/errorCodes.test.mjs +10 -0
- package/test/esm/esm.test.mjs +13 -0
- package/test/esm/index.test.js +8 -0
- package/test/esm/named-exports.mjs +14 -0
- package/test/esm/other.mjs +8 -0
- package/test/esm/plugin.mjs +8 -0
- package/test/fastify-instance.test.js +300 -0
- package/test/find-route.test.js +152 -0
- package/test/fluent-schema.test.js +209 -0
- package/test/genReqId.test.js +426 -0
- package/test/handler-context.test.js +45 -0
- package/test/handler-timeout.test.js +367 -0
- package/test/has-route.test.js +88 -0
- package/test/header-overflow.test.js +55 -0
- package/test/helper.js +496 -0
- package/test/hooks-async.test.js +1099 -0
- package/test/hooks.on-listen.test.js +1162 -0
- package/test/hooks.on-ready.test.js +421 -0
- package/test/hooks.test.js +3578 -0
- package/test/http-methods/copy.test.js +35 -0
- package/test/http-methods/custom-http-methods.test.js +114 -0
- package/test/http-methods/get.test.js +412 -0
- package/test/http-methods/head.test.js +263 -0
- package/test/http-methods/lock.test.js +108 -0
- package/test/http-methods/mkcalendar.test.js +143 -0
- package/test/http-methods/mkcol.test.js +35 -0
- package/test/http-methods/move.test.js +42 -0
- package/test/http-methods/propfind.test.js +136 -0
- package/test/http-methods/proppatch.test.js +105 -0
- package/test/http-methods/report.test.js +142 -0
- package/test/http-methods/search.test.js +233 -0
- package/test/http-methods/trace.test.js +21 -0
- package/test/http-methods/unlock.test.js +38 -0
- package/test/http2/closing.test.js +270 -0
- package/test/http2/constraint.test.js +109 -0
- package/test/http2/head.test.js +34 -0
- package/test/http2/plain.test.js +68 -0
- package/test/http2/secure-with-fallback.test.js +113 -0
- package/test/http2/secure.test.js +67 -0
- package/test/http2/unknown-http-method.test.js +34 -0
- package/test/https/custom-https-server.test.js +58 -0
- package/test/https/https.test.js +136 -0
- package/test/imports.test.js +17 -0
- package/test/inject.test.js +502 -0
- package/test/input-validation.js +335 -0
- package/test/internals/all.test.js +38 -0
- package/test/internals/content-type-parser.test.js +111 -0
- package/test/internals/context.test.js +31 -0
- package/test/internals/decorator.test.js +156 -0
- package/test/internals/errors.test.js +982 -0
- package/test/internals/handle-request.test.js +270 -0
- package/test/internals/hook-runner.test.js +449 -0
- package/test/internals/hooks.test.js +96 -0
- package/test/internals/initial-config.test.js +383 -0
- package/test/internals/logger.test.js +163 -0
- package/test/internals/plugin.test.js +170 -0
- package/test/internals/promise.test.js +63 -0
- package/test/internals/reply-serialize.test.js +714 -0
- package/test/internals/reply.test.js +1920 -0
- package/test/internals/req-id-gen-factory.test.js +133 -0
- package/test/internals/request-validate.test.js +1402 -0
- package/test/internals/request.test.js +506 -0
- package/test/internals/schema-controller-perf.test.js +40 -0
- package/test/internals/server.test.js +91 -0
- package/test/internals/validation.test.js +352 -0
- package/test/issue-4959.test.js +118 -0
- package/test/keep-alive-timeout.test.js +42 -0
- package/test/listen.1.test.js +154 -0
- package/test/listen.2.test.js +113 -0
- package/test/listen.3.test.js +83 -0
- package/test/listen.4.test.js +168 -0
- package/test/listen.5.test.js +122 -0
- package/test/logger/instantiation.test.js +341 -0
- package/test/logger/logger-test-utils.js +47 -0
- package/test/logger/logging.test.js +460 -0
- package/test/logger/options.test.js +579 -0
- package/test/logger/request.test.js +292 -0
- package/test/logger/response.test.js +183 -0
- package/test/logger/tap-parallel-not-ok +0 -0
- package/test/max-requests-per-socket.test.js +113 -0
- package/test/middleware.test.js +37 -0
- package/test/noop-set.test.js +19 -0
- package/test/nullable-validation.test.js +187 -0
- package/test/options.error-handler.test.js +5 -0
- package/test/options.test.js +5 -0
- package/test/output-validation.test.js +140 -0
- package/test/patch.error-handler.test.js +5 -0
- package/test/patch.test.js +5 -0
- package/test/plugin.1.test.js +230 -0
- package/test/plugin.2.test.js +314 -0
- package/test/plugin.3.test.js +287 -0
- package/test/plugin.4.test.js +504 -0
- package/test/plugin.helper.js +8 -0
- package/test/plugin.name.display.js +10 -0
- package/test/post-empty-body.test.js +38 -0
- package/test/pretty-print.test.js +366 -0
- package/test/promises.test.js +125 -0
- package/test/proto-poisoning.test.js +145 -0
- package/test/put.error-handler.test.js +5 -0
- package/test/put.test.js +5 -0
- package/test/register.test.js +184 -0
- package/test/reply-code.test.js +148 -0
- package/test/reply-early-hints.test.js +100 -0
- package/test/reply-error.test.js +815 -0
- package/test/reply-trailers.test.js +445 -0
- package/test/reply-web-stream-locked.test.js +37 -0
- package/test/request-error.test.js +624 -0
- package/test/request-header-host.test.js +339 -0
- package/test/request-id.test.js +118 -0
- package/test/request-timeout.test.js +53 -0
- package/test/route-hooks.test.js +635 -0
- package/test/route-prefix.test.js +904 -0
- package/test/route-shorthand.test.js +48 -0
- package/test/route.1.test.js +259 -0
- package/test/route.2.test.js +100 -0
- package/test/route.3.test.js +213 -0
- package/test/route.4.test.js +127 -0
- package/test/route.5.test.js +211 -0
- package/test/route.6.test.js +306 -0
- package/test/route.7.test.js +406 -0
- package/test/route.8.test.js +225 -0
- package/test/router-options.test.js +1108 -0
- package/test/same-shape.test.js +124 -0
- package/test/schema-examples.test.js +661 -0
- package/test/schema-feature.test.js +2198 -0
- package/test/schema-serialization.test.js +1171 -0
- package/test/schema-special-usage.test.js +1348 -0
- package/test/schema-validation.test.js +1572 -0
- package/test/scripts/validate-ecosystem-links.test.js +339 -0
- package/test/serialize-response.test.js +186 -0
- package/test/server.test.js +347 -0
- package/test/set-error-handler.test.js +69 -0
- package/test/skip-reply-send.test.js +317 -0
- package/test/stream-serializers.test.js +40 -0
- package/test/stream.1.test.js +94 -0
- package/test/stream.2.test.js +129 -0
- package/test/stream.3.test.js +198 -0
- package/test/stream.4.test.js +176 -0
- package/test/stream.5.test.js +188 -0
- package/test/sync-routes.test.js +32 -0
- package/test/throw.test.js +359 -0
- package/test/toolkit.js +63 -0
- package/test/trust-proxy.test.js +162 -0
- package/test/type-provider.test.js +22 -0
- package/test/types/content-type-parser.test-d.ts +72 -0
- package/test/types/decorate-request-reply.test-d.ts +18 -0
- package/test/types/dummy-plugin.ts +9 -0
- package/test/types/errors.test-d.ts +90 -0
- package/test/types/fastify.test-d.ts +352 -0
- package/test/types/hooks.test-d.ts +550 -0
- package/test/types/import.ts +2 -0
- package/test/types/instance.test-d.ts +588 -0
- package/test/types/logger.test-d.ts +277 -0
- package/test/types/plugin.test-d.ts +97 -0
- package/test/types/register.test-d.ts +237 -0
- package/test/types/reply.test-d.ts +254 -0
- package/test/types/request.test-d.ts +188 -0
- package/test/types/route.test-d.ts +553 -0
- package/test/types/schema.test-d.ts +135 -0
- package/test/types/serverFactory.test-d.ts +37 -0
- package/test/types/type-provider.test-d.ts +1213 -0
- package/test/types/using.test-d.ts +17 -0
- package/test/upgrade.test.js +52 -0
- package/test/url-rewriting.test.js +122 -0
- package/test/use-semicolon-delimiter.test.js +168 -0
- package/test/validation-error-handling.test.js +900 -0
- package/test/versioned-routes.test.js +603 -0
- package/test/web-api.test.js +616 -0
- package/test/wrap-thenable.test.js +30 -0
- package/types/content-type-parser.d.ts +75 -0
- package/types/context.d.ts +22 -0
- package/types/errors.d.ts +92 -0
- package/types/hooks.d.ts +875 -0
- package/types/instance.d.ts +609 -0
- package/types/logger.d.ts +107 -0
- package/types/plugin.d.ts +44 -0
- package/types/register.d.ts +42 -0
- package/types/reply.d.ts +81 -0
- package/types/request.d.ts +95 -0
- package/types/route.d.ts +199 -0
- package/types/schema.d.ts +61 -0
- package/types/server-factory.d.ts +19 -0
- package/types/type-provider.d.ts +130 -0
- package/types/utils.d.ts +98 -0
|
@@ -0,0 +1,321 @@
|
|
|
1
|
+
<h1 align="center">Fastify</h1>
|
|
2
|
+
|
|
3
|
+
## Database
|
|
4
|
+
|
|
5
|
+
Fastify's ecosystem provides a handful of
|
|
6
|
+
plugins for connecting to various database engines.
|
|
7
|
+
This guide covers engines that have Fastify
|
|
8
|
+
plugins maintained within the Fastify organization.
|
|
9
|
+
|
|
10
|
+
> If a plugin for your database of choice does not exist
|
|
11
|
+
> you can still use the database as Fastify is database agnostic.
|
|
12
|
+
> By following the examples of the database plugins listed in this guide,
|
|
13
|
+
> a plugin can be written for the missing database engine.
|
|
14
|
+
|
|
15
|
+
> If you would like to write your own Fastify plugin
|
|
16
|
+
> please take a look at the [plugins guide](./Plugins-Guide.md)
|
|
17
|
+
|
|
18
|
+
### [MySQL](https://github.com/fastify/fastify-mysql)
|
|
19
|
+
|
|
20
|
+
Install the plugin by running `npm i @fastify/mysql`.
|
|
21
|
+
|
|
22
|
+
*Usage:*
|
|
23
|
+
|
|
24
|
+
```javascript
|
|
25
|
+
const fastify = require('fastify')()
|
|
26
|
+
|
|
27
|
+
fastify.register(require('@fastify/mysql'), {
|
|
28
|
+
connectionString: 'mysql://root@localhost/mysql'
|
|
29
|
+
})
|
|
30
|
+
|
|
31
|
+
fastify.get('/user/:id', function(req, reply) {
|
|
32
|
+
fastify.mysql.query(
|
|
33
|
+
'SELECT id, username, hash, salt FROM users WHERE id=?', [req.params.id],
|
|
34
|
+
function onResult (err, result) {
|
|
35
|
+
reply.send(err || result)
|
|
36
|
+
}
|
|
37
|
+
)
|
|
38
|
+
})
|
|
39
|
+
|
|
40
|
+
fastify.listen({ port: 3000 }, err => {
|
|
41
|
+
if (err) throw err
|
|
42
|
+
console.log(`server listening on ${fastify.server.address().port}`)
|
|
43
|
+
})
|
|
44
|
+
```
|
|
45
|
+
|
|
46
|
+
### [Postgres](https://github.com/fastify/fastify-postgres)
|
|
47
|
+
Install the plugin by running `npm i pg @fastify/postgres`.
|
|
48
|
+
|
|
49
|
+
*Example*:
|
|
50
|
+
|
|
51
|
+
```javascript
|
|
52
|
+
const fastify = require('fastify')()
|
|
53
|
+
|
|
54
|
+
fastify.register(require('@fastify/postgres'), {
|
|
55
|
+
connectionString: 'postgres://postgres@localhost/postgres'
|
|
56
|
+
})
|
|
57
|
+
|
|
58
|
+
fastify.get('/user/:id', function (req, reply) {
|
|
59
|
+
fastify.pg.query(
|
|
60
|
+
'SELECT id, username, hash, salt FROM users WHERE id=$1', [req.params.id],
|
|
61
|
+
function onResult (err, result) {
|
|
62
|
+
reply.send(err || result)
|
|
63
|
+
}
|
|
64
|
+
)
|
|
65
|
+
})
|
|
66
|
+
|
|
67
|
+
fastify.listen({ port: 3000 }, err => {
|
|
68
|
+
if (err) throw err
|
|
69
|
+
console.log(`server listening on ${fastify.server.address().port}`)
|
|
70
|
+
})
|
|
71
|
+
```
|
|
72
|
+
|
|
73
|
+
### [Redis](https://github.com/fastify/fastify-redis)
|
|
74
|
+
Install the plugin by running `npm i @fastify/redis`
|
|
75
|
+
|
|
76
|
+
*Usage:*
|
|
77
|
+
|
|
78
|
+
```javascript
|
|
79
|
+
'use strict'
|
|
80
|
+
|
|
81
|
+
const fastify = require('fastify')()
|
|
82
|
+
|
|
83
|
+
fastify.register(require('@fastify/redis'), { host: '127.0.0.1' })
|
|
84
|
+
// or
|
|
85
|
+
fastify.register(require('@fastify/redis'), { url: 'redis://127.0.0.1', /* other redis options */ })
|
|
86
|
+
|
|
87
|
+
fastify.get('/foo', function (req, reply) {
|
|
88
|
+
const { redis } = fastify
|
|
89
|
+
redis.get(req.query.key, (err, val) => {
|
|
90
|
+
reply.send(err || val)
|
|
91
|
+
})
|
|
92
|
+
})
|
|
93
|
+
|
|
94
|
+
fastify.post('/foo', function (req, reply) {
|
|
95
|
+
const { redis } = fastify
|
|
96
|
+
redis.set(req.body.key, req.body.value, (err) => {
|
|
97
|
+
reply.send(err || { status: 'ok' })
|
|
98
|
+
})
|
|
99
|
+
})
|
|
100
|
+
|
|
101
|
+
fastify.listen({ port: 3000 }, err => {
|
|
102
|
+
if (err) throw err
|
|
103
|
+
console.log(`server listening on ${fastify.server.address().port}`)
|
|
104
|
+
})
|
|
105
|
+
```
|
|
106
|
+
|
|
107
|
+
By default `@fastify/redis` doesn't close
|
|
108
|
+
the client connection when Fastify server shuts down.
|
|
109
|
+
To opt-in to this behavior, register the client like so:
|
|
110
|
+
|
|
111
|
+
```javascript
|
|
112
|
+
fastify.register(require('@fastify/redis'), {
|
|
113
|
+
client: redis,
|
|
114
|
+
closeClient: true
|
|
115
|
+
})
|
|
116
|
+
```
|
|
117
|
+
|
|
118
|
+
### [Mongo](https://github.com/fastify/fastify-mongodb)
|
|
119
|
+
Install the plugin by running `npm i @fastify/mongodb`
|
|
120
|
+
|
|
121
|
+
*Usage:*
|
|
122
|
+
```javascript
|
|
123
|
+
const fastify = require('fastify')()
|
|
124
|
+
|
|
125
|
+
fastify.register(require('@fastify/mongodb'), {
|
|
126
|
+
// force to close the mongodb connection when app stopped
|
|
127
|
+
// the default value is false
|
|
128
|
+
forceClose: true,
|
|
129
|
+
|
|
130
|
+
url: 'mongodb://mongo/mydb'
|
|
131
|
+
})
|
|
132
|
+
|
|
133
|
+
fastify.get('/user/:id', async function (req, reply) {
|
|
134
|
+
// Or this.mongo.client.db('mydb').collection('users')
|
|
135
|
+
const users = this.mongo.db.collection('users')
|
|
136
|
+
|
|
137
|
+
// if the id is an ObjectId format, you need to create a new ObjectId
|
|
138
|
+
const id = this.mongo.ObjectId(req.params.id)
|
|
139
|
+
try {
|
|
140
|
+
const user = await users.findOne({ id })
|
|
141
|
+
return user
|
|
142
|
+
} catch (err) {
|
|
143
|
+
return err
|
|
144
|
+
}
|
|
145
|
+
})
|
|
146
|
+
|
|
147
|
+
fastify.listen({ port: 3000 }, err => {
|
|
148
|
+
if (err) throw err
|
|
149
|
+
})
|
|
150
|
+
```
|
|
151
|
+
|
|
152
|
+
### [LevelDB](https://github.com/fastify/fastify-leveldb)
|
|
153
|
+
Install the plugin by running `npm i @fastify/leveldb`
|
|
154
|
+
|
|
155
|
+
*Usage:*
|
|
156
|
+
```javascript
|
|
157
|
+
const fastify = require('fastify')()
|
|
158
|
+
|
|
159
|
+
fastify.register(
|
|
160
|
+
require('@fastify/leveldb'),
|
|
161
|
+
{ name: 'db' }
|
|
162
|
+
)
|
|
163
|
+
|
|
164
|
+
fastify.get('/foo', async function (req, reply) {
|
|
165
|
+
const val = await this.level.db.get(req.query.key)
|
|
166
|
+
return val
|
|
167
|
+
})
|
|
168
|
+
|
|
169
|
+
fastify.post('/foo', async function (req, reply) {
|
|
170
|
+
await this.level.db.put(req.body.key, req.body.value)
|
|
171
|
+
return { status: 'ok' }
|
|
172
|
+
})
|
|
173
|
+
|
|
174
|
+
fastify.listen({ port: 3000 }, err => {
|
|
175
|
+
if (err) throw err
|
|
176
|
+
console.log(`server listening on ${fastify.server.address().port}`)
|
|
177
|
+
})
|
|
178
|
+
```
|
|
179
|
+
|
|
180
|
+
### Writing plugin for a database library
|
|
181
|
+
We could write a plugin for a database
|
|
182
|
+
library too (e.g. Knex, Prisma, or TypeORM).
|
|
183
|
+
We will use [Knex](https://knexjs.org/) in our example.
|
|
184
|
+
|
|
185
|
+
```javascript
|
|
186
|
+
'use strict'
|
|
187
|
+
|
|
188
|
+
const fp = require('fastify-plugin')
|
|
189
|
+
const knex = require('knex')
|
|
190
|
+
|
|
191
|
+
function knexPlugin(fastify, options, done) {
|
|
192
|
+
if(!fastify.knex) {
|
|
193
|
+
const knex = knex(options)
|
|
194
|
+
fastify.decorate('knex', knex)
|
|
195
|
+
|
|
196
|
+
fastify.addHook('onClose', (fastify, done) => {
|
|
197
|
+
if (fastify.knex === knex) {
|
|
198
|
+
fastify.knex.destroy(done)
|
|
199
|
+
}
|
|
200
|
+
})
|
|
201
|
+
}
|
|
202
|
+
|
|
203
|
+
done()
|
|
204
|
+
}
|
|
205
|
+
|
|
206
|
+
export default fp(knexPlugin, { name: 'fastify-knex-example' })
|
|
207
|
+
```
|
|
208
|
+
|
|
209
|
+
### Writing a plugin for a database engine
|
|
210
|
+
|
|
211
|
+
In this example, we will create a basic Fastify MySQL plugin from scratch (it is
|
|
212
|
+
a stripped-down example, please use the official plugin in production).
|
|
213
|
+
|
|
214
|
+
```javascript
|
|
215
|
+
const fp = require('fastify-plugin')
|
|
216
|
+
const mysql = require('mysql2/promise')
|
|
217
|
+
|
|
218
|
+
function fastifyMysql(fastify, options, done) {
|
|
219
|
+
const connection = mysql.createConnection(options)
|
|
220
|
+
|
|
221
|
+
if (!fastify.mysql) {
|
|
222
|
+
fastify.decorate('mysql', connection)
|
|
223
|
+
}
|
|
224
|
+
|
|
225
|
+
fastify.addHook('onClose', (fastify, done) => connection.end().then(done).catch(done))
|
|
226
|
+
|
|
227
|
+
done()
|
|
228
|
+
}
|
|
229
|
+
|
|
230
|
+
export default fp(fastifyMysql, { name: 'fastify-mysql-example' })
|
|
231
|
+
```
|
|
232
|
+
|
|
233
|
+
### Migrations
|
|
234
|
+
|
|
235
|
+
Database schema migrations are an integral part of database management and
|
|
236
|
+
development. Migrations provide a repeatable and testable way to modify a
|
|
237
|
+
database's schema and prevent data loss.
|
|
238
|
+
|
|
239
|
+
As stated at the beginning of the guide, Fastify is database agnostic and any
|
|
240
|
+
Node.js database migration tool can be used with it. We will give an example of
|
|
241
|
+
using [Postgrator](https://www.npmjs.com/package/postgrator) which has support
|
|
242
|
+
for Postgres, MySQL, SQL Server and SQLite. For MongoDB migrations, please check
|
|
243
|
+
[migrate-mongo](https://www.npmjs.com/package/migrate-mongo).
|
|
244
|
+
|
|
245
|
+
#### [Postgrator](https://www.npmjs.com/package/postgrator)
|
|
246
|
+
|
|
247
|
+
Postgrator is Node.js SQL migration tool that uses a directory of SQL scripts to
|
|
248
|
+
alter the database schema. Each file in a migrations folder needs to follow the
|
|
249
|
+
pattern: ` [version].[action].[optional-description].sql`.
|
|
250
|
+
|
|
251
|
+
**version:** must be an incrementing number (e.g. `001` or a timestamp).
|
|
252
|
+
|
|
253
|
+
**action:** should be `do` or `undo`. `do` implements the version, `undo`
|
|
254
|
+
reverts it. Think about it like `up` and `down` in other migration tools.
|
|
255
|
+
|
|
256
|
+
**optional-description** describes which changes migration makes. Although
|
|
257
|
+
optional, it should be used for all migrations as it makes it easier for
|
|
258
|
+
everyone to know which changes are made in a migration.
|
|
259
|
+
|
|
260
|
+
In our example, we are going to have a single migration that creates a `users`
|
|
261
|
+
table and we are going to use `Postgrator` to run the migration.
|
|
262
|
+
|
|
263
|
+
> Run `npm i pg postgrator` to install dependencies needed for the
|
|
264
|
+
> example.
|
|
265
|
+
|
|
266
|
+
```sql
|
|
267
|
+
// 001.do.create-users-table.sql
|
|
268
|
+
CREATE TABLE IF NOT EXISTS users (
|
|
269
|
+
id SERIAL PRIMARY KEY NOT NULL,
|
|
270
|
+
created_at DATE NOT NULL DEFAULT CURRENT_DATE,
|
|
271
|
+
firstName TEXT NOT NULL,
|
|
272
|
+
lastName TEXT NOT NULL
|
|
273
|
+
);
|
|
274
|
+
```
|
|
275
|
+
```javascript
|
|
276
|
+
const pg = require('pg')
|
|
277
|
+
const Postgrator = require('postgrator')
|
|
278
|
+
const path = require('node:path')
|
|
279
|
+
|
|
280
|
+
async function migrate() {
|
|
281
|
+
const client = new pg.Client({
|
|
282
|
+
host: 'localhost',
|
|
283
|
+
port: 5432,
|
|
284
|
+
database: 'example',
|
|
285
|
+
user: 'example',
|
|
286
|
+
password: 'example',
|
|
287
|
+
});
|
|
288
|
+
|
|
289
|
+
try {
|
|
290
|
+
await client.connect();
|
|
291
|
+
|
|
292
|
+
const postgrator = new Postgrator({
|
|
293
|
+
migrationPattern: path.join(__dirname, '/migrations/*'),
|
|
294
|
+
driver: 'pg',
|
|
295
|
+
database: 'example',
|
|
296
|
+
schemaTable: 'migrations',
|
|
297
|
+
currentSchema: 'public', // Postgres and MS SQL Server only
|
|
298
|
+
execQuery: (query) => client.query(query),
|
|
299
|
+
});
|
|
300
|
+
|
|
301
|
+
const result = await postgrator.migrate()
|
|
302
|
+
|
|
303
|
+
if (result.length === 0) {
|
|
304
|
+
console.log(
|
|
305
|
+
'No migrations run for schema "public". Already at the latest one.'
|
|
306
|
+
)
|
|
307
|
+
}
|
|
308
|
+
|
|
309
|
+
console.log('Migration done.')
|
|
310
|
+
|
|
311
|
+
process.exitCode = 0
|
|
312
|
+
} catch(err) {
|
|
313
|
+
console.error(err)
|
|
314
|
+
process.exitCode = 1
|
|
315
|
+
}
|
|
316
|
+
|
|
317
|
+
await client.end()
|
|
318
|
+
}
|
|
319
|
+
|
|
320
|
+
migrate()
|
|
321
|
+
```
|