当前位置:   article > 正文

STM32+ESP8266上传温湿度数据至新版OneNET_stm32f103c8t6测量温湿度并上传至onenet

stm32f103c8t6测量温湿度并上传至onenet

本文详细介绍了,STM32连接新版OneNET平台上报温湿度数据的操作与步骤。

使用资源:

STM32F103C8T6、ESP826612F、DHT11温湿度传感器

通讯协议:

MQTT

数据格式:

CJson

一、OneNET平台创建

1.  登录https://open.iot.10086.cn/,注册账号、登录。

2.  点击右上角的“开发者中心”进入产品开发界面

 3. 创建产品,点击左侧“产品中心”然后在主界面右侧“创建产品”

 创建产品信息:

添加产品物模型信息:点击创建产品的“产品开发”

 编辑物模型:点击“设置物模型”,只保留当前温度和当前湿度,删除多余物模型即可

4. 添加设备 

点击左侧“设备管理”,然后点击“添加设备”

添加设备信息: 

 5. 设备信息查看

 二、设备端代码编写

主要代码包含:esp8266.c、mqtt.c、onenet.c、main.c

1. esp8266.c

首先对GPIOA进行初始化,将GPIOA0配置为复位引脚。然后进行ESP8266模块的复位操作。接着调用ESP8266_Clear函数清空ESP8266模块的接收缓存区。之后分别发送AT指令,设置模块为STA模式、连接WiFi、连接到OneNet平台。最后输出ESP8266初始化成功的信息。

  1. //单片机头文件
  2. #include "stm32f10x.h"
  3. //网络设备驱动
  4. #include "esp8266.h"
  5. //硬件驱动
  6. #include "delay.h"
  7. #include "usart.h"
  8. //C库
  9. #include <string.h>
  10. #include <stdio.h>
  11. #define ESP8266_WIFI_INFO "AT+CWJAP=\"xxxxxx\",\"xxxxxxx\"\r\n"
  12. #define ESP8266_ONENET_INFO "AT+CIPSTART=\"TCP\",\"mqtts.heclouds.com\",1883\r\n" //新版OneNET地址
  13. unsigned char esp8266_buf[128];
  14. unsigned short esp8266_cnt = 0, esp8266_cntPre = 0;
  15. //==========================================================
  16. // 函数名称: ESP8266_Clear
  17. //
  18. // 函数功能: 清空缓存
  19. //
  20. // 入口参数: 无
  21. //
  22. // 返回参数: 无
  23. //
  24. // 说明:
  25. //==========================================================
  26. void ESP8266_Clear(void)
  27. {
  28. memset(esp8266_buf, 0, sizeof(esp8266_buf));
  29. esp8266_cnt = 0;
  30. }
  31. //==========================================================
  32. // 函数名称: ESP8266_WaitRecive
  33. //
  34. // 函数功能: 等待接收完成
  35. //
  36. // 入口参数: 无
  37. //
  38. // 返回参数: REV_OK-接收完成 REV_WAIT-接收超时未完成
  39. //
  40. // 说明: 循环调用检测是否接收完成
  41. //==========================================================
  42. _Bool ESP8266_WaitRecive(void)
  43. {
  44. if(esp8266_cnt == 0) //如果接收计数为0 则说明没有处于接收数据中,所以直接跳出,结束函数
  45. return REV_WAIT;
  46. if(esp8266_cnt == esp8266_cntPre) //如果上一次的值和这次相同,则说明接收完毕
  47. {
  48. esp8266_cnt = 0; //清0接收计数
  49. return REV_OK; //返回接收完成标志
  50. }
  51. esp8266_cntPre = esp8266_cnt; //置为相同
  52. return REV_WAIT; //返回接收未完成标志
  53. }
  54. //==========================================================
  55. // 函数名称: ESP8266_SendCmd
  56. //
  57. // 函数功能: 发送命令
  58. //
  59. // 入口参数: cmd:命令
  60. // res:需要检查的返回指令
  61. //
  62. // 返回参数: 0-成功 1-失败
  63. //
  64. // 说明:
  65. //==========================================================
  66. _Bool ESP8266_SendCmd(char *cmd, char *res)
  67. {
  68. unsigned char timeOut = 200;
  69. Usart_SendString(USART2, (unsigned char *)cmd, strlen((const char *)cmd));
  70. while(timeOut--)
  71. {
  72. if(ESP8266_WaitRecive() == REV_OK) //如果收到数据
  73. {
  74. if(strstr((const char *)esp8266_buf, res) != NULL) //如果检索到关键词
  75. {
  76. ESP8266_Clear(); //清空缓存
  77. return 0;
  78. }
  79. }
  80. delay_ms(10);
  81. }
  82. return 1;
  83. }
  84. //==========================================================
  85. // 函数名称: ESP8266_SendData
  86. //
  87. // 函数功能: 发送数据
  88. //
  89. // 入口参数: data:数据
  90. // len:长度
  91. //
  92. // 返回参数: 无
  93. //
  94. // 说明:
  95. //==========================================================
  96. void ESP8266_SendData(unsigned char *data, unsigned short len)
  97. {
  98. char cmdBuf[200];
  99. ESP8266_Clear(); //清空接收缓存
  100. sprintf(cmdBuf, "AT+CIPSEND=%d\r\n", len); //发送命令
  101. if(!ESP8266_SendCmd(cmdBuf, ">")) //收到‘>’时可以发送数据
  102. {
  103. Usart_SendString(USART2, data, len); //发送设备连接请求数据
  104. }
  105. }
  106. //==========================================================
  107. // 函数名称: ESP8266_GetIPD
  108. //
  109. // 函数功能: 获取平台返回的数据
  110. //
  111. // 入口参数: 等待的时间(乘以10ms)
  112. //
  113. // 返回参数: 平台返回的原始数据
  114. //
  115. // 说明: 不同网络设备返回的格式不同,需要去调试
  116. // 如ESP8266的返回格式为 "+IPD,x:yyy" x代表数据长度,yyy是数据内容
  117. //==========================================================
  118. unsigned char *ESP8266_GetIPD(unsigned short timeOut)
  119. {
  120. char *ptrIPD = NULL;
  121. do
  122. {
  123. if(ESP8266_WaitRecive() == REV_OK) //如果接收完成
  124. {
  125. ptrIPD = strstr((char *)esp8266_buf, "IPD,"); //搜索“IPD”头
  126. if(ptrIPD == NULL) //如果没找到,可能是IPD头的延迟,还是需要等待一会,但不会超过设定的时间
  127. {
  128. //UsartPrintf(USART_DEBUG, "\"IPD\" not found\r\n");
  129. }
  130. else
  131. {
  132. ptrIPD = strchr(ptrIPD, ':'); //找到':'
  133. if(ptrIPD != NULL)
  134. {
  135. ptrIPD++;
  136. return (unsigned char *)(ptrIPD);
  137. }
  138. else
  139. return NULL;
  140. }
  141. }
  142. // UsartPrintf(USART_DEBUG,"ESP8266_WaitRecive: %s\n", esp8266_buf); //提示打印信息
  143. delay_ms(5); //延时等待
  144. } while(timeOut--);
  145. return NULL; //超时还未找到,返回空指针
  146. }
  147. //==========================================================
  148. // 函数名称: ESP8266_Init
  149. //
  150. // 函数功能: 初始化ESP8266
  151. //
  152. // 入口参数: 无
  153. //
  154. // 返回参数: 无
  155. //
  156. // 说明:
  157. //==========================================================
  158. void ESP8266_Init(void)
  159. {
  160. GPIO_InitTypeDef GPIO_Initure;
  161. RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
  162. //ESP8266复位引脚
  163. GPIO_Initure.GPIO_Mode = GPIO_Mode_Out_PP;
  164. GPIO_Initure.GPIO_Pin = GPIO_Pin_0; //GPIOA0-复位
  165. GPIO_Initure.GPIO_Speed = GPIO_Speed_50MHz;
  166. GPIO_Init(GPIOA, &GPIO_Initure);
  167. GPIO_WriteBit(GPIOA, GPIO_Pin_0, Bit_RESET);
  168. delay_ms(250);
  169. GPIO_WriteBit(GPIOA, GPIO_Pin_0, Bit_SET);
  170. delay_ms(500);
  171. ESP8266_Clear();
  172. UsartPrintf(USART_DEBUG, "1. AT\r\n");
  173. while(ESP8266_SendCmd("AT\r\n", "OK"))
  174. delay_ms(500);
  175. UsartPrintf(USART_DEBUG, "2. CWMODE\r\n");
  176. while(ESP8266_SendCmd("AT+CWMODE=1\r\n", "OK"))
  177. delay_ms(500);
  178. UsartPrintf(USART_DEBUG, "3. CWJAP\r\n");
  179. while(ESP8266_SendCmd(ESP8266_WIFI_INFO, "GOT IP"))
  180. delay_ms(500);
  181. UsartPrintf(USART_DEBUG, "4. CIPSTART\r\n");
  182. while(ESP8266_SendCmd(ESP8266_ONENET_INFO, "CONNECT"))
  183. delay_ms(500);
  184. UsartPrintf(USART_DEBUG, "5. ESP8266 Init OK\r\n");
  185. }
  186. //==========================================================
  187. // 函数名称: USART2_IRQHandler
  188. //
  189. // 函数功能: 串口2收发中断
  190. //
  191. // 入口参数: 无
  192. //
  193. // 返回参数: 无
  194. //
  195. // 说明:
  196. //==========================================================
  197. void USART2_IRQHandler(void)
  198. {
  199. if(USART_GetITStatus(USART2, USART_IT_RXNE) != RESET) //接收中断
  200. {
  201. if(esp8266_cnt >= sizeof(esp8266_buf)) esp8266_cnt = 0; //防止串口被刷爆
  202. esp8266_buf[esp8266_cnt++] = USART2->DR;
  203. USART_ClearFlag(USART2, USART_FLAG_RXNE);
  204. }
  205. }

