Fastify

插件漫游指南

首先, 不要恐慌!

Fastify 从一开始就搭建成非常模块化的系统. 我们搭建了非常强健的 API 来允许你创建命名空间, 来添加工具方法. Fastify 创建的封装模型可以让你在任何时候将你的应用分割成不同的微服务, 而无需重构整个应用.

内容清单

注册器

就像在 JavaScript 万物都是对象, 在 Fastify 万物都是插件.
你的路由, 你的工具方法等等都是插件. 无论添加什么功能的插件, 你都可以使用 Fastify 优秀又独一无二的 API: register.

  1. fastify.register(
  2. require('./my-plugin'),
  3. { options }
  4. )

register 创建一个新的 Fastify 上下文, 这意味着如果你对 Fastify 的实例做任何改动, 这些改动不会反映到上下文的父级上. 换句话说, 封装!

为什么封装这么重要?
那么, 假设你创建了一个具有开创性的初创公司, 你会怎么做? 你创建了一个包含所有东西的 API 服务器, 所有东西都在同一个地方, 一个庞然大物!
现在, 你增长得非常迅速, 想要改变架构去尝试微服务. 通常这意味着非常多的工作, 因为交叉依赖和缺少关注点的分离.
Fastify 在这个层面上可以帮助你很多, 多亏了封装模型, 它完全避免了交叉依赖, 并且帮助你将组织成高聚合的代码块.

让我们回到如何正确地使用 register.
插件必须输出一个有以下参数的方法

  1. module.exports = function (fastify, options, done) {}

fastify 就是封装的 Fastify 实例, options 就是选项对象, 而 done 是一个在插件准备好了之后必须要调用的方法.

Fastify 的插件模型是完全可重入的和基于图(数据结构)的, 它能够处理任何异步代码并且保证插件的加载顺序, 甚至是关闭顺序! 如何做到的? 很高兴你发问了, 查看下 avvio! Fastify 在 .listen(), .inject() 或者 .ready() 被调用了之后开始加载插件.

在插件里面你可以做任何想要做的事情, 注册路由, 工具方法 (我们马上会看到这个) 和进行嵌套的注册, 只要记住当所有都设置好了后调用 done!

  1. module.exports = function (fastify, options, done) {
  2. fastify.get('/plugin', (request, reply) => {
  3. reply.send({ hello: 'world' })
  4. })
  5. done()
  6. }

那么现在你已经知道了如何使用 register API 并且知道它是怎么工作的, 但我们如何给 Fastify 添加新的功能, 并且分享给其他的开发者?

装饰器

好了, 假设你写了一个非常好的工具方法, 因此你决定在你所有的代码里都能够用这个方法. 你改怎么做? 可能是像以下代码一样:

  1. // your-awesome-utility.js
  2. module.exports = function (a, b) {
  3. return a + b
  4. }
  1. const util = require('./your-awesome-utility')
  2. console.log(util('that is ', 'awesome'))

现在你需要在所有需要这个方法的文件中引入它. (别忘了你可能在测试中也需要它).

Fastify 提供了一个更优雅的方法, 装饰器. 创建一个装饰器非常简单, 只要使用 decorate API:

  1. fastify.decorate('util', (a, b) => a + b)

现在你可以在任意地方通过 fastify.util 调用你的方法, 甚至在你的测试中.
这里神奇的是: 你还记得之前我们讨论的封装? 同时使用 registerdecorate 可以实现, 让我用例子来阐明这个事情:

  1. fastify.register((instance, opts, done) => {
  2. instance.decorate('util', (a, b) => a + b)
  3. console.log(instance.util('that is ', 'awesome'))
  4. done()
  5. })
  6. fastify.register((instance, opts, done) => {
  7. console.log(instance.util('that is ', 'awesome')) // 这里会抛错
  8. done()
  9. })

在第二个注册器中调用 instance.util 会抛错, 因为 util 只存在第一个注册器的上下文中.
让我们更深入地看一下: 当使用 register API 每次都会创建一个新的上下文而且这避免了上文提到的这个状况.

但是注意, 封装只会在父级和同级中有效, 不会在子级中有效.

  1. fastify.register((instance, opts, done) => {
  2. instance.decorate('util', (a, b) => a + b)
  3. console.log(instance.util('that is ', 'awesome'))
  4. fastify.register((instance, opts, done) => {
  5. console.log(instance.util('that is ', 'awesome')) // 这里不会抛错
  6. done()
  7. })
  8. done()
  9. })
  10. fastify.register((instance, opts, done) => {
  11. console.log(instance.util('that is ', 'awesome')) // 这里会抛错
  12. done()
  13. })

