main.lua 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234
  1. --[[
  2. socket客户端演示
  3. 包括但不限于以下模组
  4. 1. EC618系列 -- Air780E/Air780EG/Air600E/Air700E
  5. 2. ESP32系列 -- ESP32C3/ESP32S3/ESP32C2
  6. 3. Air105搭配W5500
  7. 4. 其他适配了socket层的bsp
  8. 支持的协议有: TCP/UDP/TLS-TCP/DTLS, 更高层级的协议,如http有单独的库
  9. 提示:
  10. 1. socket支持多个连接的, 通常最多支持8个, 可通过不同的taskName进行区分
  11. 2. 支持与http/mqtt/websocket/ftp库同时使用, 互不干扰
  12. 3. 支持IP和域名, 域名是自动解析的, 但解析域名也需要耗时
  13. 4. 加密连接(TLS/SSL)需要更多内存, 这意味着能容纳的连接数会小很多, 同时也更慢
  14. 对于多个网络出口的场景, 例如Air780E+W5500组成4G+以太网:
  15. 1. 在socket.create函数设置网络适配器的id
  16. 2. 请到同级目录查阅更细致的demo
  17. 如需使用ipv6, 请查阅 demo/ipv6, 本demo只涉及ipv4
  18. ]]
  19. -- LuaTools需要PROJECT和VERSION这两个信息
  20. PROJECT = "scdemo"
  21. VERSION = "1.0.0"
  22. log.info("main", PROJECT, VERSION)
  23. -- 一定要添加sys.lua !!!!
  24. sys = require("sys")
  25. sysplus = require("sysplus")
  26. libnet = require "libnet"
  27. local uart_id = 1
  28. -- Air780E的AT固件默认会为开机键防抖, 导致部分用户刷机很麻烦
  29. if rtos.bsp() == "EC618" and pm and pm.PWK_MODE then
  30. pm.power(pm.PWK_MODE, false)
  31. end
  32. if wdt then
  33. --添加硬狗防止程序卡死,在支持的设备上启用这个功能
  34. wdt.init(9000)--初始化watchdog设置为9s
  35. sys.timerLoopStart(wdt.feed, 3000)--3s喂一次狗
  36. end
  37. --=============================================================
  38. -- 测试网站 https://netlab.luatos.com/ 点击 打开TCP 获取测试端口号
  39. -- 要按实际情况修改
  40. local host = "81.70.22.216" -- 服务器ip或者域名, 都可以的
  41. local port = 9999 -- 服务器端口号
  42. local is_udp = false -- 如果是UDP, 要改成true, false就是TCP
  43. local is_tls = false -- 加密与否, 要看服务器的实际情况
  44. --=============================================================
  45. -- 处理未识别的网络消息
  46. local function netCB(msg)
  47. --log.info("未处理消息", msg[1], msg[2], msg[3], msg[4])
  48. end
  49. uart.setup(uart_id, 115200) -- 注意, 是UART1, 不是虚拟串口, 演示目的
  50. uart.on(uart_id, "sent", function(id, len)
  51. log.info("sent", id, len)
  52. -- local start = mcu.ticks()
  53. uart.wait485(uart_id)
  54. -- local done = mcu.ticks()
  55. -- local used = done - start
  56. -- log.info("time use", used)
  57. sys.publish("uart_tx_done")
  58. end)
  59. -- 统一联网函数
  60. sys.taskInit(function()
  61. -----------------------------
  62. -- 统一联网函数, 可自行删减
  63. ----------------------------
  64. if wlan and wlan.connect then
  65. -- wifi 联网, ESP32系列均支持, 要根据实际情况修改ssid和password!!
  66. local ssid = "luatos1234"
  67. local password = "12341234"
  68. log.info("wifi", ssid, password)
  69. -- TODO 改成自动配网
  70. wlan.init()
  71. wlan.setMode(wlan.STATION) -- 默认也是这个模式,不调用也可以
  72. wlan.connect(ssid, password, 1)
  73. elseif mobile then
  74. -- EC618系列, 如Air780E/Air600E/Air700E
  75. -- mobile.simid(2) -- 自动切换SIM卡, 按需启用
  76. -- 模块默认会自动联网, 无需额外的操作
  77. elseif w5500 then
  78. -- w5500 以太网
  79. w5500.init(spi.HSPI_0, 24000000, pin.PC14, pin.PC01, pin.PC00)
  80. w5500.config() --默认是DHCP模式
  81. w5500.bind(socket.ETH0)
  82. elseif socket then
  83. -- 适配了socket库也OK, 就当1秒联网吧
  84. sys.timerStart(sys.publish, 1000, "IP_READY")
  85. else
  86. -- 其他不认识的bsp, 循环提示一下吧
  87. while 1 do
  88. sys.wait(1000)
  89. log.info("bsp", "本bsp可能未适配网络层, 请查证")
  90. end
  91. end
  92. -- 默认都等到联网成功
  93. sys.waitUntil("IP_READY")
  94. sys.publish("net_ready")
  95. end)
  96. -- 演示task
  97. local function sockettest()
  98. -- 等待联网
  99. sys.waitUntil("net_ready")
  100. socket.sntp()
  101. -- 开始正在的逻辑, 发起socket链接,等待数据/上报心跳
  102. local taskName = "sc"
  103. local topic = taskName .. "_txrx"
  104. log.info("topic", topic)
  105. local txqueue = {}
  106. sysplus.taskInitEx(sockettask, taskName, netCB, taskName, txqueue, topic)
  107. while 1 do
  108. local result, tp, data = sys.waitUntil(topic, 30000)
  109. -- log.info("event", result, tp, data)
  110. if not result then
  111. -- 等很久了,没数据上传/下发, 发个日期心跳包吧
  112. --table.insert(txqueue, string.char(0))
  113. --sys_send(taskName, socket.EVENT, 0)
  114. elseif tp == "uplink" then
  115. -- 上行数据, 主动上报的数据,那就发送呀
  116. -- table.insert(txqueue, data)
  117. -- sys_send(taskName, socket.EVENT, 0)
  118. elseif tp == "downlink" then
  119. -- 下行数据,接收的数据, 从ipv6task来的
  120. -- 其他代码可以通过 sys.publish()
  121. log.info("socket", "收到下发的数据了", #data)
  122. uart.write(uart_id, data)
  123. end
  124. end
  125. end
  126. function sockettask(d1Name, txqueue, rxtopic)
  127. -- 打印准备连接的服务器信息
  128. log.info("socket", host, port, is_udp and "UDP" or "TCP", is_tls and "TLS" or "RAW")
  129. -- 准备好所需要的接收缓冲区
  130. local rx_buff = zbuff.create(10*1024)
  131. local netc = socket.create(nil, d1Name)
  132. socket.config(netc, nil, is_udp, is_tls)
  133. log.info("任务id", d1Name)
  134. while true do
  135. local data_counter = 0
  136. -- 连接服务器, 15秒超时
  137. log.info("socket", "开始连接服务器")
  138. sysplus.cleanMsg(d1Name)
  139. local result = libnet.connect(d1Name, 15000, netc, host, port)
  140. if result then
  141. log.info("socket", "服务器连上了")
  142. --libnet.tx(d1Name, 0, netc, "helloworld")
  143. else
  144. log.info("socket", "服务器没连上了!!!")
  145. end
  146. while result do
  147. -- 连接成功之后, 先尝试接收
  148. -- log.info("socket", "调用rx接收数据")
  149. local succ, param = socket.rx(netc, rx_buff)
  150. if not succ then
  151. log.info("服务器断开了", succ, param, ip, port)
  152. break
  153. end
  154. -- 如果服务器有下发数据, used()就必然大于0, 进行处理
  155. while rx_buff:used() > 0 do
  156. log.info("socket", "收到服务器数据,长度", rx_buff:used())
  157. data_counter = data_counter + rx_buff:used()
  158. -- local data = rx_buff:query() -- 获取数据
  159. uart.tx(uart_id, rx_buff)
  160. log.info("数据统计", "累计接收字节数", data_counter)
  161. log.info("lua", rtos.meminfo())
  162. -- log.info("sys", rtos.meminfo("sys"))
  163. if sys.waitUntil("uart_tx_done") then
  164. -- sys.wait(10) -- 再等10ms
  165. end
  166. rx_buff:del()
  167. sysplus.cleanMsg(d1Name)
  168. socket.rx(netc, rx_buff)
  169. end
  170. -- log.info("libnet", "调用wait开始等待消息")
  171. -- 等待事件, 例如: 服务器下发数据, 有数据准备上报, 服务器断开连接
  172. result, param, param2 = libnet.wait(d1Name, 500, netc)
  173. -- log.info("libnet", "wait", result, param, param2)
  174. if not result then
  175. -- 网络异常了, 那就断开了, 执行清理工作
  176. log.info("socket", "服务器断开了", result, param)
  177. break
  178. end
  179. if #txqueue > 0 then
  180. -- 有待上报的数据,处理之
  181. while #txqueue > 0 do
  182. local data = table.remove(txqueue, 1)
  183. if not data then
  184. break
  185. end
  186. result,param = libnet.tx(d1Name, 15000, netc,data)
  187. log.info("libnet", "发送数据的结果", result, param)
  188. if not result then
  189. log.info("socket", "数据发送异常", result, param)
  190. break
  191. end
  192. end
  193. end
  194. -- 循环尾部, 继续下一轮循环
  195. end
  196. -- 能到这里, 要么服务器断开连接, 要么上报(tx)失败, 或者是主动退出
  197. libnet.close(d1Name, 5000, netc)
  198. -- log.info(rtos.meminfo("sys"))
  199. sys.wait(300000) -- 这是重连时长, 自行调整
  200. end
  201. end
  202. sys.taskInit(sockettest)
  203. -- 用户代码已结束---------------------------------------------
  204. -- 结尾总是这一句
  205. sys.run()
  206. -- sys.run()之后后面不要加任何语句!!!!!