2. mqtt.c

MQTT程序编写流程如下:

引入MQTT库和所需的头文件。

连接WiFi网络。

连接MQTT服务器。

订阅需要接收的主题。

在程序中实现消息的发布和接收。

关闭MQTT连接。

主要代码如下:

  1. //协议头文件
  2. #include "MqttKit.h"
  3. //C库
  4. #include <string.h>
  5. #include "usart.h"
  6. #define CMD_TOPIC_PREFIX "$creq"
  7. //==========================================================
  8. // 函数名称: EDP_NewBuffer
  9. //
  10. // 函数功能: 申请内存
  11. //
  12. // 入口参数: edpPacket:包结构体
  13. // size:大小
  14. //
  15. // 返回参数: 无
  16. //
  17. // 说明: 1.可使用动态分配来分配内存
  18. // 2.可使用局部或全局数组来指定内存
  19. //==========================================================
  20. void MQTT_NewBuffer(MQTT_PACKET_STRUCTURE *mqttPacket, uint32 size)
  21. {
  22. uint32 i = 0;
  23. if(mqttPacket->_data == NULL)
  24. {
  25. mqttPacket->_memFlag = MEM_FLAG_ALLOC;
  26. mqttPacket->_data = (uint8 *)MQTT_MallocBuffer(size);
  27. if(mqttPacket->_data != NULL)
  28. {
  29. mqttPacket->_len = 0;
  30. mqttPacket->_size = size;
  31. for(; i < mqttPacket->_size; i++)
  32. mqttPacket->_data[i] = 0;
  33. }
  34. }
  35. else
  36. {
  37. mqttPacket->_memFlag = MEM_FLAG_STATIC;
  38. for(; i < mqttPacket->_size; i++)
  39. mqttPacket->_data[i] = 0;
  40. mqttPacket->_len = 0;
  41. if(mqttPacket->_size < size)
  42. mqttPacket->_data = NULL;
  43. }
  44. }
  45. //==========================================================
  46. // 函数名称: MQTT_DeleteBuffer
  47. //
  48. // 函数功能: 释放数据内存
  49. //
  50. // 入口参数: edpPacket:包结构体
  51. //
  52. // 返回参数: 无
  53. //
  54. // 说明:
  55. //==========================================================
  56. void MQTT_DeleteBuffer(MQTT_PACKET_STRUCTURE *mqttPacket)
  57. {
  58. if(mqttPacket->_memFlag == MEM_FLAG_ALLOC)
  59. MQTT_FreeBuffer(mqttPacket->_data);
  60. mqttPacket->_data = NULL;
  61. mqttPacket->_len = 0;
  62. mqttPacket->_size = 0;
  63. mqttPacket->_memFlag = MEM_FLAG_NULL;
  64. }
  65. int32 MQTT_DumpLength(size_t len, uint8 *buf)
  66. {
  67. int32 i = 0;
  68. for(i = 1; i <= 4; ++i)
  69. {
  70. *buf = len % 128;
  71. len >>= 7;
  72. if(len > 0)
  73. {
  74. *buf |= 128;
  75. ++buf;
  76. }
  77. else
  78. {
  79. return i;
  80. }
  81. }
  82. return -1;
  83. }
  84. int32 MQTT_ReadLength(const uint8 *stream, int32 size, uint32 *len)
  85. {
  86. int32 i;
  87. const uint8 *in = stream;
  88. uint32 multiplier = 1;
  89. *len = 0;
  90. for(i = 0; i < size; ++i)
  91. {
  92. *len += (in[i] & 0x7f) * multiplier;
  93. if(!(in[i] & 0x80))
  94. {
  95. return i + 1;
  96. }
  97. multiplier <<= 7;
  98. if(multiplier >= 2097152) //128 * *128 * *128
  99. {
  100. return -2; // error, out of range
  101. }
  102. }
  103. return -1; // not complete
  104. }
  105. //==========================================================
  106. // 函数名称: MQTT_UnPacketRecv
  107. //
  108. // 函数功能: MQTT数据接收类型判断
  109. //
  110. // 入口参数: dataPtr:接收的数据指针
  111. //
  112. // 返回参数: 0-成功 其他-失败原因
  113. //
  114. // 说明:
  115. //==========================================================
  116. uint8 MQTT_UnPacketRecv(uint8 *dataPtr)
  117. {
  118. uint8 status = 255;
  119. uint8 type = dataPtr[0] >> 4; //类型检查
  120. if(type < 1 || type > 14)
  121. return status;
  122. if(type == MQTT_PKT_PUBLISH)
  123. {
  124. uint8 *msgPtr;
  125. uint32 remain_len = 0;
  126. msgPtr = dataPtr + MQTT_ReadLength(dataPtr + 1, 4, &remain_len) + 1;
  127. if(remain_len < 2 || dataPtr[0] & 0x01) //retain
  128. return 255;
  129. if(remain_len < ((uint16)msgPtr[0] << 8 | msgPtr[1]) + 2)
  130. return 255;
  131. if(strstr((int8 *)msgPtr + 2, CMD_TOPIC_PREFIX) != NULL) //如果是命令下发
  132. status = MQTT_PKT_CMD;
  133. else
  134. status = MQTT_PKT_PUBLISH;
  135. }
  136. else
  137. status = type;
  138. return status;
  139. }
  140. //==========================================================
  141. // 函数名称: MQTT_PacketConnect
  142. //
  143. // 函数功能: 连接消息组包
  144. //
  145. // 入口参数: user:用户名:产品ID
  146. // password:密码:鉴权信息或apikey
  147. // devid:设备ID
  148. // cTime:连接保持时间
  149. // clean_session:离线消息清除标志
  150. // qos:重发标志
  151. // will_topic:异常离线topic
  152. // will_msg:异常离线消息
  153. // will_retain:消息推送标志
  154. // mqttPacket:包指针
  155. //
  156. // 返回参数: 0-成功 其他-失败
  157. //
  158. // 说明:
  159. //==========================================================
  160. uint8 MQTT_PacketConnect(const int8 *user, const int8 *password, const int8 *devid,
  161. uint16 cTime, uint1 clean_session, uint1 qos,
  162. const int8 *will_topic, const int8 *will_msg, int32 will_retain,
  163. MQTT_PACKET_STRUCTURE *mqttPacket)
  164. {
  165. uint8 flags = 0;
  166. uint8 will_topic_len = 0;
  167. uint16 total_len = 15;
  168. int16 len = 0, devid_len = strlen(devid);
  169. int16 user_len = strlen(user),password_len = strlen(password);
  170. if(!devid)
  171. return 1;
  172. total_len += devid_len + 2;
  173. //断线后,是否清理离线消息:1-清理 0-不清理--------------------------------------------
  174. if(clean_session)
  175. {
  176. flags |= MQTT_CONNECT_CLEAN_SESSION;
  177. }
  178. //异常掉线情况下,服务器发布的topic------------------------------------------------------
  179. if(will_topic)
  180. {
  181. flags |= MQTT_CONNECT_WILL_FLAG;
  182. will_topic_len = strlen(will_topic);
  183. total_len += 4 + will_topic_len + strlen(will_msg);
  184. }
  185. //qos级别--主要用于PUBLISH(发布态)消息的,保证消息传递的次数-----------------------------
  186. switch((unsigned char)qos)
  187. {
  188. case MQTT_QOS_LEVEL0:
  189. flags |= MQTT_CONNECT_WILL_QOS0; //最多一次
  190. break;
  191. case MQTT_QOS_LEVEL1:
  192. flags |= (MQTT_CONNECT_WILL_FLAG | MQTT_CONNECT_WILL_QOS1); //最少一次
  193. break;
  194. case MQTT_QOS_LEVEL2:
  195. flags |= (MQTT_CONNECT_WILL_FLAG | MQTT_CONNECT_WILL_QOS2); //只有一次
  196. break;
  197. default:
  198. return 2;
  199. }
  200. //主要用于PUBLISH(发布态)的消息,表示服务器要保留这次推送的信息,如果有新的订阅者出现,就把这消息推送给它。如果不设那么推送至当前订阅的就释放了
  201. if(will_retain)
  202. {
  203. flags |= (MQTT_CONNECT_WILL_FLAG | MQTT_CONNECT_WILL_RETAIN);
  204. }
  205. //账号为空 密码为空---------------------------------------------------------------------
  206. if(!user || !password)
  207. {
  208. return 3;
  209. }
  210. flags |= MQTT_CONNECT_USER_NAME | MQTT_CONNECT_PASSORD;
  211. total_len += strlen(user) + strlen(password) + 4;
  212. //分配内存-----------------------------------------------------------------------------
  213. MQTT_NewBuffer(mqttPacket, total_len);
  214. if(mqttPacket->_data == NULL)
  215. return 4;
  216. memset(mqttPacket->_data, 0, total_len);
  217. /*************************************固定头部***********************************************/
  218. //固定头部----------------------连接请求类型---------------------------------------------
  219. mqttPacket->_data[mqttPacket->_len++] = MQTT_PKT_CONNECT << 4;
  220. //固定头部----------------------剩余长度值-----------------------------------------------
  221. len = MQTT_DumpLength(total_len - 5, mqttPacket->_data + mqttPacket->_len);
  222. if(len < 0)
  223. {
  224. MQTT_DeleteBuffer(mqttPacket);
  225. return 5;
  226. }
  227. else
  228. mqttPacket->_len += len;
  229. /*************************************可变头部***********************************************/
  230. //可变头部----------------------协议名长度 和 协议名--------------------------------------
  231. mqttPacket->_data[mqttPacket->_len++] = 0;
  232. mqttPacket->_data[mqttPacket->_len++] = 4;
  233. mqttPacket->_data[mqttPacket->_len++] = 'M';
  234. mqttPacket->_data[mqttPacket->_len++] = 'Q';
  235. mqttPacket->_data[mqttPacket->_len++] = 'T';
  236. mqttPacket->_data[mqttPacket->_len++] = 'T';
  237. //可变头部----------------------protocol level 4-----------------------------------------
  238. mqttPacket->_data[mqttPacket->_len++] = 4;
  239. //可变头部----------------------连接标志(该函数开头处理的数据)-----------------------------
  240. mqttPacket->_data[mqttPacket->_len++] = flags;
  241. //可变头部----------------------保持连接的时间(秒)----------------------------------------
  242. mqttPacket->_data[mqttPacket->_len++] = MOSQ_MSB(cTime);
  243. mqttPacket->_data[mqttPacket->_len++] = MOSQ_LSB(cTime);
  244. /*************************************消息体************************************************/
  245. //消息体----------------------------devid长度、devid-------------------------------------
  246. mqttPacket->_data[mqttPacket->_len++] = MOSQ_MSB(devid_len);
  247. mqttPacket->_data[mqttPacket->_len++] = MOSQ_LSB(devid_len);
  248. strncat((int8 *)mqttPacket->_data + mqttPacket->_len, devid, devid_len);
  249. mqttPacket->_len += devid_len;
  250. // UsartPrintf(USART_DEBUG,"111: %s\r\n", PROID);
  251. // UsartPrintf(USART_DEBUG,"mqttPacket._data content: %s\n", mqttPacket->_data);
  252. //消息体----------------------------will_flag 和 will_msg---------------------------------
  253. if(flags & MQTT_CONNECT_WILL_FLAG)
  254. {
  255. unsigned short mLen = 0;
  256. if(!will_msg)
  257. will_msg = "";
  258. mLen = strlen(will_topic);
  259. mqttPacket->_data[mqttPacket->_len++] = MOSQ_MSB(mLen);
  260. mqttPacket->_data[mqttPacket->_len++] = MOSQ_LSB(mLen);
  261. strncat((int8 *)mqttPacket->_data + mqttPacket->_len, will_topic, mLen);
  262. mqttPacket->_len += mLen;
  263. mLen = strlen(will_msg);
  264. mqttPacket->_data[mqttPacket->_len++] = MOSQ_MSB(mLen);
  265. mqttPacket->_data[mqttPacket->_len++] = MOSQ_LSB(mLen);
  266. strncat((int8 *)mqttPacket->_data + mqttPacket->_len, will_msg, mLen);
  267. mqttPacket->_len += mLen;
  268. }
  269. //消息体----------------------------use---------------------------------------------------
  270. if(flags & MQTT_CONNECT_USER_NAME)
  271. {
  272. unsigned short user_len = strlen(user);
  273. mqttPacket->_data[mqttPacket->_len++] = MOSQ_MSB(user_len);
  274. mqttPacket->_data[mqttPacket->_len++] = MOSQ_LSB(user_len);
  275. strncat((int8 *)mqttPacket->_data + mqttPacket->_len, user, user_len);
  276. mqttPacket->_len += user_len;
  277. }
  278. //消息体----------------------------password----------------------------------------------
  279. if(flags & MQTT_CONNECT_PASSORD)
  280. {
  281. unsigned short psw_len = strlen(password);
  282. mqttPacket->_data[mqttPacket->_len++] = MOSQ_MSB(psw_len);
  283. mqttPacket->_data[mqttPacket->_len++] = MOSQ_LSB(psw_len);
  284. strncat((int8 *)mqttPacket->_data + mqttPacket->_len, password, psw_len);
  285. mqttPacket->_len += psw_len;
  286. }
  287. return 0;
  288. }
  289. //==========================================================
  290. // 函数名称: MQTT_UnPacketConnectAck
  291. //
  292. // 函数功能: 连接消息解包
  293. //
  294. // 入口参数: rev_data:接收的数据
  295. //
  296. // 返回参数: 1、255-失败 其他-平台的返回码
  297. //
  298. // 说明:
  299. //==========================================================
  300. uint8 MQTT_UnPacketConnectAck(uint8 *rev_data)
  301. {
  302. if(rev_data[1] != 2)
  303. return 1;
  304. if(rev_data[2] == 0 || rev_data[2] == 1)
  305. return rev_data[3];
  306. else
  307. return 255;
  308. }

