Fork me on GitHub

Symbiot

Understand drivers


This section gives you more details about drivers and their underlying components.

Drivers

A driver describes all the possible interactions between Symbiot and a given thing. It is essentially a composition of input definitions and output definitions.

An input definition is a component of a driver describing the data flow from the thing to Symbiot. A driver may have zero, one or more input definitions, depending on the thing it is integrating with. For instance, a driver for a weather station may define one which fetches all the metrics of the station every 10 minutes and another one which raises an alert when the temperature climbs over a given threshold. A driver for your air conditioning system on the other hand might not define any input definition because it does not emit any data.

An output definitions is a component of a driver describing the data flow from Symbiot to the thing. A driver may have zero, one or more output definitions, depending on the thing it is integrating with. For instance, a driver for your air conditioning system may define one to set the expected temperature. A Facebook driver may provide a few for posting, poking, liking, sharing or anything showing your awesomeness to the world.

In the simplest (and rather common) cases, writing a driver does not require you to write any code: all you need is to write a Json descriptor. For more advanced use cases you may need to write a bit of code, but you do that outside of Symbiot in the programming language of your choice (we’ll get to that later). A descriptor looks like this:

{
  "id" : "my-unique-driver-id",
  "name" : "My Driver",
  "description" : "An example of driver",
  "version" : "0.1.0",
  "inputDefinitions": [ {...}, {...}, {...} ],
  "outputDefinitions": [ {...}, {...}, {...} ]
}

Name it the way you want (with a .json extension) and place it in a drivers folder next to your Symbiot Jar file.

Input definitions

There are different ways to send data to Symbiot. Depending on the communication capabilities of the thing the driver is written for or the programming language you want to use for the driver implementation, you will privilege one or the other integration model.

API Push

Json payloads will be pushed from the thing (or an intermediate process) to the Symbiot API on /inputs/{inputId}/data. This assumes that the original source of events (ex: a toaster) has configurable Webhooks or is under the control of the driver developer (ex: the developer has developed a toaster driver in Node.js which interacts somehow with the toaster and posts Json payloads to the Symbiot API).

This is how to define an API Push input definition in Json:

{
  "type": "apiPushInputDefinition",
  "id" : "my-unique-id-within-the-driver",
  "name": "My input definition",
  "description": "This input accepts incoming data on the Symbiot API",
  "schema": {
    "$schema": "http://json-schema.org/draft-04/schema#",
    "type": "object",
    "properties": {
      "someString" : {
        "type": "string"
      }
    },
    "required": ["someString"]
  }
}

The schema node is a Json Schema describing the format of the data which will be posted on the API endpoint. It is very important since the very core of Symbiot relies on this data format to help you build streams from an input to an output. That’s where you have a bit of job to do: you have to formalize the format of your incoming data (or at least the part that is of interest for you) in a Json Schema. The above example describes a Json payload containing a single string node called “someString” (an example of valid payload would be { "someString" : "hello world" }).

Later on, when you will instantiate inputs for this input definition in the Symbiot UI, you will receive the inputId which determines the endpoint where payloads need to be posted (/inputs/{inputId}).

API Pull

Symbiot does a regular polling on an external API to fetch valid Json payloads. This assumes that the source of events exposes a RESTful API (which works nicely with the Symbiot poller). Ex: the toaster exposes an API to fetch its status; Symbiot will query GET http://192.168.0.14/toaster_api/status every 5 seconds and expects to receive a valid Json payload.

This is how to define an API Pull input definition in Json:

{
  "type": "apiPullInputDefinition",
  "id" : "my-unique-id-within-the-driver",
  "name": "My input definition",
  "description": "This input is pulling data from the thing (or an intermediate process)",
  "frequency": 5,
  "url": "http://192.168.0.14/toaster_api/status",
  "method": "GET",
  "headers": {
    "Authorization": "Basic eW91QXJlQ3VyaW91czpnb29kRm9yWW91"
  },
  "schema": {
    "$schema": "http://json-schema.org/draft-04/schema#",
    "type": "object",
    "properties": {
      "someString" : {
        "type": "string"
      }
    },
    "required": ["someString"]
  }
}

Just as for the API Push input definition, the schema node is a Json Schema describing the format of the data which will be pulled. It’s your job to specify the format of the data your are pulling from the endpoint configured (or at least the part that is of interest for you).

Later on, when you will instantiate inputs for this input definition in the Symbiot UI, you will have the possibility to override any of the frequency, url, method and headers fields, which define where to fetch the data from. Just consider the values defined in your Json descriptor as convenient defaults.

Filesystem watcher

Symbiot watches a folder in the filesystem where valid Json payloads can be dropped (or modified). This assumes that the source of events does not talk HTTP and uses the filesystem to exchange data (note that this method works nicely with FTP or a system like DropBox). Ex: a program written in C collects information from a USB toaster, converts it to Json in memory and writes it under /symbiot/toasterInputs (and Symbiot watches this folder).

This is how to define a Filesystem input definition in Json:

