Event Emitter
AdonisJS has an inbuilt event emitter created on top of emittery. Emittery dispatches events asynchronously and fixes many common issues with the Node.js default Event emitter.
AdonisJS further enhances emittery with additional features.
- Provide static type safety by defining a list of events and their associated data types.
- Support for class-based events and listeners. Moving listeners to dedicated files keep your codebase tidy and easy to test.
- Ability to fake events during tests.
Basic usage
The event listeners are defined inside the start/events.ts
file. You may create this file using the make:preload
ace command.
node ace make:preload events
You must use the emitter.on
to listen to an event. The method accepts the event's name as the first argument and the listener as the second argument.
import emitter from '@adonisjs/core/services/emitter'
emitter.on('user:registered', function (user) {
console.log(user)
})
Once you have defined the event listener, you can emit the user:registered
event using the emitter.emit
method. It takes the event name as the first argument and the event data as the second argument.
import emitter from '@adonisjs/core/services/emitter'
export default class UsersController {
async store() {
const user = await User.create(data)
emitter.emit('user:registered', user)
}
}
You may use the emitter.once
to listen to an event once.
emitter.once('user:registered', function (user) {
console.log(user)
})
Making events type-safe
AdonisJS makes it mandatory to define static types for every event you want to emit within your application. The types are registered within the types/events.ts
file.
In the following example, we register the User
model as the data type for the user:registered
event.
If you find defining types for every event cumbersome, you may switch to class-based events.
import User from '#models/User'
declare module '@adonisjs/core/types' {
interface EventsList {
'user:registered': User
}
}
Class-based listeners
Like HTTP controllers, listener classes offer an abstraction layer to move inline event listeners inside dedicated files. Listener classes are stored inside the app/listeners
directory and you may create a new listener using the make:listener
command.
See also: Make listener command
node ace make:listener sendVerificationEmail
The listener file is scaffolded with the class
declaration and handle
method. In this class, you can define additional methods to listen to multiple events (if needed).
import User from '#models/user'
export default class SendVerificationEmail {
handle(user: User) {
// Send email
}
}
As the final step, you must bind the listener class to an event within the start/events.ts
file. You may import the listener using the #listeners
alias. The aliases are defined using the subpath imports feature of Node.js.
import emitter from '@adonisjs/core/services/emitter'
import SendVerificationEmail from '#listeners/send_verification_email'
emitter.on('user:registered', [SendVerificationEmail, 'handle'])
Lazy-loading listeners
It is recommended to lazy load listeners to speed up the application boot phase. Lazy loading is as simple as moving the import statement behind a function and using dynamic imports.
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'])
Dependency injection
You cannot inject the HttpContext
inside a listener class. Because events are processed asynchronously, the listener might run after the HTTP request is finished.
The listener classes are instantiated using the IoC container; therefore, you can type-hint dependencies inside the class constructor or the method which handles the event.
In the following example, we type-hint the TokensService
as a constructor argument. When invoking this listener, the IoC container will inject an instance of the TokensService
class.
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)
}
}
In the following example, we inject the TokensService
inside the handle method. However, do remember, the first argument will always be the event payload.
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)
}
}
Class-based events
An event is a combination of the event identifier (usually a string-based event name) and the associated data. For example:
user:registered
is the event identifier (aka the event name).- An instance of the User model is the event data.
Class-based events encapsulate the event identifier and the event data within the same class. The class constructor serves as the identifier, and an instance of the class holds the event data.
You may create an event class using the make:event
command.
See also: Make event command
node ace make:event UserRegistered
The event class has no behavior. Instead, it is a data container for the event. You may accept event data via the class constructor and make it available as an instance property.
import { BaseEvent } from '@adonisjs/core/events'
import User from '#models/user'
export default class UserRegistered extends BaseEvent {
constructor(public user: User) {
super()
}
}
Listening to class-based events
You may attach listeners for class-based events using the emitter.on
method. The first argument is the event class reference, followed by the listener.
import emitter from '@adonisjs/core/services/emitter'
import UserRegistered from '#events/user_registered'
emitter.on(UserRegistered, function (event) {
console.log(event.user)
})
In the following example, we use both class-based events and listeners.
import emitter from '@adonisjs/core/services/emitter'
import UserRegistered from '#events/user_registered'
const SendVerificationEmail = () => import('#listeners/send_verification_email')
emitter.on(UserRegistered, [SendVerificationEmail])
Emitting class-based events
You may emit a class-based event using the static dispatch
method. The dispatch
method takes arguments accepted by the event class constructor.
import User from '#models/user'
import UserRegistered from '#events/user_registered'
export default class UsersController {
async store() {
const user = await User.create(data)
/**
* Dispatch/emit the event
*/
UserRegistered.dispatch(user)
}
}
Simplifying events listening experience
If you decide to use class-based events and listeners, you may use the emitter.listen
method to simplify the process of binding listeners.
The emitter.listen
method accepts the event class as the first argument and an array of class-based listeners as the second argument. In addition, the registered listener must have a handle
method to handle the event.
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')
])
Handling errors
By default, the exceptions raised by the listeners will result in unhandledRejection. Therefore, it is recommended to self capture and handle the error using the emitter.onError
method.
The emitter.onError
method accepts a callback which receives the event name, error, and event data.
import emitter from '@adonisjs/core/services/emitter'
emitter.onError((event, error, eventData) => {
})
Listening to all events
You can listen to all the events using the emitter.onAny
method. The method accepts the listener callback as the only parameter.
import emitter from '@adonisjs/core/services/emitter'
emitter.onAny((name, event) => {
console.log(name)
console.log(event)
})
Unsubscribing from events
The emitter.on
method returns an unsubscribe function you may call to remove the event listener subscription.
import emitter from '@adonisjs/core/services/emitter'
const unsubscribe = emitter.on('user:registered', () => {})
// Remove the listener
unsubscribe()
You may also use the emitter.off
method to remove the event listener subscription. The method accepts the event name/class as the first argument and a reference to the listener as the second argument.
import emitter from '@adonisjs/core/services/emitter'
function sendEmail () {}
// Listen for event
emitter.on('user:registered', sendEmail)
// Remove listener
emitter.off('user:registered', sendEmail)
emitter.offAny
The emitter.offAny
removes a wildcard listener, listening for all the events.
emitter.offAny(callback)
emitter.clearListeners
The emitter.clearListeners
method removes all the listeners for a given event.
//String-based event
emitter.clearListeners('user:registered')
//Class-based event
emitter.clearListeners(UserRegistered)
emitter.clearAllListeners
The emitter.clearAllListeners
method removes all the listeners for all the events.
emitter.clearAllListeners()
List of available events
Please check the events reference guide to view the list of available events.
Faking events during tests
Event listeners are often used for performing side effects after a given action. For example: Send an email to a newly registered user, or send a notification after an order status update.
When writing tests, you might want to avoid sending emails to the user created during the test.
You may disable event listeners by faking the event emitter. In the following example, we call emitter.fake
before making an HTTP request to signup a user. After the request, we use the events.assertEmitted
method to ensure that the SignupController
emits a specific event.
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',
})
})
// Assert the event was emitted
events.assertEmitted(UserRegistered)
})
- The
event.fake
method returns an instance of the EventBuffer class, and you may use it for assertions and finding emitted events. - The
emitter.restore
method restores the fake. After restoring the fake, the events will be emitted normally.
Faking specific events
The emitter.fake
method fakes all the events if you invoke the method without any arguments. If you want to fake a specific event, pass the event name or the class as the first argument.
// Fakes only the user:registered event
emitter.fake('user:registered')
// Fake multiple events
emitter.fake([UserRegistered, OrderUpdated])
Calling the emitter.fake
method multiple times will remove the old fakes.
Events assertions
You may use assertEmitted
, assertNotEmitted
, assertNoneEmitted
and the assertEmittedCount
methods to write assertions for faked events.
The assertEmitted
and assertNotEmitted
methods accepts either the event name or the class constructor as the first argument and an optional finder function that must return a boolean to mark the event as emitted.
const events = emitter.fake()
events.assertEmitted('user:registered')
events.assertNotEmitted(OrderUpdated)
events.assertEmitted(OrderUpdated, ({ data }) => {
/**
* Only consider the event as emitted, if
* the orderId matches
*/
return data.order.id === orderId
})
// Assert count of a specific event
events.assertEmittedCount(OrderUpdated, 1)
// Assert no events were emitted
events.assertNoneEmitted()