PS: 如果你需要全局的工具方法, 请注意要声明在应用根作用域上. 或者你可以使用 fastify-plugin 工具, 参考.

decorate 不是唯一可以用来扩展服务器的功能的 API, 你还可以使用 decorateRequestdecorateReply.

decorateRequestdecorateReply? 为什么我们已经有了 decorate 还需要它们?
好问题, 是为了让开发者更方便地使用 Fastify. 让我们看看这个例子:

  1. fastify.decorate('html', payload => {
  2. return generateHtml(payload)
  3. })
  4. fastify.get('/html', (request, reply) => {
  5. reply
  6. .type('text/html')
  7. .send(fastify.html({ hello: 'world' }))
  8. })

这个可行, 但可以变得更好!

  1. fastify.decorateReply('html', function (payload) {
  2. this.type('text/html') // this 是 'Reply' 对象
  3. this.send(generateHtml(payload))
  4. })
  5. fastify.get('/html', (request, reply) => {
  6. reply.html({ hello: 'world' })
  7. })

你可以对 request 对象做同样的事:

  1. fastify.decorate('getHeader', (req, header) => {
  2. return req.headers[header]
  3. })
  4. fastify.addHook('preHandler', (request, reply, done) => {
  5. request.isHappy = fastify.getHeader(request.raw, 'happy')
  6. done()
  7. })
  8. fastify.get('/happiness', (request, reply) => {
  9. reply.send({ happy: request.isHappy })
  10. })

这个也可行, 但可以变得更好!

  1. fastify.decorateRequest('setHeader', function (header) {
  2. this.isHappy = this.headers[header]
  3. })
  4. fastify.decorateRequest('isHappy', false) // 这会添加到 Request 对象的原型中, 好快!
  5. fastify.addHook('preHandler', (request, reply, done) => {
  6. request.setHeader('happy')
  7. done()
  8. })
  9. fastify.get('/happiness', (request, reply) => {
  10. reply.send({ happy: request.isHappy })
  11. })

我们见识了如何扩展服务器的功能并且如何处理封装系统, 但是假如你需要加一个方法, 每次在服务器 “emits“ 事件的时候执行这个方法, 该怎么做?

钩子方法

你刚刚构建了工具方法, 现在你需要在每个请求的时候都执行这个方法, 你大概会这样做:

  1. fastify.decorate('util', (request, key, value) => { request[key] = value })
  2. fastify.get('/plugin1', (request, reply) => {
  3. fastify.util(request, 'timestamp', new Date())
  4. reply.send(request)
  5. })
  6. fastify.get('/plugin2', (request, reply) => {
  7. fastify.util(request, 'timestamp', new Date())
  8. reply.send(request)
  9. })

我想大家都同意这个代码是很糟的. 代码重复, 可读性差并且不能扩展.

那么你该怎么消除这个问题呢? 是的, 使用钩子方法!

  1. fastify.decorate('util', (request, key, value) => { request[key] = value })
  2. fastify.addHook('preHandler', (request, reply, done) => {
  3. fastify.util(request, 'timestamp', new Date())
  4. done()
  5. })
  6. fastify.get('/plugin1', (request, reply) => {
  7. reply.send(request)
  8. })
  9. fastify.get('/plugin2', (request, reply) => {
  10. reply.send(request)
  11. })

现在每个请求都会运行工具方法, 很显然你可以注册任意多的需要的钩子方法.
有时, 你希望只在一个路由子集中执行钩子方法, 这个怎么做到? 对了, 封装!

  1. fastify.register((instance, opts, done) => {
  2. instance.decorate('util', (request, key, value) => { request[key] = value })
  3. instance.addHook('preHandler', (request, reply, done) => {
  4. instance.util(request, 'timestamp', new Date())
  5. done()
  6. })
  7. instance.get('/plugin1', (request, reply) => {
  8. reply.send(request)
  9. })
  10. done()
  11. })
  12. fastify.get('/plugin2', (request, reply) => {
  13. reply.send(request)
  14. })

现在你的钩子方法只会在第一个路由中运行!

你可能已经注意到, request and reply 不是标准的 Nodejs requestresponse 对象, 而是 Fastify 对象.

如何处理封装与分发

完美, 现在你知道了(几乎)所有的扩展 Fastify 的工具. 但可能你遇到了一个大问题: 如何分发你的代码?