3. onenet.c

函数OneNet_DevLink用于与OneNet平台建立连接;函数MqttOnenet_Savedata将温湿度数据转换为JSON数据形式;函数OneNet_SendData用于上传数据到平台。该代码使用了MQTT协议进行通信。

  1. #define PROID "xxxx" //产品ID
  2. #define AUTH_INFO "xxxxx" //鉴权信息
  3. #define DEVID "xxxx" //设备名称
  4. extern unsigned char esp8266_buf[128];
  5. //==========================================================
  6. // 函数名称: OneNet_DevLink
  7. //
  8. // 函数功能: 与onenet创建连接
  9. //
  10. // 入口参数: 无
  11. //
  12. // 返回参数: 1-成功 0-失败
  13. //
  14. // 说明: 与onenet平台建立连接
  15. //==========================================================
  16. _Bool OneNet_DevLink(void)
  17. {
  18. MQTT_PACKET_STRUCTURE mqttPacket = {NULL, 0, 0, 0}; //协议包
  19. unsigned char *dataPtr;
  20. _Bool status = 1;
  21. UsartPrintf(USART_DEBUG, "OneNet_DevLink\r\n"
  22. "PROID: %s, AUIF: %s, DEVID:%s\r\n"
  23. , PROID, AUTH_INFO, DEVID);
  24. if(MQTT_PacketConnect(PROID, AUTH_INFO, DEVID, 256, 1, MQTT_QOS_LEVEL0, NULL, NULL, 0, &mqttPacket) == 0) //修改clean_session=1
  25. {
  26. ESP8266_SendData(mqttPacket._data, mqttPacket._len); //上传平台
  27. dataPtr = ESP8266_GetIPD(250); //等待平台响应
  28. if(dataPtr != NULL)
  29. {
  30. if(MQTT_UnPacketRecv(dataPtr) == MQTT_PKT_CONNACK)
  31. {
  32. switch(MQTT_UnPacketConnectAck(dataPtr))
  33. {
  34. case 0:UsartPrintf(USART_DEBUG, "Tips: 连接成功\r\n");status = 0;break;
  35. case 1:UsartPrintf(USART_DEBUG, "WARN: 连接失败:协议错误\r\n");break;
  36. case 2:UsartPrintf(USART_DEBUG, "WARN: 连接失败:非法的clientid\r\n");break;
  37. case 3:UsartPrintf(USART_DEBUG, "WARN: 连接失败:服务器失败\r\n");break;
  38. case 4:UsartPrintf(USART_DEBUG, "WARN: 连接失败:用户名或密码错误\r\n");break;
  39. case 5:UsartPrintf(USART_DEBUG, "WARN: 连接失败:非法链接(比如token非法)\r\n");break;
  40. default:UsartPrintf(USART_DEBUG, "ERR: 连接失败:未知错误\r\n");break;
  41. }
  42. }
  43. }
  44. MQTT_DeleteBuffer(&mqttPacket); //删包
  45. }
  46. else
  47. UsartPrintf(USART_DEBUG, "WARN: MQTT_PacketConnect Failed\r\n");
  48. return status;
  49. }
  50. //访问ONENET需要提交JSON数据,就获取到的温湿度转换为JSON数据形式
  51. unsigned char MqttOnenet_Savedata(char *t_payload)
  52. {
  53. extern u8 temperature;
  54. extern u8 humidity;
  55. // u8 temperature = 12; //调试使用
  56. // u8 humidity = 12;
  57. char json[]="{\"id\":\"0\",\"version\":\"1.0\",\"params\":{\"humidity_value\":{\"value\":%d},\"temp_value\":{\"value\":%d}}}"; //更换了JSON数据形式,符合OneNET需求
  58. char t_json[200];
  59. unsigned short json_len;
  60. sprintf(t_json, json, temperature, humidity);
  61. json_len = strlen(t_json)/sizeof(char);
  62. memcpy(t_payload, t_json, json_len);
  63. return json_len;
  64. }
  65. //==========================================================
  66. // 函数名称: OneNet_SendData
  67. //
  68. // 函数功能: 上传数据到平台
  69. //
  70. // 入口参数: type:发送数据的格式
  71. //
  72. // 返回参数: 无
  73. //
  74. // 说明:
  75. //==========================================================
  76. void OneNet_SendData(void)
  77. {
  78. MQTT_PACKET_STRUCTURE mqttPacket = {NULL, 0, 0, 0}; //协议包
  79. char buf[128];
  80. short body_len = 0, i = 0;
  81. UsartPrintf(USART_DEBUG, "Tips: OneNet_SendData-MQTT\r\n");
  82. memset(buf, 0, sizeof(buf)); //清空buff
  83. body_len=MqttOnenet_Savedata(buf);
  84. if(body_len)
  85. {
  86. if(MQTT_PacketSaveData(DEVID, body_len, NULL, 5, &mqttPacket) == 0) //封包
  87. {
  88. for(; i < body_len; i++)
  89. mqttPacket._data[mqttPacket._len++] = buf[i];
  90. ESP8266_SendData(mqttPacket._data, mqttPacket._len); //上传数据到平台
  91. UsartPrintf(USART_DEBUG, "Send %d Bytes\r\n", mqttPacket._len);
  92. MQTT_DeleteBuffer(&mqttPacket); //删包
  93. }
  94. else
  95. {
  96. UsartPrintf(USART_DEBUG, "WARN: EDP_NewBuffer Failed\r\n");
  97. }
  98. }
  99. }
  100. //==========================================================
  101. // 函数名称: OneNet_RevPro
  102. //
  103. // 函数功能: 平台返回数据检测
  104. //
  105. // 入口参数: dataPtr:平台返回的数据
  106. //
  107. // 返回参数: 无
  108. //
  109. // 说明:
  110. //==========================================================
  111. void OneNet_RevPro(unsigned char *cmd)
  112. {
  113. MQTT_PACKET_STRUCTURE mqttPacket = {NULL, 0, 0, 0}; //协议包
  114. char *req_payload = NULL;
  115. char *cmdid_topic = NULL;
  116. unsigned short req_len = 0;
  117. unsigned char type = 0;
  118. short result = 0;
  119. char *dataPtr = NULL;
  120. char numBuf[10];
  121. int num = 0;
  122. type = MQTT_UnPacketRecv(cmd);
  123. switch(type)
  124. {
  125. case MQTT_PKT_CMD: //命令下发
  126. result = MQTT_UnPacketCmd(cmd, &cmdid_topic, &req_payload, &req_len); //解出topic和消息体
  127. if(result == 0)
  128. {
  129. UsartPrintf(USART_DEBUG, "cmdid: %s, req: %s, req_len: %d\r\n", cmdid_topic, req_payload, req_len);
  130. if(MQTT_PacketCmdResp(cmdid_topic, req_payload, &mqttPacket) == 0) //命令回复组包
  131. {
  132. UsartPrintf(USART_DEBUG, "Tips: Send CmdResp\r\n");
  133. ESP8266_SendData(mqttPacket._data, mqttPacket._len); //回复命令
  134. MQTT_DeleteBuffer(&mqttPacket); //删包
  135. }
  136. }
  137. break;
  138. case MQTT_PKT_PUBACK: //发送Publish消息,平台回复的Ack
  139. if(MQTT_UnPacketPublishAck(cmd) == 0)
  140. UsartPrintf(USART_DEBUG, "Tips: MQTT Publish Send OK\r\n");
  141. break;
  142. default:
  143. result = -1;
  144. break;
  145. }
  146. ESP8266_Clear(); //清空缓存
  147. if(result == -1)
  148. return;
  149. dataPtr = strchr(req_payload, '}'); //搜索'}'
  150. if(dataPtr != NULL && result != -1) //如果找到了
  151. {
  152. dataPtr++;
  153. while(*dataPtr >= '0' && *dataPtr <= '9') //判断是否是下发的命令控制数据
  154. {
  155. numBuf[num++] = *dataPtr++;
  156. }
  157. num = atoi((const char *)numBuf); //转为数值形式
  158. }
  159. if(type == MQTT_PKT_CMD || type == MQTT_PKT_PUBLISH)
  160. {
  161. MQTT_FreeBuffer(cmdid_topic);
  162. MQTT_FreeBuffer(req_payload);
  163. }
  164. }

