Tuoreniot.c 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487
  1. #include "stm32f10x.h"
  2. #include <stdio.h>
  3. #include <stdarg.h>
  4. #include <string.h>
  5. #include <stdlib.h>
  6. #include "Tuoreniot.h"
  7. #include "mbedtls_util.h"
  8. #include "Log_Module.h"
  9. #include "At_Module.h"
  10. #include "Common_Util.h"
  11. #include "ec800m.h"
  12. #include "cJSON.h"
  13. #include "FlashSetup.h"
  14. #include "Timer_Module.h"
  15. #include "CoAP_Util.h"
  16. // 泵号
  17. extern uint8_t mcu_id[8];
  18. // 注册时用的密钥
  19. char regist_key[17]="tuorenzhinenghua";
  20. // 注册数据
  21. uint8_t regist_data[1024];
  22. uint16_t regist_data_length;
  23. #if PROD_ENV
  24. // 生产环境
  25. char * REGIST_SERVER = "iot.tuoren.com" ; //注册服务器地址
  26. uint16_t REGIST_PORT = 8888; //端口号
  27. #endif
  28. #if TEST_ENV
  29. char * REGIST_SERVER = "18017zi0ec26.vicp.fun" ; //注册服务器地址
  30. uint16_t REGIST_PORT = 58562; //端口号
  31. #endif
  32. // 从flash中读取注册响应
  33. static void ReadRegistResponseFromFlash(void);
  34. // 将注册响应写入到flash中
  35. static void WriteRegistResponseToFlash(void);
  36. // 注册结果
  37. struct TUORENIOT_RegistResponseStruct regist_response = {
  38. .version = 1
  39. };
  40. // 初始化clientID
  41. static char ClientID[60];
  42. static void InitClientId(void)
  43. {
  44. // 设置clientid
  45. memset(ClientID, 0, sizeof(ClientID));
  46. strcat(ClientID, regist_response.productId);
  47. strcat(ClientID, "&");
  48. strcat(ClientID, regist_response.deviceId);
  49. }
  50. /*************************************** 注册信息打包函数 ***************************************/
  51. void TUORENIOT_PackRegistRequest(uint8_t * pack_data, uint16_t * pack_data_length, struct TUORENIOT_RegistRequestStruct * registParams)
  52. {
  53. Log_Printf_Debug("mcu_id: %s\r\n", mcu_id);
  54. byteToHexStr(mcu_id, registParams->deviceId, sizeof(mcu_id));
  55. sprintf((char *)(pack_data + 2), "{\"version\":\"%d\",\"networkType\":\"%d\",\"networkProtocol\":\"%d\",\"productId\":\"%s\",\"deviceId\":\"%s\",\"userId\":\"%s\"}",
  56. registParams->version, registParams->networkType, registParams->networkProtocol, registParams->productId, registParams->deviceId, registParams->userId); // 拼接AT指令
  57. Log_Printf_Debug("%s\r\n", (char *)(pack_data + 2));
  58. uint16_t datalen = utils_aes128_ECB_base64_enc_with_length(regist_key, (pack_data + 2));
  59. pack_data[0] = datalen>>8;//头2个字节赋值数据长度
  60. pack_data[1] = datalen;
  61. * pack_data_length = datalen + 2;
  62. }
  63. /*************************************** 注册数据解密函数 ***************************************/
  64. static int wifi_count=0; //WiFi信息计数变量
  65. uint8_t TUORENIOT_AnalysisRegistData(uint8_t * regist_data, uint16_t regist_data_length)
  66. {
  67. Log_Printf_Debug("regist_data: \r\n");
  68. Log_SendHex(regist_data, regist_data_length);
  69. Log_Printf_Debug("\r\n");
  70. uint8_t result;
  71. utils_aes128_ECB_base64_dec(regist_key,regist_data,regist_data_length);
  72. Log_Printf_Debug("解密数据(%d):%s\r\n", regist_data_length, (char *)regist_data);
  73. cJSON *json = cJSON_Parse((char *)regist_data);
  74. if(!json)
  75. {
  76. Log_Printf_Debug("json parse error,%s\r\n",cJSON_GetErrorPtr());
  77. result = 0;
  78. }
  79. else if(cJSON_GetObjectItem(json, "code")->valueint == 200)
  80. {
  81. regist_response.code = cJSON_GetObjectItem(json, "code")->valueint;
  82. regist_response.version = cJSON_GetObjectItem(json, "version")->valueint;
  83. regist_response.platform = cJSON_GetObjectItem(json, "platform")->valueint;
  84. regist_response.networkType = cJSON_GetObjectItem(json, "networkType")->valueint;
  85. regist_response.networkProtocol = cJSON_GetObjectItem(json, "networkProtocol")->valueint;
  86. strcpy(regist_response.productId, cJSON_GetObjectItem(json, "productId")->valuestring);
  87. strcpy(regist_response.deviceId, cJSON_GetObjectItem(json, "deviceId")->valuestring);
  88. strcpy(regist_response.userId, cJSON_GetObjectItem(json, "userId")->valuestring);
  89. strcpy(regist_response.deviceSecret, cJSON_GetObjectItem(json, "deviceSecret")->valuestring);
  90. strcpy(regist_response.address, cJSON_GetObjectItem(json, "address")->valuestring);
  91. regist_response.port = cJSON_GetObjectItem(json, "port")->valueint;
  92. // 判断是否解析WiFi信息
  93. if(regist_response.networkType == 3)
  94. {
  95. Log_Printf_Debug("---------------------解析WiFi信息----------------------\r\n");
  96. cJSON *wifi_Array = cJSON_GetObjectItem(json, "wifi");
  97. regist_response.wifiNumber = cJSON_GetArraySize(wifi_Array);
  98. Log_Printf_Debug("wifiNumber=%d\r\n",regist_response.wifiNumber);
  99. if (cJSON_IsArray(wifi_Array))
  100. {
  101. cJSON *wifiElement = NULL;
  102. cJSON_ArrayForEach(wifiElement, wifi_Array)
  103. {
  104. // 提取wifiName
  105. cJSON *wifiName = cJSON_GetObjectItem(wifiElement, "wifiName");
  106. strcpy(regist_response.wifi[wifi_count].wifiName,wifiName->valuestring);
  107. // 提取wifiPassword
  108. cJSON *wifiPassword = cJSON_GetObjectItem(wifiElement, "wifiPassword");
  109. strcpy(regist_response.wifi[wifi_count].wifiPassword,wifiPassword->valuestring);
  110. //解析WiFi信息计数
  111. wifi_count++;
  112. }
  113. }
  114. //校验解析的WiFi信息数目是否正确
  115. if(wifi_count != regist_response.wifiNumber)
  116. {
  117. Log_Printf_Debug("wifi_count=%d\r\n",wifi_count);
  118. Log_Printf_Debug("解析wifi信息ERROR!!!\r\n");
  119. result = 0;
  120. cJSON_Delete(json);
  121. return result;
  122. }
  123. //打印解析的WiFi信息
  124. Log_Printf_Debug("解析出%d组wifi信息如下:\r\n",wifi_count);
  125. for(int i=0;i<wifi_count;i++)
  126. {
  127. Log_Printf_Debug("wifi[%d]->wifiName: %s\r\n", i,regist_response.wifi[i].wifiName);
  128. Log_Printf_Debug("wifi[%d]->wifiPassword: %s\r\n", i,regist_response.wifi[i].wifiPassword);
  129. }
  130. Log_Printf_Debug("-------------------------结束--------------------------\r\n");
  131. }
  132. result = 1;
  133. WriteRegistResponseToFlash();
  134. }
  135. else
  136. {
  137. result = 0;
  138. }
  139. cJSON_Delete(json);
  140. return result;
  141. }
  142. static uint32_t SeqOffset = 0;
  143. static char Random[20];
  144. static char Token[50];
  145. // 判断是否认证
  146. uint8_t TUORENIOT_IsAuthentication(void)
  147. {
  148. if(strlen(Token) > 0) // 已认证
  149. {
  150. return 1;
  151. }
  152. else
  153. {
  154. return 0;
  155. }
  156. }
  157. // 拼接认证报文
  158. static char StrForSignature[100];
  159. static char sign[50];
  160. static uint8_t payload[256];
  161. void TUORENIOT_PackAuthMessage(uint8_t * coap_message, uint16_t * coap_message_length)
  162. {
  163. // Log_Printf_Debug("<<STEP: 拼接认证报文\r\n");
  164. // TUORENIOT_PrintfRegistResponseStruct();
  165. SeqOffset = 888;
  166. // 初始化clientId
  167. InitClientId();
  168. // CoAP协议
  169. CoAP_Init(coap_message);
  170. CoAP_Set_T(0);
  171. CoAP_Set_Code(2);
  172. CoAP_Set_MessageID(SeqOffset);
  173. // option
  174. // host
  175. CoAP_Set_Option_Str(3, regist_response.address);
  176. // port
  177. CoAP_Set_Option_Short(7, regist_response.port);
  178. // path
  179. CoAP_Set_Option_Str(11, "auth");
  180. // content-format
  181. CoAP_Set_Option_Short(12, 50);
  182. // accept
  183. CoAP_Set_Option_Short(17, 50);
  184. // 待签名数据
  185. // 签名相关
  186. memset(StrForSignature, 0, sizeof(StrForSignature));
  187. sprintf(StrForSignature, "clientId%sdeviceName%sproductKey%sseq%d", ClientID, regist_response.deviceId,regist_response.productId, SeqOffset);
  188. // Log_Printf_Debug("认证报文StrForSignature:\r\n%s\r\n", StrForSignature);
  189. // 签名
  190. memset(sign, 0, sizeof(sign));
  191. utils_hmac_sha1_str(StrForSignature, strlen((char * )StrForSignature), sign, regist_response.deviceSecret, strlen(regist_response.deviceSecret));
  192. // Log_Printf_Debug("签名sign:\r\n%s\r\n", sign);
  193. // payload
  194. memset(payload, 0, sizeof(payload));
  195. sprintf((char * )payload, "{\"clientId\":\"%s\",\"signmethod\":\"hmacsha1\",\"sign\":\"%s\",\"productKey\":\"%s\",\"deviceName\":\"%s\",\"seq\":\"%d\"}", ClientID, sign, regist_response.productId, regist_response.deviceId, SeqOffset);
  196. // 设置payload
  197. CoAP_Set_Payload_Str((char *)payload);
  198. // Log_Printf_Debug("payload:\r\n%s\r\n", payload);
  199. // 生成报文
  200. * coap_message_length = CoAP_Get_Length();
  201. // Log_Printf_Debug("STEP: 拼接认证报文>>\r\n");
  202. }
  203. // 拼接数据报文
  204. // aes cbc
  205. static uint8_t aes_key[16] = {0};
  206. static char iv[] = "543yhjy97ae7fyfg";
  207. static char out[80];
  208. static char sha256_source[100];
  209. static char Rand2089[10] = {'\0'};
  210. static uint8_t Opt2089[20];
  211. void TUORENIOT_PackDataMessage(uint8_t * data, uint8_t data_length, uint8_t * coap_message, uint16_t * coap_message_length)
  212. {
  213. // Log_Printf_Debug("<<STEP: 拼接数据报文\r\n");
  214. // 初始化clientId
  215. InitClientId();
  216. // Log_Printf_Debug("SeqOffset:%d\r\n", SeqOffset);
  217. // 获取密钥
  218. memset(sha256_source, 0, sizeof(sha256_source));
  219. snprintf(sha256_source, sizeof(sha256_source), "%s,%s" ,regist_response.deviceSecret ,Random);
  220. // Log_Printf_Debug("sha256_source:\r\n%s\r\n", sha256_source);
  221. memset(out, 0, sizeof(out));
  222. utils_sha256(sha256_source, strlen(sha256_source), out);
  223. // Log_Printf_Debug("out:\r\n");
  224. // Log_SendHex((uint8_t *)out, strlen(out));
  225. // Log_Printf_Debug("\r\n");
  226. memset(aes_key, 0, sizeof(aes_key));
  227. memcpy(aes_key, out+8, 16); //获取key
  228. // Log_Printf_Debug("aes_key:\r\n");
  229. // Log_SendHex(aes_key, 16);
  230. // Log_Printf_Debug("\r\n");
  231. // CoAP协议
  232. CoAP_Init(coap_message);
  233. CoAP_Set_T(0);
  234. CoAP_Set_Code(2);
  235. CoAP_Set_MessageID(++SeqOffset);
  236. // option
  237. // host
  238. CoAP_Set_Option_Str(3, regist_response.address);
  239. // port
  240. CoAP_Set_Option_Short(7, regist_response.port);
  241. // path
  242. CoAP_Set_Option_Str(11, "topic");
  243. CoAP_Set_Option_Str(11, "sys");
  244. CoAP_Set_Option_Str(11, regist_response.productId);
  245. CoAP_Set_Option_Str(11, regist_response.deviceId);
  246. CoAP_Set_Option_Str(11, "thing");
  247. CoAP_Set_Option_Str(11, "model");
  248. CoAP_Set_Option_Str(11, "up_raw");
  249. // content-format
  250. CoAP_Set_Option_Short(12, 50);
  251. // accept
  252. CoAP_Set_Option_Short(17, 50);
  253. // 2088
  254. CoAP_Set_Option_Str(2088, Token);
  255. // 2089
  256. memset(Rand2089, 0, sizeof(Rand2089));
  257. memset(Opt2089, 0, sizeof(Opt2089));
  258. my_itoa(SeqOffset, Rand2089, 10);
  259. int Opt2089_Length = utils_aes128_cbc_enc_with_length(aes_key, iv, (uint8_t *)Rand2089, strlen(Rand2089), Opt2089);
  260. CoAP_Set_Option(2089,Opt2089_Length, Opt2089);
  261. // Log_Printf_Debug("Opt2089:\r\n");
  262. // Log_SendHex(Opt2089, Opt2089_Length);
  263. // Log_Printf_Debug("\r\n");
  264. // Log_Printf_Debug("data:\r\n");
  265. // Log_SendHex(data, data_length);
  266. // Log_Printf_Debug("\r\n");
  267. // payload
  268. memset(payload, 0, sizeof(payload));
  269. int payload_length = utils_aes128_cbc_enc_with_length(aes_key, iv, data, data_length, payload);
  270. CoAP_Set_Payload(payload, payload_length);
  271. // Log_Printf_Debug("payload:\r\n");
  272. // Log_SendHex(payload, payload_length);
  273. // Log_Printf_Debug("\r\n");
  274. // 生成报文
  275. * coap_message_length = CoAP_Get_Length();
  276. // Log_Printf_Debug("STEP: 拼接数据报文>>\r\n");
  277. }
  278. // 设置Token
  279. static void set_token(char * token)
  280. {
  281. memset(Token, 0, sizeof(Token));
  282. strcpy(Token, token);
  283. }
  284. static void set_random(char * random)
  285. {
  286. memset(Random, 0, sizeof(Random));
  287. strcpy(Random, random);
  288. }
  289. static void set_seqOffset(uint32_t seqOffset)
  290. {
  291. SeqOffset = seqOffset;
  292. }
  293. // 接收数据处理
  294. static uint8_t plaintext[256]; // 明文
  295. uint8_t TUORENIOT_AnalysisMessage(uint8_t * coap_message, uint16_t coap_message_length)
  296. {
  297. // Log_Printf_Debug("coap_message(%d):\r\n", coap_message_length);
  298. // Log_SendHex(coap_message, coap_message_length);
  299. // Log_Printf_Debug("\r\n");
  300. memset(payload, 0, sizeof(payload));
  301. uint16_t payload_length = CoAP_Get_Payload(coap_message, coap_message_length, payload);
  302. Log_Printf_Debug("payload长度: %d\r\n", payload_length);
  303. // 判断是认证返回,还是数据发送返回
  304. if(payload_length == 0) // 无返回
  305. {
  306. // 清空Token,重新认证
  307. memset(Token, '\0', sizeof(Token));
  308. return 0;
  309. }
  310. else if(payload[0] == '{' && payload[payload_length-1] == '}') // 认证返回
  311. {
  312. // 解析json
  313. cJSON * cjson = cJSON_Parse((char * )payload);
  314. Log_Printf_Debug("plaintext(%d): %s\r\n",payload_length, payload);
  315. if(cJSON_HasObjectItem(cjson, "random"))
  316. {
  317. char * random = cJSON_GetObjectItem(cjson, "random")->valuestring;
  318. set_random(random);
  319. }
  320. if(cJSON_HasObjectItem(cjson, "seqOffset"))
  321. {
  322. uint16_t seqOffset = cJSON_GetObjectItem(cjson, "seqOffset")->valueint;
  323. set_seqOffset(seqOffset);
  324. }
  325. if(cJSON_HasObjectItem(cjson, "token"))
  326. {
  327. char * token = cJSON_GetObjectItem(cjson, "token")->valuestring;
  328. set_token(token);
  329. }
  330. cJSON_Delete(cjson);//清除结构体
  331. return 1;
  332. }
  333. else
  334. {
  335. // 数据发送成功返回
  336. memset(plaintext, 0, sizeof(plaintext));
  337. uint16_t plaintext_length = utils_aes128_cbc_dec(aes_key, iv, (char * )payload, payload_length, (char * )plaintext);
  338. Log_Printf_Debug("\r\nplaintext(%d): %s\r\n",plaintext_length, (char *)plaintext);
  339. return 2;
  340. }
  341. }
  342. void TUORENIOT_PrintfRegistRequestStruct(struct TUORENIOT_RegistRequestStruct * regist_params_struct)
  343. {
  344. Log_Printf("------print regist params start------\r\n");
  345. Log_Printf("version=%d\r\n", regist_params_struct->version);
  346. Log_Printf("networkType=%d\r\n", regist_params_struct->networkType);
  347. Log_Printf("networkProtocol=%d\r\n", regist_params_struct->networkProtocol);
  348. Log_Printf("productId=%s\r\n", regist_params_struct->productId);
  349. Log_Printf("deviceId=%s\r\n", regist_params_struct->deviceId);
  350. Log_Printf("userId=%s\r\n", regist_params_struct->userId);
  351. Log_Printf("------print regist params end------\r\n");
  352. }
  353. void TUORENIOT_PrintfRegistResponseStruct(void)
  354. {
  355. Log_Printf("------print regist result start------\r\n");
  356. Log_Printf("code=%d\r\n", regist_response.code);
  357. Log_Printf("version=%d\r\n", regist_response.version);
  358. Log_Printf("platform=%d\r\n", regist_response.platform);
  359. Log_Printf("networkType=%d\r\n", regist_response.networkType);
  360. Log_Printf("networkProtocol=%d\r\n", regist_response.networkProtocol);
  361. Log_Printf("productId=%s\r\n", regist_response.productId);
  362. Log_Printf("deviceId=%s\r\n", regist_response.deviceId);
  363. Log_Printf("userId=%s\r\n", regist_response.userId);
  364. Log_Printf("deviceSecret=%s\r\n", regist_response.deviceSecret);
  365. Log_Printf("remoteAddress=%s\r\n", regist_response.address);
  366. Log_Printf("remotePort=%d\r\n", regist_response.port);
  367. Log_Printf("wifiNumber=%d\r\n", regist_response.wifiNumber);
  368. Log_Printf("------print regist result end------\r\n");
  369. }
  370. // 初始化
  371. void TUORENIOT_Init(void)
  372. {
  373. // 从flash中读取注册结果
  374. ReadRegistResponseFromFlash();
  375. }
  376. // 清空flash中的注册响应
  377. void TUORENIOT_ClearRegistResponseFromFlash(void)
  378. {
  379. regist_response.code = 0;
  380. FlashWriteOperate((uint8_t*)&regist_response, sizeof(regist_response), FLASH_SAVE_ADDR);
  381. Log_Printf_Debug("清空注册数据完成\r\n");
  382. }
  383. // 从flash中读取注册响应
  384. static void ReadRegistResponseFromFlash(void)
  385. {
  386. Log_Printf_Debug("sizeof(regist_response): %d\r\n", sizeof(regist_response));
  387. //读取flash
  388. Log_Printf_Debug("读取前code:%d\r\n", regist_response.code);
  389. Log_Printf_Debug("sizeof(regist_result): %d\r\n", sizeof(regist_response));
  390. FlashReadOperate((uint8_t*)&regist_response, sizeof(regist_response), FLASH_SAVE_ADDR);
  391. Log_Printf_Debug("读取后code:%d\r\n", regist_response.code);
  392. if(regist_response.code == 200)
  393. {
  394. TUORENIOT_PrintfRegistResponseStruct();
  395. }
  396. }
  397. // 将注册响应写入到flash中
  398. static void WriteRegistResponseToFlash(void)
  399. {
  400. FlashWriteOperate((uint8_t*)&regist_response, sizeof(regist_response), FLASH_SAVE_ADDR);
  401. regist_response.code = 0;
  402. Log_Printf_Debug("写入数据完成\r\n");
  403. FlashReadOperate((uint8_t*)&regist_response, sizeof(regist_response), FLASH_SAVE_ADDR);
  404. TUORENIOT_PrintfRegistResponseStruct();
  405. }
  406. // 获取code
  407. uint8_t TUORENIOT_GetCodeOfRegistResponse(void)
  408. {
  409. return regist_response.code;
  410. }