Mailbox is an NPM module built on top of the XState machine, by adding a message queue to the XState machine and letting the machine decide when to process the next message.
Mailboxes are one of the fundamental parts of the actor model. Through the mailbox mechanism, actors can decouple the reception of a message from its elaboration.
- An actor is an object that carries out its actions in response to communications it receives.
- A mailbox is nothing more than the data structure that holds messages.
if you send 3 messages to the same actor, it will just execute one at a time.
— The actor model in 10 minutes - Actors have mailboxes
The design of Mailbox is very like the the Akka Actor Model:
- Build on top of the powerful Finite State Machine (FSM) with XState library.
- Implemented Mailbox pattern for Actor Model: process one message at a time, with a message queue
- Address has been abstracted by a
Address
class for actors - Unit tests covered
The mailbox is separate (which itself can be an actor) and the timing of events forwarded to the statechart can be n-time whereas transitions within the s/c are zero-time. (tweet)
— @DavidKPiano, 🚀 Creator of XState
... statecharts process events immediately while actors (by means of message queues) give you granular control of when to process the next event. (tweet)
— @chrisshank23, core member of StateML
- [ ] Support distribution actor by adding remote actor abstraction to the
Address
class
I'm building assistant chatbot for Wechaty community, and I want to use actor model based on XState to implement it.
My actor will receive message from Wechaty, and send message to Wechaty.
However, ... (describe the async & multi-user scanerio for the conversation turns)
It turns out ... (describe that we need to make sure the incoming messages are queued when we not finished processing the last one)
Thread-safe code only manipulates shared data structures in a manner that ensures that all threads behave properly and fulfill their design specifications without unintended interaction.
A naive state machine is a mailbox actor with capacity=0
, which means it will face the Dead Letter
problem when new messages come but it has not finished processing the last one.
Assume we are a coffee maker, and we need 4 three steps to make a coffee:
- received a
MAKE_ME_COFFEE
event from a customer (sync) - get a cup (async: cost some time)
- fill coffee into the cup (async: cost some time)
- send a
COFFEE
event to the customer (sync)
The coffee maker can only make one cup of coffee at a time, which means that we can not process the MAKE_ME_COFFEE
event until we have finished the last one.
The state machine of the coffee maker is:
Here's the source code of coffee maker:
const machine = createMachine({
context: {
customer: null,
},
initial: states.idle,
states: {
[states.idle]: {
entry: Mailbox.actions.idle('coffee-maker'),
on: {
[types.MAKE_ME_COFFEE]: {
target: states.making,
actions: actions.assign((_, e) => ({ customer: e.customer })),
},
'*': states.idle,
},
},
[states.making]: {
after: {
10: states.delivering,
},
},
[states.delivering]: {
entry: Mailbox.actions.reply(ctx => Events.COFFEE(ctx.customer || 'NO CUSTOMER')),
after: {
10: states.idle,
},
exit: actions.assign({ customer: _ => null }),
},
},
})
If there's a new customer come in, and he/she want coffee, we can get a cup then fill coffee to the cup then deliver a cup of coffee to our customer. Everything is fine so far so good.
However, when there are two customer coming in together, and they talk to us at the same time and each customer want a cup of coffee. After we received the first request(event/message), we are starting to get cup and can not listen to another request anymore, which will result an event (the second one) lost (a Dead Letter).
An actor should read the messages to process from its mailbox. A mailbox is an event proxy that holds messages and deals with the backpressure. When the actor have finished processing the current event, it will receive(pull) the next event from the mailbox.
Mailbox for rescue.
Mailbox is a NPM module written in TypeScript based on the XState finite state machine to strict follow the actor model's principle:
const mailbox = Mailbox.from(machine)
Then use mailbox
instead.
Image credit: Goeth Rubber Duck, @Lanco, https://ducksinthewindow.com/goeth-rubber-duck/
SVG image generated by https://www.visioncortex.org/
Learn more about similiar (i.e. Akka) Actor & Mailbox diagram with discussion from this Issue: statelyai/xstate#2870
import * as Mailbox from 'mailbox'
- Add
Mailbox.actions.idle('child-id')
to theentry
of state of your machine which it accepting new messages, to let the Mailbox continue sending new messages from other actors. - Use
Mailbox.actions.reply('YOUR_EVENT')
to reply event messages to other actors. - Use
const mailbox = Mailbox.from(yourMachine)
to wrap your actor with mailbox address. The mailbox address is a parent XState machine which will invok your machine as child and add message queue to the child machine.
import * as Mailbox from 'mailbox'
import { createMachine } from 'xstate'
const machine = createMachine({
initial: 'idle',
states: {
idle: {
/**
* RULE #1: machine must has `Mailbox.Actions.idle('child-id')`
*/
entry: Mailbox.actions.idle('child-machine-name'),
on: {
'*': {
/**
* RULE #2: machine must use an external transision to the `idle` state when it finished processing any messages, to trigger the `entry` action.
*/
target: 'idle',
actions: actions.log('make sure the idle state will be re-entry with external trainsition when receiving event'),
},
BUSY: 'busy',
},
},
busy: {
/**
* RULE #3: machine use `Mailbox.Actions.reply(EVENT)` to reply EVENT to other actors.
*/
entry: Mailbox.actions.reply('YOUR_EVENT'),
after: {
10: 'idle',
},
},
},
})
const mailbox = Mailbox.from(yourMachine)
// just use it as a standard XState machine
You can run a full version at examples/mailbox-demo.ts and see the result:
$ ./mailbox-demo.ts
# testing raw machine ...
sending TASK
TASK_RECEIVED
sending TASK
# testing raw machine ... done
# testing mailbox-ed machine ...
sending TASK
TASK_RECEIVED
sending TASK
TASK_RECEIVED
# testing mailbox-ed machine ... done
The Duckula
is like Duck for Mailbox Actors.
Image credit: Papercraft Count Duckula
The specification has rules that a Mailbox Actor module:
- MUST export a
id
of typestring
- MUST export a
machine
of type XState machine - MUST export a
initialContext
of typefunction
, with theContext
typing, which is the initial context of the machine - MUST export a
Event
of type map offunction
which is event creators (must use typesafe-actions) - MUST export a
Type
of type map ofstring
which is event types, values in the formnpm-module-or-app/EVENT_TYPE
- MUST export a
State
of type map ofstring
which is states, values in the formnpm-module-or-app/StateName
- MUST be
UPPER_SNAKE_CASE
for the keys ofEvent
andType
- MUST be
UpperCamelCase
for the keys ofState
Similiar ideas: Duckula for Clojure
interface Duckula <...> {
id: TID
Type: TType
Event: TEvent
State: TState
machine: TMachine
initialContext: () => TContext
}
Read the source code at src/duckula/duckula.ts
import * as Mailbox from 'mailbox'
import * as states from './states.js'
// `events.js` must use `typesafe-actions`
import * as events from './events.js'
interface Context {}
const duckula = Mailbox.duckularize({
id: 'MyMachineActor',
initialContext: {} as Context,
events: [ events, [ 'EVENT1', 'EVENT2' ] ], // or: `events: events` if you do not need filtering
states: [ states, [ 'State1', 'State2' ] ], // or: `states: states` if you do not need filtering
})
// `duckula` is a `Duckula` now.
- Ding Dong Machine: <tests/machine-behaviors/ding-dong-machine.ts>
- Coffee Maker Machine: <tests/machine-behaviors/coffee-maker-machine.ts>
- Baby Machine: <tests/machine-behaviors/baby-machine.ts>
[![Mailbox.Duckula Specification](https://img.shields.io/badge/Specification-Mailbox.Duckula-blueviolet)](https://github.com/huan/mailbox#duckula-specification)
Read detail (auto-generated) docs at https://paka.dev/npm/mailbox
import * as Mailbox from 'mailbox'
const mailbox = Mailbox.from(machine, options)
Options:
interface Options {
id? : string
capacity? : number
logger? : InterpreterOptions['logger'],
devTools? : InterpreterOptions['devTools'],
}
The mailbox actor will queue the second inbound messages to the child machine, and will not pass it to the child machine until the first inbound message is processed.
However, this are cases that the child machine that needs to communicate with other actors, and receives response messages from other actors.
For outbound messages, the machine internally can send messages to other actors, and receives outbound response messages from other actors without the inbound limitation (the response of the outbound message will be passed by the mailbox queue directly).
The machine internal address will be used to send messages to other actors, and receive messages with this address will by pass the Mailbox queue, for supporting multiple outbound message communication.
sequenceDiagram
participant Consumer
participant Mailbox
participant Machine
participant Service
Consumer->>Mailbox: {type: EVENT1}
Note right of Consumer: Inbound Message Request 1
Consumer-->>Mailbox: {type: EVENT2}
Note right of Consumer: Inbound Message Request 2
Note right of Mailbox: Processing EVENT1<br>Queue EVENT2
Mailbox->>Machine: {type: EVENT1}
Machine->>Service: {type: LOG_COMMAND}
Machine->>Service: {type: DB_QUERY}
Note right of Machine: Multiple Outbound Message Requests
Service->>Machine: {type: LOG_COMMAND_RESPONSE}
Service->>Machine: {type: DB_QUERY_RESPONSE}
Note right of Machine: Multiple Outbound Message Responses
Machine->>Mailbox: {type: EVENT1_RESPONSE}
Mailbox->>Consumer: {type: EVENT1_RESPONSE}
Note right of Consumer: Inbound Message Response 1
Note right of Mailbox: Dequeue EVENT2<br>Processing EVENT2
Mailbox-->>Machine: {type: EVENT2}
Caution: be aware of the dead lock if your have two actors call each other in the same machine.
Actors have mailboxes.
In the actor model, we must follow that: "if you send 3 messages to the same actor, it will just execute one at a time."
It’s important to understand that, although multiple actors can run at the same time, an actor will process a given message sequentially. This means that if you send 3 messages to the same actor, it will just execute one at a time. To have these 3 messages being executed concurrently, you need to create 3 actors and send one message each.
Messages are sent asynchronously to an actor, that needs to store them somewhere while it’s processing another message. The mailbox is the place where these messages are stored.
an actor is started it will keep running, processing messages from its inbox and won’t stop unless you stop it. It will maintain its state throughout and only the actor has access to this state. This is unlike your traditional asynchronous programming, such as using Future in Scala or promises in javascript where once the call has finished its state between calls is not maintained and the call has finished.
Once an actor receives a message, it adds the message to its mailbox. You can think of the mailbox as queue where the actor picks up the next message it needs to process. Actors process one message at a time. The actor patterns specification does say that the order of messages received is not guaranteed, but different implementations of the actor pattern do offer choices on mailbox type which do offer options on how messages received are prioritized for processing.
- The
Mailbox.actions.idle('machine-name')('reason')
action must be put inside theentry
action of when it's ready to receive message (instates.idle
for example) - All events that received in
states.idle
must make aexternal
trancition by adding atarget
entry, so that thestates.idle
state will be entered again, which will emit theMailbox.actions.idle('machine-name')
to parent (Mailbox) to let the Mailbox know the machine is ready to receive the next message.
Learn more from validate.ts source code
Whenever a message fails to be written into an actor mailbox, the actor system redirects it to a synthetic actor called /deadLetters. The delivery guarantees of dead letter messages are the same as any other message in the system. So, it’s better not to trust so much in such messages. The main purpose of dead letters is debugging.
mailbox.onEvent(event => {
if (event.type === Mailbox.types.DEAD_LETTER) {
console.error('DEAD_LETTER:', event.payload)
}
})
Related reading:
The mailbox is unbounded by default, which means it doesn’t reject any delivered message. Besides, there’s no back pressure mechanism implemented in the actor system. Hence, if the number of incoming messages is far bigger than the actor’s execution pace, the system will quickly run out of memory.
As we said, unbounded mailboxes grow indefinitely, consuming all the available memory if the messages’ producers are far quicker than the consumers. Hence, we use this kind of mailbox only for trivial use cases.
On the other side, bounded mailboxes retain only a fixed number of messages. The actor system will discard all of the messages arriving at the actor when the mailbox is full. This way, we can avoid running out of memory.
As we did a moment ago, we can configure the mailbox’s size directly using the Mailbox.bounded factory method. Or, better, we can specify it through the configuration properties file:
const mailbox = Mailboxe.from(machine, {
capacity: 100,
})
The example above is a clear example where bounded mailboxes shine. We are not afraid of losing some messages if the counterpart maintains the system up and running.
A new question should arise: Where do the discarded messages go? Are they just thrown away? Fortunately, the actor system lets us retrieve information about discarded messages through the mechanism of dead letters — we’ll soon learn more about how this works.
Credit: https://www.baeldung.com/scala/typed-mailboxes#1-bounded-vs-unbounded
Tell, Don’t Ask!
It’s often said that we must “tell an actor and not ask something“. The reason for this is that the tell pattern represents the fully asynchronous way for two actors to communicate.
The tell pattern is entirely asynchronous. After the message is sent, it’s impossible to know if the message was received or if the process succeeded or failed.
To use the Tell Pattern, an actor must retrieve an actor reference to the actor it wants to send the message to.
Request-Response
The Ask Pattern allows us to implement the interactions that need to associate a request to precisely one response. So, it’s different from the more straightforward adapted response pattern because we can now associate a response with its request.
The Mailbox implementes the Ask Pattern by default. It will response to the original actor sender when there's any response events from the child actor.
- "The difference being that messages are directed, events are not — a message has a clear addressable recipient while an event just happen for others (0-N) to observe it." (link)
- "The difference lies in that with MessageQueues it's typical that the sender requires a response. With an EventQueue this is not necessary." (link)
- "A Message is some data sent to a specific address; An Event is some data emitted from a component for anyone listening to consume." (link)
Image source: Wechaty CQRS
Never use interpreter.send([...eventList])
to send multiple events. It will cause the mailbox to behavior not right (only the first event will be delivered).
Use eventList.forEach(e => interpreter.send(e))
to send event list.
The reason is that internally Mailbox
have three parallel states and they will run into race condition in batch-event mode.
See: XState Docs - Batched Events
- Universal Server - My favorite Erlang Program, @Joe Arm Strong, Nov 21, 2013
- The Reactive Manifesto
- Apache Commons - Five Minutes SCXML
- SCXML State Chart XML, Kai Blankenhorn, April 2003
- State machine replication
Redux is predictable states container, XState is predictable transitions container. — A Youtuber comment
Mailbox is predictable states & transitions container for actors. — Huan, creator of Wechaty, Jan. 2022
- Mailbox Interface is an Observable now.
- Fix the race condition bug by simplifing the queue state management to be atomic. (Issue #5)
- Fix wrapped machine ID conflic bug when there have nested Mailbox wrapped machines (PR #8)
- Add
Duckula
Interface for modulize Mailbox Actor. (Issue #1) - Renaming
CHILD
toACTOR
:-
CHILD_REPLY
->ACTOR_REPLY
-
CHILD_IDLE
->ACTOR_IDLE
-
- Duck types/states clean
Refactoring APIs
- Supports
Mailbox.actions.proxy(name)(target)
to proxy all events to a target (string
id,Address
, orMailbox
) - Supports
Mailbox.actions.send(target)(event, options)
where the target can be astring
id,Address
, orMailbox
for convenience.
Publish mailbox NPM module with DevOps.
- Implement the Actor State/Context Persisting Mechanism.
- Add Dead Letter Queue (DLQ)
capacity
options for dealing with the Back Pressure.
- XStates Docs - Persisting state
- nact(node.js + actors) Docs - Persist
- XState Persistence with actors and side effects
Improving the Actor Mailbox model.
Related links:
- XState Actors
- Typed Mailboxes in Scala, @Riccardo Cardin, Feb 23, 2021
- Proto.Actor - Mailboxes
- [v5] Optimize mailbox statelyai/xstate#2870
Initial version.
Related issue discussions:
- Track the Modeling Events of XState wechaty/bot5-assistant#2
- Actor model needs a Mailbox wrapper/concurrency-manager wechaty/bot5-assistant#4
- Discord Actor Model discussion
- Kotlin Concurrency with Actors, Jag Saund, Jun 14, 2018
- xstate-onentry-timing-bug.js
- Automating Spaceships with XState, @stafford Williams, Mar 22, 2021,
Great thanks to @alxhotel who owned the great NPM module name mailbox and kindly transfered it to me for this new project, thank you very much Alex!
Hi Huan, nice to meet you :) Glad to see a serial entrepreneur contributing to OSS Red heart I've added you as a maintainer! Feel free to remove me once you make sure you have control of the package. Looking forward to see what you are building :)
Cheers,
Alex (Dec 20 Mon 11:47 PM)
Huan LI is a serial entrepreneur, active angel investor with strong technology background. Huan is a widely recognized technical leader on conversational AI and open source cloud architectures. He co-authored guide books "Chatbot 0 to 1" and "Concise Handbook of TensorFlow 2" and has been recognized both by Microsoft and Google as MVP/GDE. Huan is a Chatbot Architect and speaks regularly at technical conferences around the world. Find out more about his work at https://github.com/huan/
- Docs released under Creative Commons
- Code released under the Apache-2.0 License
- Code & Docs © 2021 Huan LI <zixia@zixia.net>