C-Core API & SDK Docs 5.4.0
SDK version
Always use the latest SDK version to have access to the newest features and avoid security vulnerabilities, bugs, and performance issues.
Overview
The PubNub C-Core Software Development Kit (SDK) provides a portable C client for building real-time applications with the PubNub service. This guide introduces core concepts, helps you build a working prototype quickly, and then covers details such as thread safety and proxy configuration.
Acronyms used in this document are defined on first use:
- SDK: Software Development Kit
- API: Application Programming Interface
- POSIX: Portable Operating System Interface
- UWP: Universal Windows Platform
- SSL/TLS: Secure Sockets Layer / Transport Layer Security
- UUID: Universally Unique Identifier
- CPU: Central Processing Unit
- I/O: Input/Output
- HTTP: Hypertext Transfer Protocol
- JSON: JavaScript Object Notation
- REST: Representational State Transfer
Get code: source
https://github.com/pubnub/c-core
Hello World
There is no installation of this SDK. Just clone the Git(Hub) repo, or download a ZIP or tarball of a release from https://github.com/pubnub/c-core/releases  It has the code and example Makefiles to get you started.
Makefiles
Use the Makefiles as a starting point in your own projects (whether they are based on Make or some other build tool / system or IDE).
Makefile for POSIX without SSL/TLS support
The Makefile for POSIX without SSL/TLS support is available at /posix/posix.mk  See /posix/README.md for info on how to build on POSIX (and POSIX-like) systems.
Makefile for POSIX with SSL/TLS support
The Makefile for POSIX with SSL/TLS, (via OpenSSL) is available at /openssl/posix.mk in the repo. See /openssl/README.md for info on how to build w/OpenSSL on POSIX and other OpenSSL related data.
Makefile for Windows with UWP support
The Makefile for Windows C with UWP support is available at /windows/uwp.mk in the repo. See /windows/README.md for info on how to build with UWP on Windows.
Makefile for Windows with UWP and SSL/TLS support
The Makefile for Windows C with UWP and SSL/TLS, (via OpenSSL) is available at /openssl/uwp.mk in the repo. See /openssl/README.md for info on how to build with UWP on Windows.
Including the correct header
The calling pattern you choose to use with PubNub (Synchronous vs Callback) will determine which header to import. More information on these calling patterns can be found later on in this guide.
Sync
If using the synchronous (sync) pattern, import only pubnub_sync.h:
1#include "pubnub_sync.h"
Callback
If using the callback pattern, import pubnub_callback.h and pthread.h:
1#include "pubnub_callback.h"
2#include <pthread.h>
Runtime selection
If using the runtime selection pattern, import pubnub_ntf_enforcement.h:
1#include "core/pubnub_ntf_enforcement.h"
Memory allocation
This client uses dynamic memory allocation for the PubNub contexts, but the usage is the same as for any other Pubnub C client - always use pubnub_alloc() to create a context (and check its return value) and always use pubnub_free() to dispose of a context.
Required UUID
Always set the UUID to uniquely identify the user or device that connects to PubNub. This UUID should be persisted, and should remain unchanged for the lifetime of the user or the device. If you don't set the UUID, you won't be able to connect to PubNub.
1#include "pubnub_alloc.h"
2#include <stdio.h>
3int main()
4{
5    pubnub_t *ctx = pubnub_alloc();
6    if (NULL == ctx) {
7        puts("Couldn't allocate a Pubnub context");
8        return -1;
9    }
10    /* Do something with ctx...
11        and then: */
12    pubnub_free(ctx);
13    return 0;
14}
Timers
We only provide one timer - the (total) transaction timer. In general, it is started when a transaction is started and stopped when a transaction is finished. If it expires, the transaction will be cancelled. Keep in mind that this canceling is local, so, for example, if you already published a message, but, for some reason, the HTTP response didn't arrive in time, this canceling will not revoke the publish - it will just stop the wait for response.
If the transaction timer expires, the outcome of the transaction will be timeout - different than when you cancel a transaction yourself.
The actual duration of the timer is at least as long as you set it. It could be significantly higher, depending on various platform issues. But, in general, it will be close to what you set.
You should set the timer after initializing the context and before starting a transaction. The duration you set will be used for all subsequent transactions, regardless of their type (i.e. for publish and subscribe and all other).
Thread safety
C-core supports thread-safe operation, though, for performance, you may think about not using it. To use thread-safety support, define the preprocessor symbol PUBNUB_THREADSAFE (just define it, the value does not matter).
Thread-safe usage
Thread safety is internal. Just because you can access the PubNub context through the PubNub C-core SDK API from different threads safely, doesn't mean you're off the hook for your own data that is related to a context. For example, if you're using the callback interface and signalling an event from it to other (worker) thread(s), you have to synchronise that data transfer yourself.
If you compiled thread-safety support in, you are free to access the same context from different threads, pretty much in any way you wish. However, there are some advised guidelines you should follow:
- If you're using the sync blocking interface, threads that come to wait on the context may wait a long time, so try to avoid it (also, re-think your whole need for a thread-safe C-core)
- If you're using the sync non-blocking interface by calling pubnub_await, things are pretty much the same as for sync blocking interface
- If you're using the sync non-blocking interface and avoid pubnub_await, waiting threads will not block so long, but, pretty much the only useful thing you can do is cancel a transaction from another thread.
- Using the sync interface, it's perfectly fine to call pubnub_awaitorpubnub_last_resultin different threads, but, you probably shouldn't do that, as it will make debugging harder.
- If you're using the callback interface, it's perfectly fine to call Functions from your callback, but, you should avoid doing that, except for some helper functions. Following this guideline will make your debugging, thus life, a lot easier
Thread-unsafe usage
If you compile without thread-safety support, obviously, you will have an SDK which is not thread safe - that is, it is not safe to use a single context from more than one thread at the same time. So, if you're using such SDK configuration in a multithreaded code, which, on POSIX, you likely are, then:
- If at all possible, use a single context from only one thread - the one that created it
- If 1. is not possible, provide some synchronization yourself, for example, using pthread condition variables, or just mutexes, or some higher abstraction, like message queues
- As a special case, if you're using the callback interface, you can start a transaction in one thread and then don't touch the context from that thread any more - use it only in the callback. This is safe.
Context usage
Keep in mind that it is perfectly safe to use different contexts from different threads at the same time. To each (thread) its own (context).
Transactions and operations
The C-Core SDK operates as a set of transactions. A transaction is initiated by the client SDK and is defined as a single message exchange between the SDK and PubNub service. Every interaction that the client SDK initiates with PubNub is sequenced as a series of transactions which ultimately results in a PubNub service-specific operation.
Status and events
The SDK provides a set of status and event identifiers which can help developers interact with the library. The status identifier codes are returned as part of the SDK's API invocation. These are used by the developer to check for status of transactions or for detecting normal / abnormal conditions in an API call. Some of the commonly used status codes are as follows
- PNR_OK: Success, the transaction finished successfully
- PNR_STARTED: The previously initiated transaction has started.
- PNR_IN_PROGRESS: Indicates that the previous transaction with PubNub service is still in progress.
Refer to the API docs for a complete list of status identifiers supported by the library.
Events refer to the PubNub REST operations which are initiated by the client SDK. The most common example of events are subscribe and publish. A client subscribing for a channel is a subscribe event and a client publishing a message on a channel is a publish event.
Some of the common event identifiers are as follows:
- PBTT_SUBSCRIBE: Subscriber operation
- PBTT_PUBLISH: Publish operation
Refer to the API docs for a complete list of operations supported by the SDK.
Calling patterns
This SDK provides sync, callback (notification), and runtime_selection (runtime selection of the calling pattern) interfaces for retrieving the outcome of a Pubnub request/transaction/operation.
Calling patterns at a glance
| Calling pattern | Best for | Pros | Considerations | 
|---|---|---|---|
| Sync | Simple flows and prototypes | Easy to use and reason about | Can block threads; consider non-blocking I/O for responsiveness | 
| Callback | Event-driven or multi-operation flows | More flexible; lower CPU usage | Harder to implement; requires synchronization | 
| Runtime selection | Libraries or apps that need both patterns | Dynamic choice per context | Requires additional setup and enforcement per context | 
Sync
The sync interface works like this:
- Start a transaction (say, publish - using pubnub_publish())
- Either pubnub_await()the outcome, or use your own loop in which you checkif (PNR_STARTED != pubnub_last_result())
- Handle the outcome as you wish
This is illustrated in the Hello World example below (which is the same for any platform that supports sync interface).
Callback
The callback interface is somewhat more flexible, uses less CPU resources, but is, in general, a little harder to use. One way to use it is to emulate the sync interface:
- Create a callback function (my_callback) per the prototype required bypubnub_register_callback()
- In my_callback(), use a condition variable to signal that outcome was received
- Set the callback via pubnub_register_callback()
- Start a transaction (say, publish - using pubnub_publish())
- Wait on the condition variable (the same one used in my_callback)
- Handle the outcome as you wish
This is illustrated in the Hello World example below, using pthreads condition variable. Obviously, on platforms that don't support pthreads you will use some similar API (for example, SetEvent/WaitForSingleObject on Windows).
There are other ways to use the callback interface, like the state machine or similar, where the callback will handle the outcome of a transaction but will also start the next Pubnub transaction, or do other stuff that it needs to do. This is very application specific, so we don't provide an example here.
Runtime selection
The runtime_selection interface allows you to select the calling pattern at runtime for PubNub context. You can select the sync or callback interfaces. By default it selects the sync API.
#include "core/pubnub_ntf_enforcement.h"
// other necessary includes
pubnub_t*       pbp_sync  = pubnub_alloc();
pubnub_t*       pbp_callback  = pubnub_alloc();
// you MUST enforce the calling pattern before initializing the context
pubnub_enforce_api(pbp_sync, PNA_SYNC);
pubnub_enforce_api(pbp_callback, PNA_CALLBACK);
pubnub_init(pbp_sync, "my_publish_key", "my_subscribe_key");
pubnub_init(pbp_callback, "my_publish_key", "my_subscribe_key");
Proxy configuration
Proxy setting
You need to link in the required modules and set the proxy. Use int pubnub_set_proxy_manual() from pubnub_proxy.h. If the proxy server is an authenticating one, you may need to set the username/password, with pubnub_set_proxy_authentication_username_password().
Sync code sample (portable)
Required UUID
Always set the UUID to uniquely identify the user or device that connects to PubNub. This UUID should be persisted, and should remain unchanged for the lifetime of the user or the device. If you don't set the UUID, you won't be able to connect to PubNub.
1#include "pubnub_sync.h"
2#include <stdio.h>
3int main()
4{
5    enum pubnub_res pbresult;
6    pubnub_t *ctx = pubnub_alloc();
7    if (NULL == ctx) {
8        puts("Couldn't allocate a Pubnub context");
9        return -1;
10    }
11    pubnub_init(ctx, "demo", "demo");
12    pubnub_set_user_id(ctx, "myUniqueUser_Id");
13    /* Initial Subscribe on the "hello_world" channel */
14    pubnub_subscribe(ctx, "hello_world", NULL);
15    pbresult = pubnub_await(ctx);
Callback code sample (syncing with main, POSIX-specific)
1#include "pubnub_callback.h"
2#include <stdbool.h>
3#include <pthread.h>
4#include <stdio.h>
5
6struct MyUserData {
7    pthread_mutex_t mut;
8    bool triggered;
9    pthread_cond_t cond;
10};
11static struct MyUserData m_user_data;
12
13void example_callback(pubnub_t *pb, enum pubnub_trans trans, enum pubnub_res result, void *user_data)
14{
15    struct MyUserData *my_user_data = user_data;
Copy and paste examples
In addition to the Hello World sample code, we also provide some copy and paste snippets of common API functions:
Init
Instantiate a new Pubnub instance. Only the subscribe_key is mandatory. Also include publish_key if you intend to publish from this instance, and the secret_key if you wish to perform Access Manager administrative operations from this C-Core instance.
Secret key security
It is not a best practice to include the secret key in client-side code for security reasons.
When you init with secret_key, you get root permissions for the Access Manager. With this feature you don't have to grant access to your servers to access channel data. The servers get all access on all channels.
Required UUID
Always set the UUID to uniquely identify the user or device that connects to PubNub. This UUID should be persisted, and should remain unchanged for the lifetime of the user or the device. If you don't set the UUID, you won't be able to connect to PubNub.
1pubnub_init(ctx, /*publish key*/"demo", /*subscribe key*/"demo");
2pubnub_set_user_id(ctx, "myUniqueUser_Id");
Subscribe
Subscribe (listen on) a channel (it's async!):
Message retrieval
Typically, you will want two separate contexts for publish and subscribe. When changing the active set of subscribed channels, first call pubnub_leave() on the old set. The pubnub_subscribe() interface is essentially a transaction to start listening on the channel for arrival of next message. This has to be followed by pubnub_get() call to retrieve the actual message, once the subscribe transaction completes successfully. This needs to be performed every time it is desired to retrieve a message from the channel.
1pubnub_subscribe(ctx, "my_channel", NULL);
2pbresult = pubnub_await(ctx);
3if (PNR_OK == pbresult) {
4    char const *message = pubnub_get(ctx);
5    while (message != NULL) {
6        message = pubnub_get(ctx);
7    }
8}
Publish
Publish a message to a channel:
1pubnub_publish(ctx, "my_channel", "\"message\"");
2pbresult = pubnub_await(ctx);
3if (PNR_OK == pbresult) {
4    /* Published successfully */
5}
Here now
Get occupancy of who's here now on the channel by UUID:
Presence
Requires you to enable the Presence add-on for your key. Refer to the How do I enable add-on features for my keys?  knowledge base article for details on enabling features.
1// Sync
2
3pubnub_here_now(ctx, "my_channel", NULL);
4pbresult = pubnub_await(ctx);
5if (PNR_OK == pbresult) {
6    char const *json_response = pubnub_get(ctx);
7}
8
9//callback
10
11int here_now(pubnub_t *pn) {
12  char const *chan = "my_channel";
13  char const *msg;
14  enum pubnub_res res;
15
Presence
Subscribe to real-time Presence events, such as join, leave, and timeout, by UUID. Setting the presence attribute to a callback will subscribe to presents events on my_channel:
Presence
Requires you to enable the Presence add-on for your key. Refer to the How do I enable add-on features for my keys?  knowledge base article for details on enabling features.
1// Sync
2
3char *presence_channel = malloc(strlen(channel) + strlen(PUBNUB_PRESENCE_SUFFIX) + 1);
4strcpy(presence_channel, channel);
5strcat(presence_channel, PUBNUB_PRESENCE_SUFFIX);
6pubnub_subscribe(ctx, presence_channel, NULL);
7pbresult = pubnub_await(ctx);
8if (PNR_OK == pbresult) {
9    char const *presence_event = pubnub_get(ctx);
10    while (presence_event != NULL) {
11        presence_event = pubnub_get(ctx);
12    }
13}
14
15// Callback
History
Retrieve published messages from archival storage:
Message Persistence
Requires that the Message Persistence add-on is enabled for your key. How do I enable add-on features for my keys? - see https://support.pubnub.com/hc/en-us/articles/360051974791-How-do-I-enable-add-on-features-for-my-keys-
1// Sync
2enum pubnub_res res;
3
4pubnub_history(pn, "history_channel", 10, false);
5res = pubnub_await(pn);
6
7if (PNR_OK == res) {
8  puts("Got history! Messages:");
9  for (;;) {
10    const char *msg = pubnub_get(pn);
11    if (NULL == msg) {
12      break;
13    }
14    puts(msg);
15  }
Unsubscribe
Stop subscribing (listening) to a channel.
To unsubscribe, you need to cancel a subscribe transaction.
- 
If you configured the SDK to be thread-safe, you can cancel at any time, but, the cancelling may actually fail - i.e., your thread may wait for another thread to finish working with the context, and by the time your cancel request gets processed, the transaction may finish. 
- 
If you configured the SDK to not be thread-safe, the only safe way to do it is to use the syncinterface and:- Set the context to use non-blocking I/O.
- Wait for the outcome in a loop, checking for pubnub_last_result()- rather than callingpubnub_await().
- If a condition occurs that prompts you to unsubscribe, callpubnub_cancel().
- Wait for the cancellation to finish (here you can call pubnub_await(), unless you want to do other stuff while you wait).
 
- Set the context to use 
1pbresult = pubnub_subscribe(ctx, "my_channel", NULL);
2/* If we don't set non-blocking I/O, we can't get out of a blocked read */
3pubnub_set_non_blocking_io(ctx);
4/* Can't use pubnub_await() here, it will block */
5while (PNR_STARTED == pbresult) {
6    pbresult = pubnub_last_result(ctx);
7    /* Somehow decide we want to quit / unsubscribe */
8    if (should_stop()) {
9        pubnub_cancel(ctx);
10        /* If we don't have anything else to do, it's OK to await now,
11        but you could again have a loop "against" pubnub_last_result()
12        */
13        pbresult = pubnub_await(ctx);
14        break;
15    }