socketTask.lua 9.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226
  1. --- testSocket
  2. -- @module testSocket
  3. -- @author AIRM2M
  4. -- @license MIT
  5. -- @copyright openLuat.com
  6. -- @release 2018.10.27
  7. require "socket"
  8. module(..., package.seeall)
  9. -- 此处的IP和端口请填上你自己的socket服务器和端口
  10. --local ip, port, c = "116.62.220.88", "20160"
  11. local dataIp, dataTcpPort, dataUdpPort = nvm.get("dataIp"), nvm.get("dataTcpPort"), nvm.get("dataUdpPort")
  12. local macAddr = nvm.get("devId")
  13. local rfidSocketRetryInterval = nvm.get("rfidSocketRetryInterval") or 1
  14. local rfidSocketRetryTimes = nvm.get("rfidSocketRetryTimes") or 5
  15. local rfidSocketRetrySleep = nvm.get("rfidSocketRetrySleep") or 20
  16. --1M ram 可用空间,超出空间,硬件会重启
  17. local upload_packet_data_queue_max_length = 50
  18. local upload_packet_data_queue = {}
  19. local function insert_upload_packet_data_queue(packet_data)
  20. log.info("#upload_packet_data_queue = ",#upload_packet_data_queue)
  21. if #upload_packet_data_queue >= upload_packet_data_queue_max_length then
  22. table.remove(upload_packet_data_queue, 1)
  23. end
  24. table.insert(upload_packet_data_queue, packet_data)
  25. end
  26. local function upload_packet_data( packet_data )
  27. if not packet_data then
  28. log.error("socketTask.upload_packet_data","packet_data empty!")
  29. return false
  30. end
  31. if not dataIp then
  32. log.error("socketTask.upload_packet_data","dataIp empty!")
  33. return false
  34. end
  35. if not dataTcpPort then
  36. log.error("socketTask.upload_packet_data","dataTcpPort empty!")
  37. return false
  38. end
  39. --上传前,先进行断电检测(断电后,不再上传数据,缓存一段时间后,写入flash,等上电后再上传)
  40. local retryConnectCnt = 0
  41. while true do
  42. while not socket.isReady() do
  43. log.info("socketTask.upload_packet_data",">>>>>>> socket is not ready, wait IP_READY_IND ...")
  44. sys.waitUntil("IP_READY_IND",300000)
  45. end
  46. local socketClient = socket.tcp()
  47. if retryConnectCnt>=rfidSocketRetryTimes then
  48. --断开socket连接
  49. socketClient:close()
  50. log.info("socketTask.upload_packet_data",">>>>>>> retryConnectCnt>=5,link.shut,wait 20s ...")
  51. link.shut()
  52. retryConnectCnt=0
  53. sys.wait(rfidSocketRetrySleep*1000)
  54. while not socket.isReady() do
  55. log.info("socketTask.upload_packet_data",">>>>>>> socket is not ready, wait IP_READY_IND again ...")
  56. sys.waitUntil("IP_READY_IND",300000)
  57. end
  58. socketClient = socket.tcp()
  59. end
  60. --阻塞执行socket connect动作,直至成功
  61. if socketClient:connect(dataIp, dataTcpPort, 10) then
  62. if socketClient:send(packet_data, 10) then
  63. --断开socket连接
  64. socketClient:close()
  65. retryConnectCnt = 0
  66. log.info("socketTask.upload_packet_data","------> packet_data upload success <------")
  67. break
  68. else
  69. --断开socket连接
  70. socketClient:close()
  71. retryConnectCnt = retryConnectCnt+1
  72. log.info("socketTask.upload_packet_data",">>>>>>> socket.send failed,wait 5s ...,retryConnectCnt = ",retryConnectCnt)
  73. sys.wait(rfidSocketRetryInterval*1000)
  74. end
  75. else
  76. --断开socket连接
  77. socketClient:close()
  78. retryConnectCnt = retryConnectCnt+1
  79. log.info("socketTask.upload_packet_data",">>>>>>> socket.connect failed,wait 5s ...,retryConnectCnt = ",retryConnectCnt)
  80. sys.wait(rfidSocketRetryInterval*1000)
  81. end
  82. end
  83. return true
  84. end
  85. --[[
  86. sys.taskInit(
  87. function()
  88. while true do
  89. --阻塞监听监听串口接收数据
  90. log.info("socketTask.insert_upload_packet_data_queue",">>>>>>>>>>> waitUntil pub_msg(timeout:3600s) ...")
  91. local result, packet_data = sys.waitUntil("pub_msg",3600*1000)
  92. if result == true then
  93. local cur_time = os.time()
  94. insert_upload_packet_data_queue(packet_data)
  95. --log.info("socketTask.insert_upload_packet_data_queue",">>>>>>>> insert upload_packet_data_queue times(s):",os.time()-cur_time)
  96. else
  97. log.info("socketTask.insert_upload_packet_data_queue",">>>>>>>> wait pub_msg timeout")
  98. end
  99. end
  100. end
  101. )
  102. ]]
  103. sys.taskInit(
  104. function()
  105. while true do
  106. --从队列逐个取出数据,并上传到服务端
  107. if #upload_packet_data_queue > 0 then
  108. local cur_time = os.time()
  109. local packet_data = table.remove(upload_packet_data_queue, 1)
  110. --log.info("socketTask.upload_packet_data_queue",">>>>>>>> pop upload_packet_data_queue times(s):",os.time()-cur_time)
  111. if packet_data then
  112. if not upload_packet_data( packet_data ) then
  113. log.error("socketTask.upload_packet_data_queue",">>>>>>>> upload_packet_data failed!!")
  114. end
  115. end
  116. else
  117. log.info("socketTask.upload_packet_data_queue",">>>>>>>> upload_packet_data_queue empty, wait 1s ...")
  118. sys.wait(1000)
  119. end
  120. end
  121. end
  122. )
  123. local function hex2bin( hexstr )
  124. local str = ""
  125. for i = 1, string.len(hexstr) - 1, 2 do
  126. local doublebytestr = string.sub(hexstr, i, i+1);
  127. local n = tonumber(doublebytestr, 16);
  128. if 0 == n then
  129. str = str .. '\00'
  130. else
  131. str = str .. string.format("%c", n)
  132. end
  133. end
  134. return str
  135. end
  136. --组装待发送数据包
  137. local function parse_packet( packet_data )
  138. if not packet_data then return end
  139. --取imei后三个字节
  140. if not macAddr then
  141. log.info("socketTask.parse_packet","macAddr empty!")
  142. macAddr = nvm.get("devId")
  143. return false
  144. end
  145. if #macAddr < 6 then
  146. log.info("socketTask.parse_packet","parse_packet fail,macAddr length < 6! macAddr = ",macAddr)
  147. return false
  148. end
  149. local packet_bin = string.char(0x9c)..string.char(0xbc)..string.char(0x01)..hex2bin(macAddr:sub(-6))
  150. local length = 0
  151. local tmp_bin = ""
  152. for key, rfid in pairs(packet_data["buf"]) do
  153. if type(rfid) == "table" then
  154. if not rfid.time then
  155. log.info("socketTask.parse_packet",">>>>>>>>> rfid.time empty! ", type(rfid))
  156. for k, v in pairs(rfid) do
  157. log.info("socketTask.parse_packet",">>>>>>>>> rfid", "k:", k, ", v:", v)
  158. end
  159. return false
  160. end
  161. tmp_bin = tmp_bin .. hex2bin(key)..string.char(rfid.type)..string.char(rfid.status)
  162. ..string.char(rfid.rssi)..hex2bin(string.format( "%08X", rfid.time ))
  163. ..string.char(rfid.recv_count)..string.char(60)..string.char(1)
  164. else
  165. log.info("socketTask.parse_packet",">>>>>>>>> rfid error! key = ", key)
  166. return false
  167. end
  168. end
  169. packet_bin = packet_bin .. hex2bin(string.format( "%04X", tmp_bin:len())) .. tmp_bin
  170. --log.info(">>>>>>>>> sub packet_bin:sub(1, 30):toHex()", packet_bin:sub(1, 30):toHex(), ", macAddr:", macAddr)
  171. return true,packet_bin
  172. end
  173. -- 测试代码,用于发送消息给socket
  174. sys.taskInit(function()
  175. while true do
  176. --log.info("socketTask.pub_msg",">>>>>>>>>>> waitUntil pub_packet_data_renlian ")
  177. --阻塞监听监听串口接收数据
  178. local cur_time = os.time()
  179. local result, packet_data = sys.waitUntil("pub_packet_data_renlian",300*1000)
  180. if result then
  181. --log.info("socketTask.pub_msg",">>>>>>>>> recv pub_packet_data_renlian, times(s):", os.time()-cur_time, json.encode(packet_data))
  182. --[[
  183. --设备解析rfid性能测试代码
  184. local rfid_str = ""
  185. for key, rfid in pairs(packet_data["buf"]) do
  186. rfid_str = rfid_str..key..","
  187. end
  188. log.info("socketTask.pub_msg", "timestamp:", cur_time, "rfid_str:",rfid_str)
  189. ]]
  190. cur_time = os.time()
  191. local res, parse_packet_data = parse_packet(packet_data)
  192. if res then
  193. log.info("socketTask.pub_msg",">>>>>>>>> publish pub_msg, parse_packet times(s):", os.time()-cur_time, ", parse_packet_data:sub(1, 30):toHex() = ",parse_packet_data:sub(1, 30):toHex())
  194. --log.info("socketTask.pub_msg",", parse_packet_data:toHex() = ", parse_packet_data:toHex())
  195. --sys.publish("pub_msg", parse_packet_data)
  196. insert_upload_packet_data_queue(parse_packet_data)
  197. else
  198. log.error("socketTask.pub_msg",">>>>>>>>> parse_packet failed!!")
  199. end
  200. else
  201. log.info("socketTask.pub_msg",">>>>>>>>> wait pub_packet_data_renlian timeout")
  202. --sys.wait(1000)
  203. end
  204. end
  205. end)
  206. sys.timerLoopStart(function()
  207. log.info("打印占用的内存:", _G.collectgarbage("count"))-- 打印占用的RAM
  208. log.info("打印可用的空间", rtos.get_fs_free_size())-- 打印剩余FALSH,单位Byte
  209. end, 10000)