Skip to content

alephzero/alephzero

Repository files navigation



AlephZero

Simple, Robust, Fast IPC.

OverviewTransportProtocolExamplesInstallationAcross Dockers

Overview

Presentation from March 25, 2020

AlephZero is a library for message based communication between programs running on the same machine.

Simple

AlephZero's main goal is to be simple to use. Nothing is higher priority.

There is no "master" process in between your nodes that is needed to do handshakes or exchanges of any kind. All you need is the topic name.

See the Examples.

Robust

This is probably the main value of AlephZero, above similar libraries.

AlephZero uses a lot of tricks to ensure the state of all channels is consistent, even when programs die. This includes double-buffering the state of the communication channel and robustifying the locks and notification channels.

Fast

AlephZero uses shared memory across multiple processes to read and write messages, minimizing the involvement of the kernel. The kernel only really gets involved in notifying a process that a new message exists, and for that we use futex (fast user-space mutex).

TODO: Benchmarks

Transport

AlephZero, at its core, is a simple allocator on top of a contiguous region of memory. Usually, shared-memory. The allocator of choice is a circular-linked-list, which is fast, simple, and sufficient for the protocol listed below. It also plays well with the robustness requirement.

This has a number of implications. For one, this means that old messages are kept around until the space is needed. The oldest messages are always discarded before any more recent messages.

Protocol

Rather than exposing the low-level transport directly, AlephZero provides a few higher level protocol:

  • PubSub: Broadcast published messages. Subscribers get notified.
  • RPC: Request-response.
  • PRPC (Progressive RPC): Request-streaming response.

Examples

Many more example and an interactive experience can be found at: https://github.com/alephzero/playground

For the curious, here are some simple snippets to get you started:

To begin with, we need to include AlephZero:

#include <a0.h>

PubSub

You can have as many publisher and subscribers on the same topic as you wish. They just need to agree on the filename.

a0::Publisher p("my_pubsub_topic");
p.pub("foo");

You just published "foo" to the "my_pubsub_topic".

To read those message, you can create a subscriber on the same topic:

a0::Subscriber sub(
    "my_pubsub_topic",
    [](a0::Packet pkt) {
      std::cout << "Got: " << pkt.payload() << std::endl;
    });

The callback will trigger whenever a message is published.

The Subscriber object spawns a thread that will read the topic and call the callback.

To avoid thread creation and manually probe for messages:

a0::SubscriberSync sub_sync("my_pubsub_topic");
while (sub_sync.can_read()) {
  auto pkt = sub_sync.read();
  std::cout << "Got: " << pkt.payload() << std::endl;
}

An optional INIT can be added to specify where the subscriber starts reading.

  • INIT_AWAIT_NEW (default): Start with messages published after the creation of the subscriber.
  • INIT_MOST_RECENT: Start with the most recently published message. Useful for state and configuration. But be careful, this can be quite old!
  • INIT_OLDEST: Topics keep a history of 16MB (unless configures otherwise). Start with the oldest thing still in there.

An optional ITER can be added to specify how to continue reading messages. After each callback:

  • ITER_NEXT (default): grab the sequentially next message. When you don't want to miss a thing.
  • ITER_NEWEST: grab the newest available unread message. When you want to keep up with the firehose.

RPC

Create an RpcServer:

a0::RpcServer server(
    "my_rpc_topic",
    /* onrequest = */ [](a0::RpcRequest req) {
        std::cout << "Got: " << req.pkt().payload() << std::endl;
        req.reply("echo " + std::string(req.pkt().payload()));
    },
    /* oncancel = */ nullptr);

Create an RpcClient:

a0::RpcClient client("my_rpc_topic");
client.send("client msg", [](a0::Packet reply) {
  std::cout << "Got: " << reply.payload() << std::endl;
});

Installation

Install From Source

Ubuntu Dependencies

apt install g++ make

Alpine Dependencies

apk add g++ linux-headers make

Download And Install

git clone https://github.com/alephzero/alephzero.git
cd alephzero
make install -j

Install From Package

Coming soon-ish. Let me know if you want this and I'll prioritize it. External support is much appreciated.

Integration

Command Line

Add the following to g++ / clang commands.

-L${libdir} -lalephzero -lpthread

Package-cfg

pkg-config --cflags --libs alephzero

CMake

Coming soon-ish. Let me know if you want this and I'll prioritize it. External support is much appreciated.

Bazel

Coming soon-ish. Let me know if you want this and I'll prioritize it.

Across Dockers

For programs running across different dockers to be able to communicate, we need to have them match up on two flags: --ipc and --pid.

  • --ipc shares the /dev/shm filesystem. This is necessary to open the same file topics.
  • --pid shares the process id namespace. This is necessary for the locking and notification systems.

In the simplest case, you can set them both to host and talk through the system's global /dev/shm and process id namespace.

docker run --ipc=host --pid=host --name=foo foo_image
docker run --ipc=host --pid=host --name=bar bar_image

Or, you can mark one as shareable and have the others connect to it:

docker run --ipc=shareable --pid=shareable --name=foo foo_image
docker run --ipc=container:foo --pid=container:foo --name=bar bar_image