Skip to content

Lightweight Node.js based wrapper for the SAP HANA Cloud Platform IoT Services API.

License

Notifications You must be signed in to change notification settings

NoelHendrikx/hcp-iot-api-node

 
 

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

35 Commits
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

HANA Cloud Platform IoT Services API (hcp-iot-api)

This is a lightweight Node.js based wrapper for the SAP HANA Cloud Platform IoT Services API. Instead of wrangling with authentication and request configuration, you can use the libraries methods to communicate with the API. Both services (Remote Device Management Servcie and Message Management Service) have been implemented in their own class (see documentation below).

There is also a utility class available to request OAuth tokens for configured HCP clients.

Installation

You can install the module through the public npm registry by running the following command in CLI:

npm install --save hcp-iot-api

Promises

The library is using promises for a cleaner syntax and to avoid nested callbacks. So when there is the need to have things be in sequence, you would normally have to write nested callbacks in JavaScript:

// This is how the API would look without promises
rdms.createDeviceType({...} , function (deviceType) {
	rdms.createMessageType({...}, function(messageType) {		
		rdms.registerDevice({...}, function(device) {
			...
		}, function(error) { console.log(error); });
	}, function(error) { console.log(error); });
}, function(error) { console.log(error); });

Because the libraries methods return Promises, it is much easer to deal with sychronous requests, avoid handling errors multiple times and keep everything more readable. So the API actually looks like this:

rdms.createDeviceType({...})
	 .then(function(deviceType){
		 return rdms.createMessageType({...});
	 })
	 .then(function(messageType){
		 return rdms.registerDevice({...});
	 }).
	 .then(function(device){
		 ...
	 }).
	 catch(function(error) { console.log(error) });

Every time you want something to run in sync, just put the code in the then function and return the promise. More on promises can be found here.

Remote Device Management Service (RDMS)

The IoT Services Cockit inside HCP contains a user interface to manage devices, device types, message types, etc. The Remote Device Management Service can be used to programatically do the administration without the need to access the cockpit. Especially in big projects it may be a good solution to store the definition of message types, etc. as code fragements in one central place with versioning support (e.g. git).

Setup

The RDMS always needs the users HCP username and password for authentication. This information needs to be given when initializing a new object.

var API = require("hcp-iot-api");
var rdms = new API.RemoteDeviceManagementService({
	"account": "<user>", 
	"password": "<password>"
});

Reading and posting data

The API allows to completly configure the HCP IoT services without the need to access a GUI. Instead there are fuctions available to read, create and delete all kinds of entity types. A simple example showing how to fetch all message types:

rdms.getMessageTypes()
	.then(function(messageTypes) {
		// Do something meaningful with the returned array of objects
	})
.catch(function(error) { console.log(error.message) });

And another one showing how to create a device type and after that, create a corresponding message type. The available fields can be read in the official documentation.

