AdonisJS v7 已完成功能开发,正在进行最终验证。 了解更多

事件发射器

事件发射器 (Event Emitter)

AdonisJS 有一个内置的事件发射器,它是基于 emittery 构建的。Emittery 异步分发事件,并 修复了 Node.js 默认事件发射器的许多常见问题

AdonisJS 进一步增强了 emittery 的功能:

  • 通过定义事件列表及其相关数据类型,提供静态类型安全。
  • 支持基于类的事件和监听器。将监听器移动到专用文件可以保持代码库整洁且易于测试。
  • 能够在测试期间模拟事件。

基本用法

事件监听器定义在 start/events.ts 文件中。你可以使用 make:preload ace 命令创建此文件。

node ace make:preload events

你必须使用 emitter.on 来监听事件。该方法接受事件名称作为第一个参数,监听器作为第二个参数。

start/events.ts
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 的子路径导入功能 定义的。

start/events.ts
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

事件类没有行为。相反,它是事件的数据容器。你可以通过类构造函数接受事件数据,并将其作为实例属性提供。

app/events/user_registered.ts
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。

事件断言

你可以使用 assertEmittedassertNotEmittedassertNoneEmittedassertEmittedCount 方法为模拟事件编写断言。

assertEmittedassertNotEmitted 方法接受事件名称或类构造函数作为第一个参数,以及一个可选的查找器函数(必须返回布尔值以将事件标记为已触发)。

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()