{
  "type": "filesystemInputDefinition",
  "id" : "my-unique-id-within-the-driver",
  "name": "My input definition",
  "description": "This is watching changes in a local folder",
  "folder": "/symbiot/toasterInputs",
  "regex": "*.json",
  "deleteAfterRead": true,
  "schema": {
    "$schema": "http://json-schema.org/draft-04/schema#",
    "type": "object",
    "properties": {
      "someString" : {
        "type": "string"
      }
    },
    "required": ["someString"]
  }
}

The deleteAfterRead parameter specifies if the files must be consumed (deleted upon reading) or left in the folder (and monitored for changes). And again, the schema node is a Json Schema describing the format of the data which will be pulled. It’s your job to specify the format of the data your are matching with your folder and regex (or at least the part that is of interest for you).

Later on, when you will instantiate inputs for this input definition in the Symbiot UI, you will have the possibility to override any of the folder, regex and deleteAfterRead fields. Just consider the values defined in your Json descriptor as convenient defaults.

Output definitions

There are different ways to let Symbiot send data to a target thing. As for input definitions, you will choose one or the other integration model depending on the communication capabilities of the thing your driver is written for, or depending on the programming language you use for the possible intermediate system between Symbiot and the thing. As you will see, these are very similar to the input definitions presented above.

Webhook

Json payloads will be pushed from Symbiot to the thing (or an intermediate process) over HTTP to the provided endpoint. This assumes that the target (ex: a toaster) exposes a RESTful API or is under the control of the driver developer (ex: the developer has written a toaster driver in Node.js which exposes an API and somehow transmits the data to the toaster).

This is how to define a Webhook output definition in Json:

{
  "type": "webhookOutputDefinition",
  "id" : "my-unique-id-within-the-driver",
  "name": "My output definition",
  "description": "This output sends the data to an API endpoint",
  "url": "http://192.168.0.14/toaster_api/heatValue",
  "method": "POST",
  "headers": {
    "Authorization": "Basic eW91QXJlQ3VyaW91czpnb29kRm9yWW91"
  },
  "schema": {
    "$schema": "http://json-schema.org/draft-04/schema#",
    "type": "object",
    "properties": {
      "someString" : {
        "type": "string"
      }
    },
    "required": ["someString"]
  }
}

Again, the schema node is a Json Schema describing the format of the data which will be pulled. It’s your job to specify the format of the data expected by the endpoint where the data will be pushed.

Later on, when you will instantiate outputs for this output definition in the Symbiot UI, you will have the possibility to override any of the url, method and header fields. Just consider the values defined in your Json descriptor as convenient defaults.

Polling

Json payloads will be temporarily stored on Symbiot and exposed on /outputs/{outputId}/data (reachable with a GET method). This assumes that the target cannot be reached with a Webhook (which would have been a more elegant option) and will instead actively poll the outputs from the above endpoint (ex: the developer has written a Cron job which checks for new outputs every 3 seconds before relaying them to the toaster to control the heat level).

This is how to define a Polling output definition in Json:

{
  "type": "pollingOutputDefinition",
  "id" : "my-unique-id-within-the-driver",
  "name": "My output definition",
  "description": "This output temporarily exposes the data as a REST resource",
  "ttl": 30,
  "schema": {
    "$schema": "http://json-schema.org/draft-04/schema#",
    "type": "object",
    "properties": {
      "someString" : {
        "type": "string"
      }
    },
    "required": ["someString"]
  }
}

The ttl node allows to specify for how long (in seconds) the resource shall be retained. The schema node is a Json Schema describing the format of the resource. Once again, it’s your job to specify this format.

Later on, when you will instantiate outputs for this output definition in the Symbiot UI, you will receive the {outputId} used in the resource path (which determines where your polling component needs to query the data).

Filesystem writer

Json payloads will be saved to disk in a given folder (and named after the current timestamp, with .json extension). Well, at this stage you should get the point: if your thing or your programming language is not quite HTTP-friendly or for whatever reason you want to use the filesystem as an integration point (ex: for testing during your driver implementation), then that’s how you can do it.

Here is how to define a Filesystem output definition in Json:

{
  "type": "filesystemOutputDefinition",
  "id" : "my-unique-id-within-the-driver",
  "name": "My output definition",
  "description": "This output stores my Json payload on disk",
  "folder": "/symbiot/toasterOutputs",
  "schema": {
    "$schema": "http://json-schema.org/draft-04/schema#",
    "type": "object",
    "properties": {
      "someString" : {
        "type": "string"
      }
    },
    "required": ["someString"]
  }
}

The schema node is a Json Schema describing the format of the resulting Json payload. Once again, it’s your job to specify this format. So do it and do it well.

Later on, when you will instantiate outputs for this output definition in the Symbiot UI, you will have the possibility to override the folder parameter, if needed.

Summary

To write a driver, you first have to identify what are the inputs and outputs, and describe their data format with a Json Schema. For each input definition you have to choose one of the three integration models (push, pull or filesystem). For each output you have to choose one of the three integration models (webhook, polling or filesystem). Put all this in a nice Json file, name it something .json and place it in the driversfolder next to the Symbiot Jar file. Symbiot will automatically load your driver at startup (or complain if something’s wrong).