rdms.createDeviceType({ "name": "Device Type 1" })
.then(function (deviceType) {
	return rdms.createMessageType({'device_type': deviceType.id, ...});	// Create the message type
.then(function (messageType) {
	...		// Maybe do something else
.catch(function(error) { console.log(error.message) });

Registering devices

It is possible to dynamically register devices via API. A key requirement is the former definition of a device type, because its id and token need to be applied. A possible call could look like:

rdms.registerDevice({
	"name": "Device 1",
	"device_type": deviceType.id,
		"attributes": [
			{ "key": "customKey", "value": "custom value" }
		]
	}, deviceType.token);
})
.then(function(device) {
	var deviceToken = device.token;   // Needs to be stored!
	var deviceId = device.id;
	...
});  

When registering a device, the API returns the devices JSON object containing a token. This is only returned once directly after creation, so it is important to store it for later usage. When using the MMS to send sensor data from a device, this token needs to be passed in as the OAuth token. If you do not store the token programatically, it can be reset in the IoT Cockpit.

Message Management Service (MMS)

The Message Management Service is the actual service to send data from an IoT device into the Hana Cloud Platform and/or back to the device.

Setup

The MMS is a bit more complex in terms of authorization. There are several methods, which are bound to a specific device. For exampe, if you want to send sensor data from a device into the HCP (mms.sendData(...)), you need the deviceId and deviceToken information. Other methods (e.g. mms.pushToDevice(...)) require a user authentication via HTTP Basic Auth (account/password) or OAuth (oauthToken). The authentication information could be assigned in the constructor or when calling a method (see full API below). Passing all parameters in upfront would look like this:

var API = require("hcp-iot-api");
var mms = new API.MessageManagementService({
	"account": "<username>",		// This one is mandatory!
	"password": "<password>",
	"deviceToken": "<deviceToken>",
	"deviceId": "<deviceId>",
	"oauthToken": "<oauthToken>"
});

Send sensor data via HTTP(S)

The main purpose of the MMS is to send sensor data from the device into the HCP. The API is straight forward when using HTTP(S) as the message protocol:

mms.sendData({
	"messageType": "<messageTypeId>",
	"messages": [{
		"sensor1": "Value 1",
		"sensor2": "Value 2"
	}]
})
.catch(function(error) { console.log(error.message) });

If the deviceId and deviceToken have not been included in the constructor, they can be passed to the sendData method as the second and third parameter:

var mms = new API.MessageManagementService({ "account": "<username>" });
mms.sendData({
	"messageType": "<messageTypeId>",
	"messages": [{
		"sensor1": "Value 1",
		"sensor2": "Value 2"
	}]
}, "<deviceId>", "<deviceToken>")
.catch(function(error) { console.log(error.message) });

Sending information back to a device

Using MMS it is also possible, to send information to a device. When sending the information, it ca be specified if the message should be delivered via HTTP or a Websocket connection. It is also required to register a message type with direction ToDevice and use this one, when sending the information. Authentication for using the pushToDevice function is either via password or via oauthToken.

var mms = new API.MessageManagementService({
  "account": "<username>", 
  "password": "<password>"
});

mms.pushToDevice("<deviceId>", {
  "method": "http", // or "ws"
  "sender": "My IoT application", 
  "messageType": "<messageTypeId>",
  "messages":[
    {
      "abc": "switch on", 
    } 
  ]  
});

If HTTP as the transport protocol, the message is being stored inside the HCP in table T_IOT_HTTP_PUSH and can be retrieved by polling via the mms.getData() method. Again, if specificed in the constructor, the method parameters can be omitted.

var mms = new API.MessageManagementService({
  "account": "<username>", 
  "password": "<password>"
});

mms.getData("<deviceId>", "<deviceToken>")
.then(function(messages) {
	// Fetch the existing messages for this device
});

Using websockets

The MMS also allows a persistent connection to the API via websocket protocol. This should be used if a bidirectional communication is required. When using websockets, messages send to a device do not need to be pulled, but are pushed through the websocket connection.

The following example shows how to establish a websocket connection and once this is established, send data into the HCP. The sendData method detects the open connection and will send the data over the connection instead of over HTTP(S). The onWebsocketMessage function registers a callback, which is being called whenever a message for the device arrives. Please note, that we are not using promises here, because this method may be called multiple times.

var mms = new API.MessageManagementService({
	"account": "<account>", 
	"deviceId": "<deviceId>", 
	"deviceToken": "<deviceToken>"
});

mms.openWebsocketConnection()
.then(function() {
	
	// Register callback for when data is being pushed to the device
	mms.onWebsocketMessage(function(message) {
		// Do something the message
	});
	
  	// Send data through the websocket
	mms.sendData({
		"messageType": "<messageType>",
		"messages": [{
        		"sensor1": "Value 1",
        		"sensor2": "Value 2"
      		}]
  	});  
});


// Close, when not required anymore
mms.closeWebsocketConnection()
.then(function() {
	// Maybe do something when closed
});;

MMS configuration via API

Since version 2.15.0 of the MMS, it is possible to set a wide range of options for configuration. This can be done either via GUI in the MMS Cockpit or via API. The following example shows, how you can tell the API to create all tables using the column instead of the row store. Authentication must be done either with password or oauthToken.

var mms = new API.MessageManagementService({
	"account": "<account>", 
	"password": "<password>"
});

mms.updateConfig({
	"config": [
		{ "key": "mms.processing.sql.hana.table_type", "value": "column" }
	]
}).catch(function(error) { console.log(error.message) });

Be aware, that in the official API documentation the configuration format is currently incorrect. So if you want to set settings programmatically, you have to explicitly use key and value like in the example shown above. For a list of all existing options, there is a mms.getConfig() method available, wich returns the whole configuration.



Full documentation (will be updated soon)

The following section describes the full API.

Class: RemoteDeviceManagementService

new API.RemoteDeviceManagementService([options])

  • options Object
    • account String
    • password String

Construct a new rdms object. account and password need to be set for HTTP authentication.

Example:

var API = require("hcp-iot-api");
var rdms = new API.RemoteDeviceManagementService({
	"account": "<user>", 
	"password": "<password>"
});

rdms.createDeviceType(options)

  • options Object – For a list of options check out the official docs.

Creates a new device type.

Example:

rdms.createDeviceType({ "name": "Device Type 1" })
  .then(function (deviceType) {
  ...
  })
  .catch(function(error) { console.log(error.message) });

rdms.getDeviceTypes()

Returns all existing device types.

Example:

rdms.getDeviceTypes()
  .then(function (deviceTypes) {
    ...
  })
  .catch(function(error) { console.log(error.message) });

rdms.getDeviceType(id)

  • id String

Returns one specific device type.

rdms.deleteDeviceType(id)

  • id String

Deletes the device type.


rdms.createMessageType(options)

  • options Object – For a list of options check out the official docs.

Creates a new message type. A message type is always bound to a specific device type and needs the direction set to either fromDevice or toDevice.

Example:

rdms.createMessageType({
   "device_type": deviceType.id, 
  "name": "Message Type 1",
    "direction": "fromDevice",
    "fields": [
       {
          "position": 1, 
          "name": "sensor1",
          "type": "string"
       },
       {
          "position": 2,
          "name": "sensor2",
          "type": "string"
       }
    ]
  }); 
  .catch(function(error) { console.log(error.message) });

rdms.getMessageTypes()

Returns all existing message types.

rdms.getMessageType(id)

  • id String

Returns one specific message type.

rdms.deleteMessageType(id)

  • id String

Deletes the message type.


rdms.registerDevice(options, deviceTypeToken)

  • options Object – For a list of options check out the official docs.
  • deviceTypeToken String – The token from the device type

Registers a new device. A device is always bound to a specific device type. A device can be created using the registerDevice or createDevice method. The latter one does not return the OAuth-token required to send data to the API via MMS. The token itself can be read and recreated via the IoT Cockpit.

Example:

rdms.registerDevice({
  "name": "Device 1",
  "device_type": device_type.id,
  "attributes": [
    { "key": "customKey1", "value": "custom value" },
    { "key": "customKey2", "value": "custom value" }
  ]}, deviceTypeToken); 
.catch(function(error) { console.log(error.message) });

rdms.createDevice(options)

  • options Object – For a list of options check out the official docs.

Creates a new device. To use the device for sending data, you have to read the token from the IoT Cockpit.

Example:

rdms.createDevice({
  "name": "Device 1",
  "device_type": device_type.id,
  "attributes": [
    { "key": "customKey1", "value": "custom value" },
    { "key": "customKey2", "value": "custom value" }
  ]
}); 
.catch(function(error) { console.log(error.message) });

rdms.getDevices()

Returns all existing deviced.

rdms.getDevice(id)

  • id String

Returns one specific device.

rdms.deleteDevice(id)

  • id String

Deletes the device.

rdms.getDeviceAttributes(id)

  • id String

Returns all attributes for a device.

rdms.createDeviceAttribute(id, options)

  • id String – The id of the device
  • options Object – For all options please check out the official docs

Creates a specific attribute for a device.

rdms.deleteDeviceAttribute(id, key)

  • id String
  • key String

Deletes a specific attribute of a device.


rdms.getSettings()

Returns the RDMS settings. For more information please check out the official docs


Class: MessageManagementService

new API.MessageManagementService([options])

  • options Object
    • account String (mandatory)
    • password String (optional)
    • deviceId String (optional)
    • deviceToken String (optional)

Construct a new mms object. account and password need to be set for HTTP authentication.

Example:

var API = require("hcp-iot-api");
var rdms = new API.MessageManagementService({
	"account": "<user>", 
	"deviceId": "<deviceId>",
	"deviceToken": "<deviceToken>"
});

mms.sendData(options, deviceId, deviceToken)

  • options Object – For a list of options check out the official docs.
  • deviceId String (optional) – Has to be given if not set on global level
  • deviceToken String (optional) – Has to be given if not set on global level

Send sensor data to the IoT Cockpit from a specific device.

Example:

mms.sendData({
  "mode": "async",
  "messageType": "<messageTypeId>",
  "messages": [
    {
      "sensor1": "Value 1",
      "sensor2": "Value 2"
    }
  ]
})
.catch(function(error) { console.log(error.message) });

About

Lightweight Node.js based wrapper for the SAP HANA Cloud Platform IoT Services API.

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages

  • JavaScript 100.0%