Skip to content

SiDen/satori-rtm-sdk-c

 
 

Repository files navigation

C SDK for Satori RTM

This C SDK is a lightweight SDK:

  • It has no external dependencies, in particular it does not require a JSON library
  • It does not lock you into using any threading or event loop framework, but is ready to be used in any of those
  • Does not allocate memory dynamically. All message processing is done in-place
  • Has less features than other Satori RTM SDKs, e.g. it has no auto-reconnection
  • Is likely to be used as a base for building higher-level SDKs (see ios-wrapper for example)

Build

For desktop/server

The build system is using cmake.

Supported compilers:

  • Microsoft Visual Studio 2015 or newer
  • gcc-4.9 or newer
  • clang-3.6 or newer

To build, just execute:

$ mkdir build && cd build
$ cmake ..
$ cmake --build .

To compile with support for the unit tests, first you need to create a credentials.json file in the base directory:

$ cat credentials.json
{
  "endpoint": "wss://<SATORI_HOST>/",
  "appkey": "my_appkey",
  "auth_role_name": "ROLE NAME"
  "auth_role_secret_key": "ROLE SECRET"
  "auth_restricted_channel": "RESTRICTED CHANNEL"
}
  • endpoint is your customer-specific DNS name for RTM access.
  • appkey is your application key.
  • auth_role_name is a role name that permits publishing / subscribing to auth_restricted_channel. Must be not default.
  • auth_role_secret_key is a secret key for auth_role_name.
  • auth_restricted_channel is a channel with subscribe and publish access for auth_role_name role only.

You must use DevPortal to create role and set channel permissions.

To then build with support for tests, execute:

$ mkdir build && cd build
$ cmake .. -DTESTS=1
$ cmake --build .

After that, execute:

$ ./core/test/rtm_unit_tests

As an external project

If your project is also built using CMake, you can take advantage of its ExternalProject feature. This way downloading and building the SDK is automated by CMake itself.

ESP8266/Arduino support

The SDK can generate a library for use with Arduino on ESP8266 platforms. Pass "-DARDUINO-SDK=ON" to CMake to generate a ZIP file that can be imported into Arduino. You can download a prebuilt library from the releases page on Github. Import it into Arduino using SketchInclude LibraryAdd .ZIP library.

TLS support

The SDK requires either OpenSSL, GNUTLS or Apple SSL. Pass one of "-DUSE_OPENSSL=ON", "-DUSE_GNUTLS=ON" or "-DUSE_APPLE_SSL=ON" respectively to CMake.

If no flags are passed, CMake enables Apple SSL for Mac, and for others uses OpenSSL or GnuTLS if OpenSSL is not installed.

iOS wrapper

The SatoriRtmSdkWrapper framework for iOS enables you to integrate your iOS apps with Satori RTM. Using the framework, you can publish and subscribe messages to RTM. There are multiple ways for installing the SatoriRtmSdkWrapper framework in your own project.

Installation with CocoaPods

The SatoriRtmSdkWrapper is available as a CocoaPod. Cocoapod is a dependency manager for iOS, which automates and simplifies the process of using 3rd-party frameworks like SatoriRtmSdkWrapper in your projects. Make sure cocoapods is installed. Then create a podfile for your project if it doesn't already exist. To install the SDK:

  1. Open your Podfile and add the following dependency in the target "<your_app_target>" do section:
use_frameworks!
pod 'SatoriRtmSdkWrapper', :git => "https://github.com/satori-com/satori-rtm-sdk-c.git"
  1. Save your Podfile.
  2. Run pod install from command line.

You've now installed the SatoriRtmSdkWrapper framework. Refer to Framework API usage section to get started.

Manual installation using Source code

Create a local repository by cloning the satori-rtm-sdk-c to your chosen location. Then,

Step 1: Build locally

Follow the normal installation procedure to build the desktop / server variant of the SDK. This will create a configuration header in the build directory which is used during compilation of the iOS framework.

Step 2: Build framework

There are two options to build the RTM framework for iOS:

Option 1 - Build directly from command-line

$ cd ios-wrapper/SatoriRtmSdkWrapper
$ xcodebuild -project SatoriRtmSdkWrapper.xcodeproj -scheme SatoriRtmSdkWrapper-Universal -config <config-name> # where <config-name> can be Debug or Release. Default is Debug if -config option is not specified.

Option 2 - Build in Xcode IDE

$ open SatoriRtmSdkWrapper.xcodeproj

Select SatoriRtmSdkWrapper-Universal target and build.

The SatoriRtmSdkWrapper.framework will be built under ios-wrapper/build directory.

Step 3: Add framework to your project

Once you build the framework, open your app's Xcode project and drag-and-drop the framework under "Embedded Binaries" section under the app's target. Choose "Copy items if needed" and "Create groups" in the dialog box.

Framework API usage

The SatoriRtmSdkWrapper.framework provides you with both Objective-C and C APIs to integrate within your app. Use #import <SatoriRtmSdkWrapper/SatoriRtmSdkWrapper.h> in your application class to make use of these APIs. The Objective-C specific APIs are located in SatoriRtmConnection.h and C APIs can be found in rtm.h

Objective-C Sample Code
// create a new rtm instance with url and appKey
SatoriRtmConnection *rtm = [[SatoriRtmConnection alloc] initWithUrl:"url" andAppkey:"appkey"];

// connect to rtm and provide pdu data handler block
rtm_status status = [rtm connectWithPduHandler:^(SatoriPdu * _Nonnull pdu) {
        //Use pdu
    }];

// subscribe to a channel
unsigned int reqId;
[rtm subscribe:@"channel-name" andRequestId:&reqId];

