acl/lib_acl_cpp/samples/mqtt/README.md

124 lines
5.2 KiB
Markdown
Raw Normal View History

2021-03-28 22:23:54 +08:00
# The streaming mqtt parser in acl
## About
Because most of mqtt parsers at present are bound with network IO, which limits their using scope when we just want a mqtt parser to be used in our own network libs. I've complete a streaming mqtt in FSMFinite State Machinemode which can be used by any network libs(TCP ore UDP).
The streaming mqtt parser implemented in acl was written in standard C++, which can be used client or server sides, including all the mqtt commands such as CONNECT, CONNACK, PUBLISH, PUBACK, etc.
## C++ API
The MQTT module in acl includes MQTT commands and MQTT network IO that each MQTT command was implemented as a C++ class. The user can set the MQTT C++ object to build for each MQTT command, and send the data after serializing the C++ object.
The MQTT commands classes include **`mqtt_connect, mqtt_connack, mqtt_disconnect, mqtt_publish, mqtt_puback, mqtt_pubcomp, mqtt_pubrec, mqtt_rel, mqtt_subscribe, mqtt_suback, mqtt_unsubscribe, mqtt_unsuback, mqtt_pingreq, mqtt_pingresp`** which are inheriting from the base class **`mqtt_message`**. The **`mqtt_header`** class in acl mqtt is used to parse or build the mqtt header.
The **`mqtt_client, mqtt_aclient`** are the network communication classes of acl mqtt, which **`mqtt_client`** uses the sync IO, and **`mqtt_aclient`** uses async IO.
## Samples
Although the mqtt parser of acl is alone from network IO, the network IO module for mqtt was also provided, including sync IO and async IO. Some mqtt samples were written by acl mqtt are shown in [mqtt_samples path](./).
- **mqtt_aclient:** A MQTT client in async IO mode;
- **mqtt_aserver:** A MQTT server in async IO mode;
- **mqtt_client:** A MQTT client in sync IO mode;
- **mqtt_server:** A MQTT server in sync IO mode;
- **mqtt_pub:** A MQTT publish client in sync IO mode.
### Write a MQTT client with acl mqtt
At first, we should construct a MQTT command C++ class object as below:
```C++
acl::mqtt_connect message;
message.set_cid("client-id-test-xxx");
message.set_username("user-zsx");
message.set_passwd("pass-zsx");
message.set_will_qos(acl::MQTT_QOS0);
message.set_will_topic("test/topic");
message.set_will_msg("msg-hello");
```
The second, construct a MQTT IO object and send the MQTT command object:
```C++
const char* addr = "127.0.0.1|1883";
int conn_timeout = 10 /* seconds */, rw_timeout = 0;
acl::mqtt_client conn(addr, conn_timeout, rw_timeout);
if (!conn.send(message)) {
printf("send MQTT message failed!\r\n");
} else {
printf("send MQTT message success!\r\n");
}
```
At last, waiting for the MQTT messages from MQTT server:
```C++
static bool handle_message(acl::mqtt_client& conn, acl::mqtt_message& message) {
acl::mqtt_type_t type = message.get_header().get_type();
switch (type) {
case acl::MQTT_CONNACK:
return handle_connack(conn, message);
case acl::MQTT_PINGREQ:
return handle_pingreq(conn);
case acl::MQTT_PINGRESP:
return handle_pingresp();
case acl::MQTT_DISCONNECT:
return handle_disconnect(message);
case acl::MQTT_SUBACK:
return handle_suback(conn, message);
case acl::MQTT_PUBLISH:
return handle_publish(conn, message);
default:
printf("unknown type=%d\r\n", (int) type);
return false;
}
}
static void waiting_messages(acl::mqtt_client& conn) {
while (true) {
acl::mqtt_message* res = conn.get_message();
if (res == NULL) {
printf("read message error\r\n");
break;
}
if (!handle_message(conn, *res)) {
delete res;
break;
}
delete res;
}
}
```
After getting the **CONNACK** we can send **SUBSCRIBE** command to MQTT server:
```C++
static __thread unsigned short __pkt_id = 0;
static bool handle_connack(acl::mqtt_client& conn, const acl::mqtt_message& message) {
const acl::mqtt_connack& connack = (const acl::mqtt_connack&) message;
printf("%s => connect code=%d\r\n", __FUNCTION__, connack.get_connack_code());
// constrcut SUBSCRIBE command and send it to the MQTT server
acl::mqtt_subscribe sub;
if (++__pkt_id == 0) {
__pkt_id = 1; // the pkt id must more than zero.
}
sub.set_pkt_id(__pkt_id);
sub.add_topic("test/topic1", acl::MQTT_QOS1);
sub.add_topic("test/topic2", acl::MQTT_QOS1);
sub.add_topic("test/topic3", acl::MQTT_QOS1);
if (conn.send(sub)) {
printf("send subscribe ok\r\n");
return true;
}
printf("send subscribe error\r\n");
return false;
}
```
We can easily complete The left functions in `handle_message()` above including `handle_pingreq, handle_pingresp, handle_disconnect, handle_suback, handle_publish`.
### Write a MQTT server with acl mqtt
## Reference
The source files are in [mqtt src files](../../src/mqtt/), and the header files are in [mqtt header files](../../include/acl_cpp/mqtt/).