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)
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 toauth_restricted_channel
. Must be notdefault
.auth_role_secret_key
is a secret key forauth_role_name
.auth_restricted_channel
is a channel with subscribe and publish access forauth_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
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.
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 Sketch → Include Library → Add .ZIP library.
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.
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.
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:
- 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"
- Save your Podfile.
- Run
pod install
from command line.
You've now installed the SatoriRtmSdkWrapper framework. Refer to Framework API usage section to get started.
Create a local repository by cloning the satori-rtm-sdk-c to your chosen location. Then,
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.
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.
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.
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
// 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];
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:
- Download and install the appropriate distribution from https://slproweb.com/products/Win32OpenSSL.html
- Update your project properties to add additional include folders (pointing to the above install location)
- Update project properties to refer to library path of the above installation
- 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);
}
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.
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
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");
See Migration notes
- Handling of network errors is left to the user