// publish a string or json to a channel
unsigned int reqId;
[rtm publishString:@"Hello world" toChannel:@"channel-name" andRequestId:&reqId];
[rtm publishJson:@"{\"key\":\"value\"}" toChannel:@"channel-name" andRequestId:&reqId];

// Enable or disable verbose logging of all incoming and outgoing PDUs
rtm.enableVerboseLogging = YES;
rtm.enableVerboseLogging = NO;

// Use wait or waitWithTimeout methods to block until at least one data message gets processed
[rtm wait];
[rtm waitWithTimeout:15];

// For non-blocking wait, use poll
while([rtm poll] >= 0) { sleep(1); }

// Make sure to disconnect when rtm connection is no longer needed
[rtm disconnect];

For Windows (Visual Studio)

Assuming Visual Studio and cmake are installed, open Developer Command Prompt for Visual Studio.

C:\satori-rtm-sdk-c> mkdir vsprj
C:\satori-rtm-sdk-c> cd vsprj
C:\satori-rtm-sdk-c\vsprj> cmake -DBUILD_SHARED_LIBS=ON -G "Visual Studio 14 2015" ..
C:\satori-rtm-sdk-c\vsprj> msbuild satori-sdk-c.sln

Adjust the generator name from "Visual Studio 14 2015" to another if necessary.

For TLS support only "-DUSE_OPENSSL=ON" is supported at this time.

To compile with OpenSSL you need to:

  1. Download and install the appropriate distribution from https://slproweb.com/products/Win32OpenSSL.html
  2. Update your project properties to add additional include folders (pointing to the above install location)
  3. Update project properties to refer to library path of the above installation
  4. Add OpenSSL library files to your linker input libraries. The names for those depend on OpenSSL version: libssl.lib and libcrypto.lib for 1.1.x and libeay.lib and ssleay.lib for 1.0.x. There also could be some variations in naming like ssleay32MD.lib depending on the OpenSSL distribution.

Additionally, WinSock subsystem must be initialized prior to connecting to RTM.

    #include <WinSock2.h>
    #include <Windows.h>
    #include <WS2tcpip.h>

    ...

    WORD wVersionRequested;
    WSADATA wsaData;
    int err;

    wVersionRequested = MAKEWORD(2, 2);
    err = WSAStartup(wVersionRequested, &wsaData);
    if (err != 0) {
        fprintf(stderr, "WSAStartup failed with code %d", err);
    }

Usage

This is an example subscriber program:

#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <rtm.h>

static char const *endpoint = "YOUR_ENDPOINT";
static char const *appkey = "YOUR_ENDPOINT";
static char const *channel = "YOUR_CHANNEL";

void pdu_handler(rtm_client_t *rtm, rtm_pdu_t const *pdu) {
    switch (pdu->action) {
        case RTM_ACTION_SUBSCRIPTION_DATA: {
            char *message;
            while ((message = rtm_iterate(&pdu->message_iterator))) {
                printf("Got message %s\n", message);
            }
            break;
        }
        case RTM_ACTION_SUBSCRIBE_OK: {
            printf("Successfully subscribed to %s\n", channel);
            break;
        }
        default:
            break;
    }
}

int main(void) {
    void *memory = malloc(rtm_client_size);
    rtm_client_t *rtm = rtm_init(memory, pdu_handler, 0);
    int rc = rtm_connect(rtm, endpoint, appkey);

    if (rc != RTM_OK) {
        printf("rtm_connect failed with status %d\n", rc);
        goto cleanup;
    }

    printf("Connected to RTM!\n");

    unsigned int ack_id;
    rc = rtm_subscribe(rtm, channel, &ack_id);

    if (rc != RTM_OK) {
        printf("rtm_subscribe failed with status %d\n", rc);
        goto cleanup;
    }

    rc = rtm_wait_timeout(rtm, 10 /* seconds */);
    if (rc != RTM_OK) {
        fprintf(stderr, "Failed to receive subscribe reply\n");
        goto cleanup;
    }

    while(1) {
        rc = rtm_wait_timeout(rtm, 10 /* seconds */);
        if (rc != RTM_OK && rc != RTM_ERR_TIMEOUT) {
            fprintf(stderr, "Failed to receive a message from channel\n");
            goto cleanup;
        }
    }

cleanup:
    rtm_close(rtm);
    free(rtm);
    return rc;
}

To learn more, look at tutorial and core/examples directories and visit documentation page.

Verbose logging of all incoming and outcoming PDUs

You can enable dumping of all PDUs to stderr either from your code::

  rtm_client_t *rtm = rtm_init(...);
  rtm_enable_verbose_logging(rtm);

or by setting DEBUG_SATORI_SDK environment variable prior to running your application::

  export DEBUG_SATORI_SDK=1
  ./my_program

Using https proxy

The SDK supports working through an https (not http) proxy.

Instead of calling rtm_connect use rtm_connect_via_https_proxy that additionally takes proxy_host and proxy_port like this:

rtm_client_t *client = rtm_init(memory, &pdu_handler, &my_state);
rtm_connect_via_https_proxy(client, "YOUR_ENDPOINT", "YOUR_APPKEY", "http://127.0.0.1:7711");

Migrating from C Core SDK v1

See Migration notes

Missing functionality

  • Handling of network errors is left to the user

About

C SDK for Satori RTM

Resources

License

Stars

Watchers

Forks

Packages

No packages published

Languages

  • C 62.4%
  • C++ 21.4%
  • CMake 6.0%
  • Objective-C 5.1%
  • Python 2.1%
  • Swift 1.8%
  • Other 1.2%