鉴权信息和获取方式参考:OneNET - 中国移动物联网开放平台 (10086.cn)icon-default.png?t=N7T8https://open.iot.10086.cn/doc/v5/fuse/detail/1464

 4. main.c

        主要包括硬件初始化、接入OneNET平台、读取温湿度、发送数据等功能。其中,timeCount变量用于控制发送数据的时间间隔,dataPtr变量用于存储ESP8266模块接收到的数据。如果timeCount计数达到50,则执行发送数据的操作,将温湿度数据上传至OneNET平台并清空ESP8266模块缓存。同时,使用ESP8266_GetIPD函数获取ESP8266模块接收到的数据,并使用OneNet_RevPro函数处理该数据。

  1. int main(void)
  2. {
  3. unsigned short timeCount = 0; //发送间隔变量
  4. unsigned char *dataPtr = NULL;
  5. delay_init(); //延时函数初始化
  6. Usart1_Init(115200); //串口1初始化为115200
  7. Usart2_Init(115200); //串口2初始化为115200
  8. LED_Init(); //初始化与LED连接的硬件接口
  9. UsartPrintf(USART_DEBUG, " Hardware init OK\r\n");
  10. ESP8266_Init();
  11. while(OneNet_DevLink()) //接入OneNET
  12. delay_ms(500);
  13. while(1)
  14. {
  15. DHT11_Read_Data(&temperature,&humidity); //读取温湿度值
  16. delay_ms(100);
  17. if(++timeCount >= 50) //发送间隔5s
  18. {
  19. UsartPrintf(USART_DEBUG, "温度:%d\r\n",temperature);
  20. UsartPrintf(USART_DEBUG, "湿度:%d\r\n",humidity);
  21. UsartPrintf(USART_DEBUG, "OneNet_SendData\r\n");
  22. // data_len=MqttOnenet_Savedata(send_jason,temperature, humidity);
  23. OneNet_SendData(); //发送数据
  24. timeCount = 0;
  25. ESP8266_Clear();
  26. }
  27. dataPtr = ESP8266_GetIPD(0);
  28. if(dataPtr != NULL)
  29. OneNet_RevPro(dataPtr);
  30. }
  31. }

三、上报效果

 不知为何,源码始终上传不成功,需要源码请在评论区留言。

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/Gausst松鼠会/article/detail/424636
推荐阅读
相关标签
  

闽ICP备14008679号