2021-08-01 14:10:21 +01:00
|
|
|
// Copyright (c) 2020 Cesanta Software Limited
|
|
|
|
// All rights reserved
|
|
|
|
//
|
|
|
|
// Example MQTT client. It performs the following steps:
|
|
|
|
// 1. Connects to the MQTT server specified by `s_url` variable
|
|
|
|
// 2. When connected, subscribes to the topic `s_topic`
|
|
|
|
// 3. Publishes message `hello` to the `s_topic`
|
|
|
|
// 4. Receives that message back from the subscribed topic and exits
|
|
|
|
//
|
2023-02-21 12:07:48 -03:00
|
|
|
// To enable SSL/TLS, see https://mongoose.ws/tutorials/tls/#how-to-build
|
2021-08-01 14:10:21 +01:00
|
|
|
|
2023-07-25 18:41:41 -03:00
|
|
|
#include "mongoose.h"
|
|
|
|
|
2023-09-17 09:07:57 +01:00
|
|
|
static const char *s_url =
|
2023-07-25 18:41:41 -03:00
|
|
|
#if MG_TLS
|
2023-09-17 09:07:57 +01:00
|
|
|
"wss://broker.hivemq.com:8884/mqtt";
|
2023-02-22 16:38:21 -03:00
|
|
|
#else
|
2023-09-17 09:07:57 +01:00
|
|
|
"ws://broker.hivemq.com:8000/mqtt";
|
2023-02-22 16:38:21 -03:00
|
|
|
#endif
|
|
|
|
|
2021-08-01 14:10:21 +01:00
|
|
|
static const char *s_topic = "mg/test";
|
|
|
|
|
2024-01-08 17:34:34 -03:00
|
|
|
static void fn(struct mg_connection *c, int ev, void *ev_data) {
|
2021-08-01 14:10:21 +01:00
|
|
|
if (ev == MG_EV_ERROR) {
|
|
|
|
// On error, log error message
|
2022-02-12 18:17:25 +00:00
|
|
|
MG_ERROR(("%p %s", c->fd, (char *) ev_data));
|
2023-09-17 09:07:57 +01:00
|
|
|
} else if (ev == MG_EV_CONNECT) {
|
|
|
|
if (mg_url_is_ssl(s_url)) {
|
|
|
|
struct mg_tls_opts opts = {.ca = mg_unpacked("/certs/ca.pem"),
|
|
|
|
.name = mg_url_host(s_url)};
|
|
|
|
mg_tls_init(c, &opts);
|
|
|
|
}
|
2021-08-01 14:10:21 +01:00
|
|
|
} else if (ev == MG_EV_WS_OPEN) {
|
|
|
|
// WS connection established. Perform MQTT login
|
2022-02-12 18:17:25 +00:00
|
|
|
MG_INFO(("Connected to WS. Logging in to MQTT..."));
|
2023-09-17 09:07:57 +01:00
|
|
|
struct mg_mqtt_opts opts = {
|
|
|
|
.qos = 1, .topic = mg_str(s_topic), .message = mg_str("goodbye")};
|
2021-08-01 14:10:21 +01:00
|
|
|
size_t len = c->send.len;
|
2021-11-02 16:40:25 +00:00
|
|
|
mg_mqtt_login(c, &opts);
|
2021-08-01 14:10:21 +01:00
|
|
|
mg_ws_wrap(c, c->send.len - len, WEBSOCKET_OP_BINARY);
|
2022-07-04 17:47:17 +01:00
|
|
|
c->is_hexdumping = 1;
|
2021-08-01 14:10:21 +01:00
|
|
|
} else if (ev == MG_EV_WS_MSG) {
|
|
|
|
struct mg_mqtt_message mm;
|
|
|
|
struct mg_ws_message *wm = (struct mg_ws_message *) ev_data;
|
2022-07-04 17:47:17 +01:00
|
|
|
uint8_t version = c->is_mqtt5 ? 5 : 4;
|
2022-02-12 18:17:25 +00:00
|
|
|
MG_INFO(("GOT %d bytes WS msg", (int) wm->data.len));
|
2024-03-15 07:42:24 +00:00
|
|
|
while ((mg_mqtt_parse((uint8_t *) wm->data.buf, wm->data.len, version,
|
2022-07-04 14:32:12 -03:00
|
|
|
&mm)) == MQTT_OK) {
|
2021-08-01 14:10:21 +01:00
|
|
|
switch (mm.cmd) {
|
|
|
|
case MQTT_CMD_CONNACK:
|
|
|
|
mg_call(c, MG_EV_MQTT_OPEN, &mm.ack);
|
|
|
|
if (mm.ack == 0) {
|
|
|
|
struct mg_str topic = mg_str(s_topic), data = mg_str("hello");
|
|
|
|
size_t len = c->send.len;
|
2022-02-12 18:17:25 +00:00
|
|
|
MG_INFO(("CONNECTED to %s", s_url));
|
2023-04-20 09:59:27 +03:00
|
|
|
struct mg_mqtt_opts sub_opts;
|
|
|
|
memset(&sub_opts, 0, sizeof(sub_opts));
|
|
|
|
sub_opts.topic = topic;
|
|
|
|
sub_opts.qos = 1;
|
|
|
|
mg_mqtt_sub(c, &sub_opts);
|
2022-06-28 16:27:49 -03:00
|
|
|
len = mg_ws_wrap(c, c->send.len - len, WEBSOCKET_OP_BINARY);
|
2024-03-15 07:42:24 +00:00
|
|
|
MG_INFO(("SUBSCRIBED to %.*s", (int) topic.len, topic.buf));
|
2023-04-20 09:59:27 +03:00
|
|
|
struct mg_mqtt_opts pub_opts;
|
|
|
|
memset(&pub_opts, 0, sizeof(pub_opts));
|
|
|
|
pub_opts.topic = topic;
|
|
|
|
pub_opts.message = data;
|
|
|
|
pub_opts.qos = 1, pub_opts.retain = false;
|
|
|
|
mg_mqtt_pub(c, &pub_opts);
|
2024-03-15 07:42:24 +00:00
|
|
|
MG_INFO(("PUBLISHED %.*s -> %.*s", (int) data.len, data.buf,
|
|
|
|
(int) topic.len, topic.buf));
|
2021-08-01 14:10:21 +01:00
|
|
|
len = mg_ws_wrap(c, c->send.len - len, WEBSOCKET_OP_BINARY);
|
|
|
|
} else {
|
2022-02-12 18:17:25 +00:00
|
|
|
MG_ERROR(("%lu MQTT auth failed, code %d", c->id, mm.ack));
|
2023-08-17 16:45:10 -03:00
|
|
|
c->is_draining = 1;
|
2021-08-01 14:10:21 +01:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case MQTT_CMD_PUBLISH: {
|
2022-02-12 18:17:25 +00:00
|
|
|
MG_DEBUG(("%lu [%.*s] -> [%.*s]", c->id, (int) mm.topic.len,
|
2024-03-15 07:42:24 +00:00
|
|
|
mm.topic.buf, (int) mm.data.len, mm.data.buf));
|
|
|
|
MG_INFO(("RECEIVED %.*s <- %.*s", (int) mm.data.len, mm.data.buf,
|
|
|
|
(int) mm.topic.len, mm.topic.buf));
|
2023-08-17 16:45:10 -03:00
|
|
|
c->is_draining = 1;
|
2021-08-01 14:10:21 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2024-03-15 07:42:24 +00:00
|
|
|
wm->data.buf += mm.dgram.len;
|
2021-08-01 14:10:21 +01:00
|
|
|
wm->data.len -= mm.dgram.len;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ev == MG_EV_ERROR || ev == MG_EV_CLOSE) {
|
2024-01-08 17:34:34 -03:00
|
|
|
*(bool *) c->fn_data = true; // Signal that we're done
|
2021-08-01 14:10:21 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int main(void) {
|
2022-08-01 11:19:32 +01:00
|
|
|
struct mg_mgr mgr; // Event manager
|
|
|
|
bool done = false; // Event handler flips it to true when done
|
|
|
|
mg_mgr_init(&mgr); // Initialise event manager
|
|
|
|
mg_log_set(MG_LL_DEBUG); // Set log level
|
2023-11-26 15:38:28 -03:00
|
|
|
mg_ws_connect(&mgr, s_url, fn, &done, "%s", // Create client connection
|
|
|
|
"Sec-Websocket-Protocol: mqtt\r\n"); // Request MQTT protocol
|
|
|
|
while (done == false) mg_mgr_poll(&mgr, 1000); // Event loop
|
|
|
|
mg_mgr_free(&mgr); // Finished, cleanup
|
2021-08-01 14:10:21 +01:00
|
|
|
return 0;
|
|
|
|
}
|