AMEBO
Amebo is the simplest pubsub server to use, deploy, understand or maintain. It was
built to enable communication between applications i.e. microservices or
modules (if you are using monoliths) collectively called producers, and hopes to be able to serve as a small
but capable and simpler alternative to Kafka, RabbitMQ, SQS/SNS.
-
Availability: Amebo runs on battle tested open source tools i.e. Redis, Postgres and provides the same level of availability guarantees
-
Reliability: Amebo has been used at scale by open source projects to handle 100's of millions of request
-
Latency: Amebo guarantees sub 100ms latencies at scale (barring network and hardware limitations)
How It Works
Amebo has only 4 concepts (first class objects) to understand or master.
1. Producers
These can be microservices or modules (in a monolith) - that create and receive notifications about events. All applications must be registered on
amebo ;-) before they can publish events.
2. Action
This is something that can happen in an application i.e. creating a customer, deleting an order. They are registered
on Amebo by their parent application, and all actions must provide a valid JSON Schema (can be empty "{}") that Amebo
can use to validate action events before sending to subscribers.
3. Events
An event is the occurence of an action and in practice is a HTTP request sent by an application to Amebo to signal it about the occurence of an action locally. Events can have a json payload that must match the JSON Schema of its parent action.
4. Subscribers
These are HTTP URLs (can be valid hostname for loopback interface on TCP/IP as well) endpoints registered by applications to
receive action payloads.
GETTING STARTED
This assumes you have installed Amebo on your machine. Amebo requires Python3.6+
pip install amebo
amebo --workers 2 --address 0.0.0.0:8701
git clone https://github.com/tersoo/amebo
mv amebo /to/a/directory/of/your/choosing
export $PATH=$PATH:/to/a/directory/of/your/choosing/amebo
ambeo -w 2 -a 0.0.0.0:8701
1st : Tell Amebo about all your event producers
endpoint: /v1/producers
Schema | Example Payload |
|
---|
{
"$schema": "",
"type": "object",
"properties": {
"microservice": {"type": "string"},
"passphrase": {"type": "string"},
"location": {"type": "web", "format": "ipv4 | ipv6 | hostname | idn-hostname"}
},
"required": ["microservice", "passphrase", "location"]
}
|
{
"microservice": "customers",
"passphrase": "some-super-duper-secret-of-the-module-or-microservice",
"location": "http://0.0.0.0:3300"
}
|
2nd : Register actions that can happen in the registered producers
endpoint: /v1/actions
Endpoint JSON Schema | Example Payload |
|
---|
{
"type": "object",
"properties": {
"action": {"type": "string"},
"microservice": {"type": "string"},
"schemata": {
"type": "object",
"properties": {
"type": {"type": "string"},
"properties": {"type": "object"},
"required": {"type": "array"}
}
}
},
"required": ["action", "microservice", "schemata"]
}
|
{
"action": "customers.v1.created",
"microservice": "customers",
"schemata": {
"$id": "https://your-domain/customers/customer-created-schema-example.json",
"$schema": "https://json-schema.org/draft/2020-12/schema",
"type": "object",
"properties": {
"customer_id": {"type": "number"},
"first_name": {"type": "string"},
"last_name": {"type": "string"},
"email": {"type": "string", "format": "email"}
},
"required": ["customer_id", "email"]
}
}
|
3rd : Tell Amebo when an action occurs i.e. create an event
endpoint: /v1/events
Key | Description |
---|
action | Identifier name of the action. (As registered in the previous step.) |
deduper | Deduplication string. used to prevent the same event from being registered twice |
payload | JSON data (must confirm to the schema registerd with the action) |
Endpoint JSON Schema | Example Payload |
|
---|
{
"type": "object",
"properties": {
"event": {"type": "string"},
"microservice": {"type": "string"},
"schemata": {
"type": "string",
"format": "ipv4 | ipv6 | hostname | idn-hostname"
},
"location": {"type": "web"}
},
"required": ["microservice", "passphrase", "location"]
}
|
{
"event": "customers.v1.created",
"microservice": "customers",
"schema": {
"$id": "https://your-domain/customers/customer-created-schema-example.json",
"$schema": "https://json-schema.org/draft/2020-12/schema",
"type": "object",
"properties": {
"event": {"type": "string"},
"microservice": {"type": "string"},
"schemata": {"type": "string", "format": "ipv4 | ipv6 | hostname | idn-hostname"},
"location": {"type": "web"}
},
"required": ["microservice", "passphrase", "location"]
},
"location": "http://0.0.0.0:3300"
}
|
Finally: Create an endpoint to receive action notifications from Amebo
Other applications/modules within the same monolith can create handler endpoints that will be sent the payload with optional
encryption if an encryption key was provided by the subscriber when registering for the event.
Why? Advantages over traditional Message Oriented Middleware(s)
-
Amebo comes complete with a Schema Registry, ensuring actions conform to event schema, and makes it easy for developers to search for events by
microservice with commensurate schema (i.e. what is required, what is optional) as opposed to meetings with team mates continually.
-
GUI for tracking events, actions, subscribers. Easy discovery of what events exist, what events failed and GUI retry for specific subscribers
-
Gossiping is HTTP native i.e. subscribers receive http requests automatically at pre-registered endpoints
-
Envelope format and transmission is web native and clearly outlined by schema registry
-
Topic management is simplified as actions with versioning support baked in
-
Infinite retries (stop after $MAX_RETRIES and $MAX_MINUTES coming soon)
Trivia
The word amebo
is a West African (Nigerian origin - but used in Ghana, Benin, Cameroon etc.) slang used to describe anyone that never keeps what you tell them to themselves. A talkative, never mind their business individual (a chronic gossip).