事件发射器 (Event Emitter)
AdonisJS 有一个内置的事件发射器,它是基于 emittery 构建的。Emittery 异步分发事件,并 修复了 Node.js 默认事件发射器的许多常见问题。
AdonisJS 进一步增强了 emittery 的功能:
- 通过定义事件列表及其相关数据类型,提供静态类型安全。
- 支持基于类的事件和监听器。将监听器移动到专用文件可以保持代码库整洁且易于测试。
- 能够在测试期间模拟事件。
基本用法
事件监听器定义在 start/events.ts 文件中。你可以使用 make:preload ace 命令创建此文件。
node ace make:preload events
你必须使用 emitter.on 来监听事件。该方法接受事件名称作为第一个参数,监听器作为第二个参数。
import emitter from '@adonisjs/core/services/emitter'
emitter.on('user:registered', function (user) {
console.log(user)
})
一旦定义了事件监听器,就可以使用 emitter.emit 方法触发 user:registered 事件。它接受事件名称作为第一个参数,事件数据作为第二个参数。
import emitter from '@adonisjs/core/services/emitter'
export default class UsersController {
async store() {
const user = await User.create(data)
emitter.emit('user:registered', user)
}
}
你可以使用 emitter.once 来一次性监听事件。
emitter.once('user:registered', function (user) {
console.log(user)
})
使事件类型安全
AdonisJS 强制要求为你想要在应用程序中触发的每个事件定义静态类型。这些类型注册在 types/events.ts 文件中。
在下面的示例中,我们将 User 模型注册为 user:registered 事件的数据类型。
如果你觉得为每个事件定义类型很繁琐,你可以切换到 基于类的事件。
import User from '#models/User'
declare module '@adonisjs/core/types' {
interface EventsList {
'user:registered': User
}
}
基于类的监听器
像 HTTP 控制器一样,监听器类提供了一个抽象层,可以将内联事件监听器移动到专用文件中。监听器类存储在 app/listeners 目录中,你可以使用 make:listener 命令创建一个新的监听器。
另请参阅:Make listener 命令
node ace make:listener sendVerificationEmail
监听器文件是用 class 声明和 handle 方法生成的。在这个类中,你可以定义其他方法来监听多个事件(如果需要)。
import User from '#models/user'
export default class SendVerificationEmail {
handle(user: User) {
// 发送邮件
}
}
最后一步,你必须在 start/events.ts 文件中将监听器类绑定到事件。你可以使用 #listeners 别名导入监听器。别名是使用 Node.js 的子路径导入功能 定义的。
import emitter from '@adonisjs/core/services/emitter'
import SendVerificationEmail from '#listeners/send_verification_email'
emitter.on('user:registered', [SendVerificationEmail, 'handle'])
懒加载监听器
建议懒加载监听器以加快应用程序启动阶段。懒加载就像将导入语句移到函数后面并使用动态导入一样简单。
import emitter from '@adonisjs/core/services/emitter'
import SendVerificationEmail from '#listeners/send_verification_email'
const SendVerificationEmail = () => import('#listeners/send_verification_email')
emitter.on('user:registered', [SendVerificationEmail, 'handle'])
依赖注入
你不能在监听器类中注入 HttpContext。因为事件是异步处理的,监听器可能会在 HTTP 请求完成后运行。
监听器类使用 IoC 容器 实例化;因此,你可以在类构造函数或处理事件的方法中对依赖项进行类型提示。
在下面的示例中,我们将 TokensService 类型提示为构造函数参数。调用此监听器时,IoC 容器将注入 TokensService 类的实例。
import { inject } from '@adonisjs/core'
import TokensService from '#services/tokens_service'
@inject()
export default class SendVerificationEmail {
constructor(protected tokensService: TokensService) {}
handle(user: User) {
const token = this.tokensService.generate(user.email)
}
}
在下面的示例中,我们在 handle 方法中注入 TokensService。但是,请记住,第一个参数始终是事件负载。
import { inject } from '@adonisjs/core'
import TokensService from '#services/tokens_service'
import UserRegistered from '#events/user_registered'
export default class SendVerificationEmail {
@inject()
handle(event: UserRegistered, tokensService: TokensService) {
const token = tokensService.generate(event.user.email)
}
}
基于类的事件
事件是事件标识符(通常是基于字符串的事件名称)和相关数据的组合。例如:
user:registered是事件标识符(即事件名称)。- User 模型的实例是事件数据。
基于类的事件将事件标识符和事件数据封装在同一个类中。类构造函数充当标识符,类的实例保存事件数据。
你可以使用 make:event 命令创建一个事件类。
另请参阅:Make event 命令
node ace make:event UserRegistered
事件类没有行为。相反,它是事件的数据容器。你可以通过类构造函数接受事件数据,并将其作为实例属性提供。
import { BaseEvent } from '@adonisjs/core/events'
import User from '#models/user'
export default class UserRegistered extends BaseEvent {
constructor(public user: User) {
super()
}
}
监听基于类的事件
你可以使用 emitter.on 方法为基于类的事件附加监听器。第一个参数是事件类引用,后跟监听器。
import emitter from '@adonisjs/core/services/emitter'
import UserRegistered from '#events/user_registered'
emitter.on(UserRegistered, function (event) {
console.log(event.user)
})
在下面的示例中,我们要同时使用基于类的事件和监听器。
import emitter from '@adonisjs/core/services/emitter'
import UserRegistered from '#events/user_registered'
const SendVerificationEmail = () => import('#listeners/send_verification_email')
emitter.on(UserRegistered, [SendVerificationEmail])
触发基于类的事件
你可以使用 static dispatch 方法触发基于类的事件。dispatch 方法接受事件类构造函数接受的参数。
import User from '#models/user'
import UserRegistered from '#events/user_registered'
export default class UsersController {
async store() {
const user = await User.create(data)
/**
* 分发/触发事件
*/
UserRegistered.dispatch(user)
}
}
简化事件监听体验
如果你决定使用基于类的事件和监听器,你可以使用 emitter.listen 方法来简化绑定监听器的过程。
emitter.listen 方法接受事件类作为第一个参数,并将基于类的监听器数组作为第二个参数。此外,注册的监听器必须具有处理事件的 handle 方法。
import emitter from '@adonisjs/core/services/emitter'
import UserRegistered from '#events/user_registered'
emitter.listen(UserRegistered, [
() => import('#listeners/send_verification_email'),
() => import('#listeners/register_with_payment_provider'),
() => import('#listeners/provision_account')
])
处理错误
默认情况下,监听器引发的异常将导致 unhandledRejection。因此,建议使用 emitter.onError 方法自行捕获和处理错误。
emitter.onError 方法接受一个回调,该回调接收事件名称、错误和事件数据。
import emitter from '@adonisjs/core/services/emitter'
emitter.onError((event, error, eventData) => {
})
监听所有事件
你可以使用 emitter.onAny 方法监听所有事件。该方法接受监听器回调作为唯一参数。
import emitter from '@adonisjs/core/services/emitter'
emitter.onAny((name, event) => {
console.log(name)
console.log(event)
})
取消订阅事件
emitter.on 方法返回一个取消订阅函数,你可以调用该函数来删除事件监听器订阅。
import emitter from '@adonisjs/core/services/emitter'
const unsubscribe = emitter.on('user:registered', () => {})
// 删除监听器
unsubscribe()
你还可以使用 emitter.off 方法删除事件监听器订阅。该方法接受事件名称/类作为第一个参数,并将对监听器的引用作为第二个参数。
import emitter from '@adonisjs/core/services/emitter'
function sendEmail () {}
// 监听事件
emitter.on('user:registered', sendEmail)
// 删除监听器
emitter.off('user:registered', sendEmail)
emitter.offAny
emitter.offAny 删除监听所有事件的通配符监听器。
emitter.offAny(callback)
emitter.clearListeners
emitter.clearListeners 方法删除给定事件的所有监听器。
// 基于字符串的事件
emitter.clearListeners('user:registered')
// 基于类的事件
emitter.clearListeners(UserRegistered)
emitter.clearAllListeners
emitter.clearAllListeners 方法删除所有事件的所有监听器。
emitter.clearAllListeners()
可用事件列表
请查看 事件参考指南 以查看可用事件列表。
测试中的模拟事件
事件监听器通常用于在给定操作后执行副作用。例如:向新注册的用户发送电子邮件,或在订单状态更新后发送通知。
编写测试时,你可能希望避免向测试期间创建的用户发送电子邮件。
你可以通过模拟 (faking) 事件发射器来禁用事件监听器。在下面的示例中,我们在发出 HTTP 请求注册用户之前调用 emitter.fake。请求之后,我们使用 events.assertEmitted 方法确保 SignupController 触发了特定事件。
import emitter from '@adonisjs/core/services/emitter'
import UserRegistered from '#events/user_registered'
test.group('User signup', () => {
test('create a user account', async ({ client, cleanup }) => {
const events = emitter.fake()
cleanup(() => {
emitter.restore()
})
await client
.post('signup')
.form({
email: 'foo@bar.com',
password: 'secret',
})
})
// 断言事件已触发
events.assertEmitted(UserRegistered)
})
event.fake方法返回 EventBuffer 类的实例,你可以使用它进行断言和查找触发的事件。emitter.restore方法恢复 fake。恢复 fake 后,事件将正常触发。
模拟特定事件
如果你在没有任何参数的情况下调用 emitter.fake 方法,它将模拟所有事件。如果你想模拟特定事件,请将事件名称或类作为第一个参数传递。
// 仅模拟 user:registered 事件
emitter.fake('user:registered')
// 模拟多个事件
emitter.fake([UserRegistered, OrderUpdated])
多次调用 emitter.fake 方法将删除旧的 fakes。
事件断言
你可以使用 assertEmitted、assertNotEmitted、assertNoneEmitted 和 assertEmittedCount 方法为模拟事件编写断言。
assertEmitted 和 assertNotEmitted 方法接受事件名称或类构造函数作为第一个参数,以及一个可选的查找器函数(必须返回布尔值以将事件标记为已触发)。
const events = emitter.fake()
events.assertEmitted('user:registered')
events.assertNotEmitted(OrderUpdated)
events.assertEmitted(OrderUpdated, ({ data }) => {
/**
* 仅当 orderId 匹配时
* 才认为事件已触发
*/
return data.order.id === orderId
})
// 断言特定事件的计数
events.assertEmittedCount(OrderUpdated, 1)
// 断言没有触发任何事件
events.assertNoneEmitted()