我们推荐将所有代码包裹在一个注册器中分发, 这样你的插件可以支持异步启动 (decorate 是一个同步 API), 例如建立数据库链接.

等等? 你不是告诉我 register 会创建封装的上下文, 那么我创建的不是就外层不可见了?
是的, 我是说过. 但我没告诉你的是, 你可以通过 fastify-plugin 模块告诉 Fastify 不要进行封装.

  1. const fp = require('fastify-plugin')
  2. const dbClient = require('db-client')
  3. function dbPlugin (fastify, opts, done) {
  4. dbClient.connect(opts.url, (err, conn) => {
  5. fastify.decorate('db', conn)
  6. done()
  7. })
  8. }
  9. module.exports = fp(dbPlugin)

你还可以告诉 fastify-plugin 去检查安装的 Fastify 版本, 万一你需要特定的 API.

正如前面所述,Fastify 在 .listen().inject() 以及 .ready() 被调用,也即插件被声明 之后 才开始加载插件。这么一来,即使插件通过 decorate 向外部的 Fastify 实例注入了变量,在调用 .listen().inject().ready() 之前,这些变量是获取不到的。

当你需要在 register 方法的 options 参数里使用另一个插件注入的变量时,你可以向 options 传递一个函数参数,而不是对象:

  1. const fastify = require('fastify')()
  2. const fp = require('fastify-plugin')
  3. const dbClient = require('db-client')
  4. function dbPlugin (fastify, opts, done) {
  5. dbClient.connect(opts.url, (err, conn) => {
  6. fastify.decorate('db', conn)
  7. done()
  8. })
  9. }
  10. fastify.register(fp(dbPlugin), { url: 'https://example.com' })
  11. fastify.register(require('your-plugin'), parent => {
  12. return { connection: parent.db, otherOption: 'foo-bar' }
  13. })

在上面的例子中,register 方法的第二个参数的 parent 变量是注册了插件的外部 Fastify 实例的一份拷贝。这就意味着我们可以获取到之前声明的插件所注入的变量了。

ESM 的支持

Node.js v13.3.0 开始, ESM 也被支持了!写插件时,你只需要将其作为 ESM 模块导出即可!

  1. // plugin.mjs
  2. async function plugin (fastify, opts) {
  3. fastify.get('/', async (req, reply) => {
  4. return { hello: 'world' }
  5. })
  6. }
  7. export default plugin

注意:Fastify 不支持具名导入 ESM 模块,但支持 default 导入。

  1. // server.mjs
  2. import Fastify from 'fastify'
  3. const fastify = Fastify()
  4. ///...
  5. fastify.listen(3000, (err, address) => {
  6. if (err) {
  7. fastify.log.error(err)
  8. process.exit(1)
  9. }
  10. })

错误处理

你的插件也可能在启动的时候失败. 或许你预料到这个并且在这种情况下有特定的处理逻辑. 你该怎么实现呢? after API 就是你需要的. after 注册一个回调, 在注册之后就会调用这个回调, 它可以有三个参数.
回调会基于不同的参数而变化:

  1. 如果没有参数并且有个错误, 这个错误会传递到下一个错误处理.
  2. 如果有一个参数, 这个参数就是错误对象.
  3. 如果有两个参数, 第一个是错误对象, 第二个是完成回调.
  4. 如果有三个参数, 第一个是错误对象, 第二个是顶级上下文(除非你同时指定了服务器和复写, 在这个情况下将会是那个复写的返回), 第三个是完成回调.

让我们看看如何使用它:

  1. fastify
  2. .register(require('./database-connector'))
  3. .after(err => {
  4. if (err) throw err
  5. })

自定义错误

假如你的插件需要暴露自定义的错误,fastify-error 能帮助你轻松地在代码或插件中生成一致的错误对象。

  1. const createError = require('fastify-error')
  2. const CustomError = createError('ERROR_CODE', 'message')
  3. console.log(new CustomError())

发布提醒

假如你要提示用户某个 API 不被推荐,或某个特殊场景需要注意,你可以使用 fastify-warning

  1. const warning = require('fastify-warning')()
  2. warning.create('FastifyDeprecation', 'FST_ERROR_CODE', 'message')
  3. warning.emit('FST_ERROR_CODE')

开始!

太棒了, 现在你已经知道了所有创建插件需要的关于 Fastify 和它的插件系统的知识, 如果你写了插件请告诉我们! 我们会将它加入到 生态 章节中!

如果你想要看看真正的插件例子, 查看:

如果感觉还差什么? 告诉我们! :)