新鲜出炉的MQTT库¶
作者:陈之敏 时间:2020年08月15日 关键字:csdk、RDA8910、二次开发、MQTT、PAHO
目录¶
最近听说有的小伙伴看了我的教程后,有一些问题都跑到官方的gitee上面去问去了。导致官方的人没搞懂问的是啥,小伙伴们也没能知道自己想要的答案。给大家造成了困扰,这里我说声抱歉。
既然出现了这个问题,我这里就声明一下,本系列教程所涉及的内容(demo)不是官方的作品。我个人觉得官方的demo内容太多太全,往往都是把一个模块内所有的东西全部放在一起。这样的话对新手不是很友好,阅读起来也比较费劲。我就把官方的部分demo进行相关的简化,并推出教程这样的话可能会对新手朋友们有一定的帮助。
有时候周末闲暇时间我也会加上一些我觉得好玩的模块在里面,这些可能在官方的demo都没有,比如cJSON、PAHO-MQTT、http-client。
这就是官方的代码仓库。
git clone --recursive https://gitee.com/openLuat/Luat_CSDK_Air724U.git
当然各位小伙伴在看本教程时,我建议还是使用我下面提供的仓库比较好,看完之后在迁移到官方的仓库⇧。
git clone --recursive https://gitee.com/chenxiahuaxu/RDA8910_CSDK.git
再看本教程过程中如果遇到了问题,可以在本人的代码仓库下面评论。也可以在本人的博客下面评论。我要是看到的话,并且这个问题在我的能力范围的话我会尽力解答的(非官方,不要对我要求太多哦,要求太多我可能就不管啦)。
一、前言¶
MQTT是一个基于客户端-服务器的消息发布/订阅传输协议。MQTT协议是轻量、简单、开放和易于实现的,这些特点使它适用范围非常广泛。在很多情况下,包括受限的环境中,如:机器与机器(M2M)通信和物联网(IoT)。其在,通过卫星链路通信传感器、偶尔拨号的医疗设备、智能家居、及一些小型化设备中已广泛使用。
物联网应用中都离不开MQTT。玩物联网的人不会用MQTT那就是没有灵魂。目前官方提供的CSDK也是没有
MQTT库的,所以为了大家,在下花了一点时间移植了一套开源的用于嵌入式领域的MQTT库。
有人讲你那个HTTP库都自己写了,这个MQTT库怎么不自己写。这里声明一下HTTP的库不是我想自己写的,是因为我没找到合适的开源的HTTP库,移植起来很困难。被逼无奈自己写了一个简单的请求库,用在嵌入式平台正常情况下也够用了。
闲话少说,开始干正事。 ## 二、编写测试程序 ##
2.1、了解本例程所用到的函数
使用MQTT服务需要包含#include "MQTTClient.h""
头文件,我们这里只用到了7个函数,分别是:
>/新的网络设置 > @param
n:指向网络结构的指针,包含网络的配置信息。@return 无/
void
NewNetwork
(Network *n)
/**@简短连接网络功能@param n:指向网络结构的指针 包含网络的配置信息。ip:服务器IP。 *port:服务器端口。 **/
int
ConnectNetwork
(Network n, charip, int port)
/创建一个MQTT客户端对象@参数客户端 @参数网络@param command_timeout_ms *@参数 **/
DLLExport void
MQTTClientInit
(MQTTClient* client, Network* network, unsigned int command_timeout_ms,unsigned char* sendbuf, size_t sendbuf_size, unsigned char* readbuf, size_t readbuf_size);
/MQTT Connect-在网络上发送MQTT连接数据包并等待Connack调用此对象之前,必须将nework对象连接到网络端点 @param选项-连接选项@返回成功代码 **/
DLLExport int
MQTTConnect
(MQTTClient* client, MQTTPacket_connectData* options);
/MQTT订阅-发送MQTT订阅数据包,并在返回之前等待suback。@param client-要使用的客户端对象 @param topicFilter-要订阅的主题过滤器@param消息-要发送的消息 *@返回成功代码 **/
DLLExport int
MQTTSubscribe
(MQTTClient* client, const char* topicFilter, enum QoS, messageHandler);
/MQTT发布-发送MQTT发布数据包,并等待所有ack完成所有QoS@param client-要使用的客户端对象 @param主题-要发布到的主题@param消息-要发送的消息 *@返回成功代码 **/
DLLExport int
MQTTPublish
(MQTTClient* client, const char, MQTTMessage);
/需要死循环调用,这个函数负责库内部一些事件处理。@param client-要使用的客户端对象 @param time-产生的时间(以毫秒为单位)@返回成功代码 **/
DLLExport int
MQTTYield
(MQTTClient* client, int time);
2.2、编写初始化程序¶
初始化程序负责建立socket套接字、连接网络、初始化MQTT对象、连接MQTT服务这几件事
NewNetwork(&myNet);
ConnectNetwork(&myNet, MQTT_IP, MQTT_PORT);
MQTTClientInit(&myclient, &myNet, 1000, sendbuf, sizeof(sendbuf), recvbuf, sizeof(recvbuf));
MQTTPacket_connectData data = MQTTPacket_connectData_initializer;
data.willFlag = 0;
data.MQTTVersion = 3;
data.clientID.cstring = (char *)"RDA8910 MQTT Test";
data.username.cstring = "Air724UG";
data.password.cstring = NULL;
data.keepAliveInterval = 120;
data.cleansession = 1;
int rc = MQTTConnect(&myclient, &data);
2.3、编写订阅消息程序¶
订阅程序分成两块。第一块是向服务器发送订阅消息。
rc = MQTTSubscribe(&myclient, "FX", QOS1, MQTT_CB);
iot_debug_print("[MQTT_Test] Subscribed %d\r\n", rc);
第二块是根据服务器下发的订阅消息,解析报文。
void MQTT_CB(MessageData *md)
{
MQTTMessage *message = md->message;
uint32 datalen = (int)message->payloadlen;
uint8 data[20] = {0};
memcpy(data, (char *)message->payload, datalen);
iot_debug_print("[MQTT_Test] topicName:%s,data:%s,datalen:%d", md->topicName, data, datalen);
}
2.4、编写发布消息程序¶
在任务内循环发布不同的消息,5s发一次。
static void Publish(void *param)
{
char PubData[20] = {0};
uint8 num = 0;
uint8 datalen = 0;
MQTTMessage message = {0};
message.qos = QOS0;
while (1)
{
datalen = sprintf(PubData, "RDA8910 Sent:%d", num);
PubData[datalen] = '\0';
message.payload = PubData;
message.payloadlen = datalen;
MQTTPublish(&myclient, "RDA8910", &message);
num += 1;
iot_os_sleep(5000);
}
}
三、编译并下载程序¶
完整代码中上线会订阅FX
这个主题,紧接着直接向RDA8910
这个主题发送一条RDA8910 Online!
,随后会定时发送其他的消息。
完整代码在这,自取。
/*
* @Author: your name
* @Date: 2020-05-19 14:05:32
* @LastEditTime: 2020-06-02 11:22:21
* @LastEditors: Please set LastEditors
* @Description: In User Settings Edit
* @FilePath: \RDA8910_CSDK\USER\user_main.c
*/
#include "string.h"
//#include "cs_types.h"
#include "osi_log.h"
#include "osi_api.h"
#include "am_openat.h"
#include "am_openat_vat.h"
#include "am_openat_common.h"
#include "iot_debug.h"
#include "iot_uart.h"
#include "iot_os.h"
#include "iot_gpio.h"
#include "iot_pmd.h"
#include "iot_adc.h"
#include "iot_vat.h"
#include "iot_network.h"
#include "iot_socket.h"
//#include "http_client.h"
#include "MQTTClient.h"
HANDLE TestTask_HANDLE = NULL;
uint8 NetWorkCbMessage = 0;
#define MQTT_IP "换成自己的ip" //例如192.168.0.1
#define MQTT_PORT 1883 //自己改端口
Network myNet;
MQTTClient myclient;
unsigned char sendbuf[1024] = {0};
unsigned char recvbuf[1024] = {0};
void MQTT_CB(MessageData *md)
{
MQTTMessage *message = md->message;
uint32 datalen = (int)message->payloadlen;
uint8 data[20] = {0};
memcpy(data, (char *)message->payload, datalen);
iot_debug_print("[MQTT_Test] topicName:%s,data:%s,datalen:%d", md->topicName, data, datalen);
}
static void Publish(void *param)
{
char PubData[20] = {0};
uint8 num = 0;
uint8 datalen = 0;
MQTTMessage message = {0};
message.qos = QOS0;
while (1)
{
datalen = sprintf(PubData, "RDA8910 Sent:%d", num);
PubData[datalen] = '\0';
message.payload = PubData;
message.payloadlen = datalen;
MQTTPublish(&myclient, "RDA8910", &message);
num += 1;
iot_os_sleep(5000);
}
}
static void TestTask(void *param)
{
bool NetLink = FALSE;
while (NetLink == FALSE)
{
T_OPENAT_NETWORK_CONNECT networkparam = {0};
switch (NetWorkCbMessage)
{
case OPENAT_NETWORK_DISCONNECT: //网络断开 表示GPRS网络不可用澹,无法进行数据连接,有可能可以打电话
iot_debug_print("[socket] OPENAT_NETWORK_DISCONNECT");
iot_os_sleep(10000);
break;
case OPENAT_NETWORK_READY: //网络已连接 表示GPRS网络可用,可以进行链路激活
iot_debug_print("[socket] OPENAT_NETWORK_READY");
memcpy(networkparam.apn, "CMNET", strlen("CMNET"));
//建立网络连接,实际为pdp激活流程
iot_network_connect(&networkparam);
iot_os_sleep(500);
break;
case OPENAT_NETWORK_LINKED: //链路已经激活 PDP已经激活,可以通过socket接口建立数据连接
iot_debug_print("[socket] OPENAT_NETWORK_LINKED");
NetLink = TRUE;
break;
}
}
if (NetLink == TRUE)
{
NewNetwork(&myNet);
ConnectNetwork(&myNet, MQTT_IP, MQTT_PORT);
MQTTClientInit(&myclient, &myNet, 1000, sendbuf, sizeof(sendbuf), recvbuf, sizeof(recvbuf));
MQTTPacket_connectData data = MQTTPacket_connectData_initializer;
data.willFlag = 0;
data.MQTTVersion = 3;
data.clientID.cstring = (char *)"RDA8910 MQTT Test";
data.username.cstring = "Air724UG";
data.password.cstring = NULL;
data.keepAliveInterval = 120;
data.cleansession = 1;
int rc = MQTTConnect(&myclient, &data);
iot_debug_print("[MQTT_Test] Connected %d\r\n", rc);
rc = MQTTSubscribe(&myclient, "FX", QOS1, MQTT_CB);
iot_debug_print("[MQTT_Test] Subscribed %d\r\n", rc);
MQTTMessage message = {0};
message.qos = QOS0;
message.payload = "RDA8910 Online!";
message.payloadlen = sizeof("RDA8910 Online!");
MQTTPublish(&myclient, "RDA8910", &message);
iot_os_create_task(Publish, NULL, 2048, 10, OPENAT_OS_CREATE_DEFAULT, "Publish");
while (1)
{
MQTTYield(&myclient, 500);
iot_os_sleep(500);
}
}
iot_os_delete_task(TestTask_HANDLE);
}
static void NetWorkCb(E_OPENAT_NETWORK_STATE state)
{
NetWorkCbMessage = state;
}
//main函数
int appimg_enter(void *param)
{
//系统休眠
iot_os_sleep(10000);
//注册网络状态回调函数
iot_network_set_cb(NetWorkCb);
//创建一个任务
//TestTask_HANDLE =
TestTask_HANDLE = iot_os_create_task(TestTask, NULL, 2048, 10, OPENAT_OS_CREATE_DEFAULT, "TestTask");
return 0;
}
//退出提示
void appimg_exit(void)
{
OSI_LOGI(0, "application image exit");
}
四、分析结果¶
我这里使用MQTT.fx
测试。使用MQTT.fx订阅RDA8910
这个主题,并且随机向FX这个主题发布消息。
可以看到MQTT.fx收到了开发板发布的消息。
开发板也能收到MQTT.fx发布的消息。
通过EMQ的控制台,看到开发板11:01
开始上线。现在时间看右小角是11:55
,运行了接近一个小时。也没出现掉线现象。稳定性也还是可以的。
五、总结¶
除了上面我们使用的函数外。MQTT还具有以下其他的库。这里做简略介绍 > /MQTT Connect-在网络上发送MQTT连接数据包并等待Connack调用此对象之前,必须将nework对象连接到网络端点 @param选项-连接选项@返回成功代码 **/
DLLExport int
MQTTConnectWithResults
(MQTTClient* client, MQTTPacket_connectData* options, MQTTConnackData* data);
/MQTT SetMessageHandler-设置或删除每个主题的消息处理程序@param client-要使用的客户端对象 @param topicFilter-主题过滤器为设置消息处理程序@param messageHandler-指向消息处理程序函数的指针或要删除的NULL *@返回成功代码 **/
DLLExport int
MQTTSetMessageHandler
(MQTTClient* c, const char* topicFilter, messageHandler messageHandler);
/MQTT订阅-发送MQTT订阅数据包,并在返回之前等待suback。@param client-要使用的客户端对象 @param topicFilter-要订阅的主题过滤器@param消息-要发送的消息 @param数据-子包授予的QoS返回@返回成功代码 **/
DLLExport int
MQTTSubscribeWithResults
(MQTTClient* client, const char* topicFilter, enum QoS, messageHandler, MQTTSubackData* data);
/MQTT订阅-发送MQTT取消订阅数据包并等待取消回复,然后再返回。@param client-要使用的客户端对象 @param topicFilter-要取消订阅的主题过滤器@返回成功代码 **/
DLLExport int
MQTTUnsubscribe
(MQTTClient* client, const char* topicFilter);
/MQTT断开连接-发送MQTT断开连接数据包并关闭连接@param client-要使用的客户端对象 *@返回成功代码 **/
DLLExport int
MQTTDisconnect
(MQTTClient* client); > /MQTT已连接@param client-要使用的客户端对象 *@return真值,指示客户端是否连接到服务器 **/DLLExport int
MQTTIsConnected
(MQTTClient* client);
不会下载的点击这里,进去查看我的
RDA8910 CSDK二次开发入门教程
专题第一篇博文1、RDA8910CSDK二次开发:环境搭建
里面讲了怎么下载 这里只是我的学习笔记,拿出来给大家分享,欢迎大家批评指正,本篇教程到此结束