.vscode | ||
CMakeModules | ||
doc | ||
platform | ||
src | ||
tests | ||
.gitignore | ||
CMakeLists.txt | ||
LICENSE | ||
Readme.md |
MQTT library for multiple platforms including embedded targets.
Goals for this library
- Readable code, modern C
- Simple and fool-proof API
- High test coverage (aim is >85% of all lines tested)
- Suitable for embedded or bigger targets (no static allocation though)
Current state
- MQTT connection as a client is working
- All packet types are implemented
- Supports MQTT 3.1.1 (aka. protocol level 4)
- Test have a coverage of about 86.5% (lines) and 100% (functions), untested code is just bail-out error handling for fatal errors (usually programming errors or network failure)
- Platform support for Linux and Windows is working
- Builds on Linux (GCC and Clang) and Windows (MSVC and Clang/c2)
TODO
- Running in MQTT Broker mode (very low prio)
- Implement Protocol level 3 (low prio)
- Implement Draft Protocol level 5
- Support ESP8266 (RTOS)
- Support ESP32 (IDF)
And no, I will not do an Arduino port.
How to use
Building on Linux
Requirements:
- CMake
- GCC
- Checkout the repo
- Run cmake:
mkdir build
cd build
cmake -DCMAKE_BUILD_TYPE=Release -DCMAKE_INSTALL_PREFIX:PATH=/ ..
- Build the library
cd build
make all
- Install/Make a Distribution (Only works in
RELEASE
mode)
make DESTDIR=/path/to/put/it install
The dynamic and static libs will be placed in the lib/
sub-dir of that folder and the include in include/mqtt/mqtt.h
Testing on Linux
Requirements:
- Local MQTT broker, simplest is mosquitto (can be run by calling
mosquitto -v
in a terminal for debug output) Gcov
andLcov
for coverage reports, be aware you'll need a git version of lcov for newer GCC versions (> 6.0.0)
Running the tests:
cd build
make check
Building a coverage report
Attention: Only works in DEBUG
mode, switch like this:
cd build
cmake -DCMAKE_BUILD_TYPE=Debug ..
Building the report:
cd build
make coverage
Be aware to create the coverage report the tests must run, so you'll need the MQTT broker.
Building on Windows
Requirements:
cmake
in path- Visual Studio (Community Edition is sufficient)
- Checked out repo
With Visual Studio (MSVC)
- Run cmake:
mkdir build
cd build
cmake -DCMAKE_BUILD_TYPE=Release .. -G "Visual Studio 15 2017 Win64"
- Build the library
cd build
cmake --build c:/absolute/path/to/build --config Debug --target ALL_BUILD -- /m /property:GenerateFullPaths=true
The dynamic and static libs will be placed in the <config>/
sub-dir of that folder and the include is in src/mqtt.h
Alternatively open the Visual Studio solution in the build
dir in Visual Studio.
Attention: As MSVC is not C99 compliant the tests for encode and decode packet will not work. Use clang
for that.
With Visual Studio (clang)
Requirements:
- Clang/C2 installed (experimental option in Visual Studio Installer)
- Run cmake:
mkdir build
cd build
cmake -DCMAKE_BUILD_TYPE=Release .. -G "Visual Studio 15 2017 Win64" -T v141_clang_c2
- Build the library
cd build
cmake --build c:/absolute/path/to/build --config Debug --target ALL_BUILD -- /m /property:GenerateFullPaths=true
The dynamic and static libs will be placed in the <config>/
sub-dir of that folder and the include is in src/mqtt.h
Alternatively open the Visual Studio solution in the build
dir in Visual Studio.
Testing on Windows
Requirements:
- Local MQTT broker, simplest is mosquitto (can be run by calling
mosquitto -v
in a terminal for debug output)
Attention: Coverage report is not possible on Windows currently
- Build everything
cd build
cmake --build c:/absolute/path/to/build --config Debug --target ALL_BUILD -- /m /property:GenerateFullPaths=true
- Run tests
cd build
ctest -C Debug -T test --output-on-failure
API
Configuration
Create an instance of the config struct as following:
MQTTConfig config = { 0 }; // IMPORTANT: zero out the struct before usage! (this is C99 style)
config.client_id = "my_client";
config.hostname = "localhost";
The snippet above is the absolute minimum you'll need, but there's more to configure if you want:
typedef struct {
char *hostname; /**< Hostname to connect to, will do DNS resolution */
uint16_t port; /**< Port the broker listens on, set to 0 for 1883 default */
char *client_id; /**< Client identification max 23 chars */
bool clean_session; /**< Set to true to reset the session on reconnect */
char *username; /**< User name, set to NULL to connect anonymously */
char *password; /**< Password, set to NULL to connect without password */
char *last_will_topic; /**< last will topic that is automatically published on connection loss */
char *last_will_message; /**< last will message */
bool last_will_retain; /**< tell server to retain last will message */
} MQTTConfig;
Connect to MQTT broker
MQTTHandle *mqtt_connect(MQTTConfig *config, MQTTEventHandler callback, void *callback_context, MQTTErrorHandler error_callback);
config
: MQTT configurationcallback
: Callback function to call on successful connectcallback_context
: Just a void pointer that is stored and given to the callback on connectionerror_callback
: Error handler callback- Returns handle to mqtt connection or NULL on error
If the error handler is called with Host not found or Connection refused, the handler is in charge of freeing the handle by returning true or re-trying by changing settings and calling mqtt_reconnect() and returning false
The error handler should be defined like this:
bool error_handler(MQTTHandle *handle, MQTTConfig *config, MQTTErrorCode code) {
return true; // kill the handle, return false to keep it
}
The event handler looks like this:
void event_handler(MQTTHandle *handle, void *context) {
// The context pointer is the same as the time when you called `mqtt_connect`
}
Re-Connect to MQTT broker
MQTTStatus mqtt_reconnect(MQTTHandle *handle, MQTTEventHandler callback, void *callback_context);
handle
: MQTT Handle frommqtt_connect
callback
: Callback to call when reconnect was successfulcallback_context
: Pointer to give the callback- Returns status code
Usually called in the MQTTErrorHandler callback, if called on a working connection the connection will be disconnected before reconnecting.
If there were registered subscriptions they will be re-instated after a successful reconnect.
Subscribe to a topic
MQTTStatus mqtt_subscribe(MQTTHandle *handle, char *topic, MQTTQosLevel qos_level, MQTTPublishEventHandler callback);
handle
: MQTT Handle frommqtt_connect
topic
: Topic to subscribeqos_level
: Maximum QoS level to subscribe forcallback
: Callback function to call when receiving something for that topic- Returns status code
You may use the same callback function for multiple topics. The callback function looks like this:
void publish_handler(MQTTHandle *handle, char *topic, char *payload) {
// the payload is zero terminated by the library, so it can be used as
// a standard c-string immediately.
}
Un-Subscribe from a topic
MQTTStatus mqtt_unsubscribe(MQTTHandle *handle, char *topic);
handle
: MQTT Handle frommqtt_connect
topic
: Topic to unsubscribe- Returns status code
Publish something to the broker
MQTTStatus mqtt_publish(MQTTHandle *handle, char *topic, char *payload, MQTTQosLevel qos_level, MQTTPublishEventCallback callback);
handle
: MQTT Handle frommqtt_connect
topic
: Topic to publish topayload
: Message payload to publishqos_level
: QoS Level for the publish (0 = Fire and forget, 1 = At least once, 2 = One time for sure)callback
: Callback function that is called when publish cleared the QoS handlers- Returns status code
This uses a c-string as the payload, theoretically the protocol would allow for binary payloads, but this is currently not supported.
Disconnect from MQTT broker
MQTTStatus mqtt_disconnect(MQTTHandle *handle, MQTTEventHandler callback, void *callback_context);
handle
: MQTT Handle frommqtt_connect
- Returns status code
Attention: do not use the handle after calling this function, all resources will be freed, this handle is now invalid!