Skip to content

Latest commit

 

History

History
135 lines (104 loc) · 7.26 KB

README.md

File metadata and controls

135 lines (104 loc) · 7.26 KB

MiniBus

travisci

MiniBus is an interprocess communication library for rapid prototyping complex software built using a service-oriented architecture design.
Similar functionality can be found in tools such as ICE, LCM, and ROS. MiniBus has been optimized to have a

  • highly flexible protocol
  • simple client API
  • minimal dependencies
  • easy to embed in existing projects

Minibus is currently only available in Python, however its simple design should allow it to be easily ported into other languages.

Install

$ sudo pip install minibus

Overview

Conceptually, a MiniBus network consists of a group of independent and loosely associated processes called Clients that can communicate with each other by passing messages (even between different machines). Such messages are broadcast over named channels called Topics. When one Client Publishes (sends) a message over a Topic, any other Clients Subscribing (listening) to that Topic will receive the message. This is called a Publish/Subscribe model, in which information must pushed from the sender to the receiver.

MiniBus also defines a remote procedure call (RPC) interface known as Services which allows MiniBus Clients known as Service Clients to call functions running on other MiniBus Clients known as Service Servers. Service Servers advertise their functions by assigning them names which can be used by Service Clients to map local functions to the remote ones. This is called a Request/Reply model, in which information from the server is polled by the client. Services are implemented using Topics.

Topics

When a client publishes a message on MiniBus, it must specify a name for the topic it will be published on. When a different client wishes to receive that same message, it must specify that it wants to listen to the topic of the same name.

Message Schema (Types)

Any data structure which can be serialized into JSON can be sent over Minibus. MiniBus topics do not have any particular message type associated with them. Instead, each client that connects to a topic as a publisher or subscriber can specifies a message schema (or data format) they expect messages to be in. Message schema are in the form of JSON Schema, and can be as generic or specific as you wish. For example, the schema { } will accept all messages while the schema {"type": "string"} accepts only strings. For more information on writing schema, check out json-schema.org.

Topic Names and Namespaces

Topic names are formatted using namespaces. A namespace is similar to a unix file system directory. All topics must begin with a forward slash / followed by some name. These names would make up the root namespace. Adding additional forward slashes / adds a sub namespace.

Namespaces are purely for organizational purposes and do not have any other special function inside MiniBus. However, subscribers can use regular expressions to listen to multiple topics at the same time, and namespaces make it easy to leverage this feature.

Some topic names can have special meanings. For example, names that start and end with a double underscore are not displayed by the mbtt tool (e.g. __hidden_topic__). This convention does not hide topics from other client, rather it is mostly used to mask internal components of the minibus system.

Services

Services use the same naming conventions and message schema concepts as topics, and make use of topic namespaces. A service is specified by a name and two schemas - one defining parameters (arguments) being passed to the service and one defining the return value. Service requests are tied directly to reply messages, so when multiple clients are connected to a single service server only the calling client will receive the corresponding reply message.

Services Over Topics

Services are implemented using MiniBus topics. The service name is used to specify a topic namespace under which three well-known topics are automatically created. One topic (__request__) is used for calling the service and sending parameters to be used for execution. If the service finishes executing normally and returns a value, this is passed back to the calling client on the __reply__ topic. Otherwise, if an exception occurs while executing the service, the error message is returned to the calling client on the __error__ topic. This implementation is abstracted by the client interface.

Python API

There are currently two Python implementations of the MiniBusClientAPI - a threaded socket version and a Twisted version.

class MiniBusClientAPI(object):
    """ Defines the public API for interacting with the minibus """
    def __init__(self, name, cryptokey=None):

    def publisher(self, topic_name, data_format):

    def subscribe(self, name_pattern, data_format, callback, headers=False):

    def unsubscribe(self, name_pattern, callback):

    def service_client(self, name, reqst_schema, reply_schema, reply_cb, err_cb):

    def service_func_client(self, name, reqst_schema, reply_schema):
        """ Returns a function that behaves like a local function.
        retval = proxyfunc(params)
        """

    def service_func_server(self, name, reqst_schema, reply_schema, func):
        """ Provides a named network service linked to a local function.
        my_func(params...)
        Client receives the value returned by the function.
        This is a convenience function that wraps the functionality of service_server()
        around a single function which returns a value to be sent back to the client.
        """

    def service_server(self, name, reqst_schema, reply_schema, func):
        """ Provides a named network service. Service work is received by the
        func parameter, including an srvid value and received parameters.
        The srvid value is used to identify the client's "work order" information
        and must be passed to one of the two reply functions.
        Service returns value to client when a value is passed to either
        service_server_return() or service_server_error().
        """

    def service_server_return(self, srvid, value):
        """ Used by a service server to send a return value to a service client """

    def service_server_error(self, srvid, value):
        """ Used by a service server to send an error value to a service client """

Bus Protocol

The current version of MiniBus sends all messages over Multicast UDP at address 228.0.0.5:8005. Future versions of the protocol are planned to use a combination of Multicast UDP, TCP, and IPC.

Message format

Messages are passed between clients formatted in serialized JSON packets defined by a bus schema. The bus schema defines a packet to contain a header object and a data field. The header is a JSON object consisting of the topic name as a string. The data field contains a string with the message as a serialized JSON object inside it.

Dependencies

These python dependencies are all available in pip, apt, and macports repositories (although they may be under different names)

  • python 2.7
  • jsonschema
  • netifaces
  • Twisted (optional: for Twisted client)
  • gnupg (optional: for encryption)