| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221 |
- --[[
- IPv6服务端演示, 仅EC618系列支持, 例如Air780E/Air600E/Air780UG/Air700E
- ]]
- -- LuaTools需要PROJECT和VERSION这两个信息
- PROJECT = "ipv6_client"
- VERSION = "1.0.0"
- log.info("main", PROJECT, VERSION)
- -- 一定要添加sys.lua !!!!
- sys = require("sys")
- sysplus = require("sysplus")
- libnet = require "libnet"
- -- Air780E的AT固件默认会为开机键防抖, 导致部分用户刷机很麻烦
- if rtos.bsp() == "EC618" and pm and pm.PWK_MODE then
- pm.power(pm.PWK_MODE, false)
- end
- -- 处理未识别的网络消息
- local function netCB(msg)
- log.info("未处理消息", msg[1], msg[2], msg[3], msg[4])
- end
- LED = gpio.setup(27, 0)
- GPIO1 = gpio.setup(1, 0)
- GPIO24 = gpio.setup(24, 0)
- HTTP_200_EMTRY = "HTTP/1.0 200 OK\r\nServer: LuatOS\r\nConnection: close\r\nContent-Length: 0\r\n\r\n"
- -- 演示task
- function ipv6test()
- -- 仅EC618系列支持, 例如Air780E/Air600E/Air780UG/Air700E
- if rtos.bsp() ~= "EC618" then
- while 1 do
- log.info("ipv6", "only Air780E/Air600E/Air780UG/Air700E supported")
- sys.wait(1000)
- end
- end
- -- 启用IPv6, 默认关闭状态,必须在驻网前开启
- -- 注意, 启用IPv6, 联网速度会慢2~3秒
- mobile.ipv6(true)
- log.info("ipv6", "等待联网")
- sys.waitUntil("IP_READY")
- log.info("ipv6", "联网完成")
- sys.wait(100)
- -- 打印一下本地ip, 一般只有ipv6才可能是公网ip, ipv4基本不可能
- -- 而且ipv6不一样是外网ip, 这是运营商决定的, 模块无能为力
- ip, mask, gw, ipv6 = socket.localIP()
- log.info("本地IP地址", ip, ipv6)
- if not ipv6 then
- log.info("没有IPV6地址,无法演示")
- -- return
- end
- -- 这里给个演示用的ddns, 临时自建的, 仅供测试
- local code, headers, body = http.request("GET", "http://81.70.22.216:8280/update?secret=luatos&domain=" .. mobile.imei() ..".dyndns&addr=" .. ipv6).wait()
- log.info("DDNS", "已映射", mobile.imei() .. ".dyndns.u8g2.com", code, body)
- log.info("shell", "telnet -6 " .. mobile.imei() .. ".dyndns.u8g2.com 14000")
- -- 开始正在的逻辑, 发起socket链接,等待数据/上报心跳
- local taskName = "ipv6client"
- local topic = taskName .. "_txrx"
- local txqueue = {}
- sysplus.taskInitEx(ipv6task, taskName, netCB, taskName, txqueue, topic)
- while 1 do
- local result, tp, data = sys.waitUntil(topic, 60000)
- if not result then
- -- 等很久了,没数据上传/下发, 发个日期心跳包吧
- --table.insert(txqueue, string.char(0))
- --sys_send(taskName, socket.EVENT, 0)
- elseif tp == "uplink" then
- -- 上行数据, 主动上报的数据,那就发送呀
- table.insert(txqueue, data)
- sys_send(taskName, socket.EVENT, 0)
- elseif tp == "downlink" then
- -- 下行数据,接收的数据, 从ipv6task来的
- -- 其他代码可以通过 sys.publish()
- log.info("socket", "收到下发的数据了", #data, data)
- -- 下面是模拟一个http服务, 因为httpsrv库还没好,先用着吧
- if data:startsWith("GET / ") then
- local httpresp = "HTTP/1.0 200 OK\r\n"
- httpresp = httpresp .. "Server: LuatOS\r\nContent-Type: text/html\r\nConnection: close\r\n"
- local fdata = io.readFile("/luadb/index.html")
- httpresp = httpresp .. string.format("Content-Length: %d\r\n\r\n", #fdata)
- httpresp = httpresp .. fdata
- table.insert(txqueue, httpresp)
- table.insert(txqueue, "close")
- sys_send(taskName, socket.EVENT, 0)
- elseif data:startsWith("GET /led/") then
- if data:startsWith("GET /led/1") then
- log.info("led", "亮起")
- LED(1)
- else
- log.info("led", "熄灭")
- LED(0)
- end
- table.insert(txqueue, HTTP_200_EMTRY)
- table.insert(txqueue, "close")
- sys_send(taskName, socket.EVENT, 0)
- elseif data:startsWith("GET /gpio24/") then
- if data:startsWith("GET /gpio24/1") then
- log.info("gpio24", "亮起")
- GPIO24(1)
- else
- log.info("gpio24", "熄灭")
- GPIO24(0)
- end
- table.insert(txqueue, HTTP_200_EMTRY)
- table.insert(txqueue, "close")
- sys_send(taskName, socket.EVENT, 0)
- elseif data:startsWith("GET /gpio1/") then
- if data:startsWith("GET /gpio1/1") then
- log.info("gpio1", "亮起")
- GPIO1(1)
- else
- log.info("gpio1", "熄灭")
- GPIO1(0)
- end
- table.insert(txqueue, HTTP_200_EMTRY)
- table.insert(txqueue, "close")
- sys_send(taskName, socket.EVENT, 0)
- elseif data:startsWith("GET ") or data:startsWith("POST ") or data:startsWith("HEAD ") then
- table.insert(txqueue, HTTP_200_EMTRY)
- table.insert(txqueue, "close")
- sys_send(taskName, socket.EVENT, 0)
- end
- end
- end
- end
- function ipv6task(d1Name, txqueue, rxtopic)
- -- 本地监听的端口
- local port = 14000
- local rx_buff = zbuff.create(1024)
- local tx_buff = zbuff.create(4 * 1024)
- local netc = socket.create(nil, d1Name)
- socket.config(netc, 14000)
- log.info("任务id", d1Name)
- while true do
- log.info("socket", "开始监控")
- local result = libnet.listen(d1Name, 0, netc)
- if result then
- log.info("socket", "监听成功")
- result = socket.accept(netc, nil) --只支持1对1
- if result then
- log.info("客户端连上了")
- end
- else
- log.info("socket", "监听失败!!")
- end
- while result do
- -- log.info("socket", "调用rx接收数据")
- local succ, param = socket.rx(netc, rx_buff)
- if not succ then
- log.info("客户端断开了", succ, param, ip, port)
- break
- end
- if rx_buff:used() > 0 then
- log.info("socket", "收到客户端数据,长度", rx_buff:used())
- local data = rx_buff:query() -- 获取数据
- sys.publish(rxtopic, "downlink", data)
- rx_buff:del()
- end
- -- log.info("libnet", "调用wait开始等待消息")
- result, param, param2 = libnet.wait(d1Name, 15000, netc)
- log.info("libnet", "wait", result, param, param2)
- if not result then
- -- 网络异常了
- log.info("socket", "客户端断开了", result, param)
- break
- elseif #txqueue > 0 then
- local force_close = false
- while #txqueue > 0 do
- local data = table.remove(txqueue, 1)
- if not data then
- break
- end
- log.info("socket", "上行数据长度", #data)
- if data == "close" then
- --sys.wait(1000)
- force_close = true
- break
- end
- tx_buff:del()
- tx_buff:copy(nil, data)
- result,param = libnet.tx(d1Name, 15000, netc, tx_buff)
- log.info("libnet", "发送数据的结果", result, param)
- if not result then
- log.info("socket", "数据发送异常", result, param)
- break
- end
- end
- if force_close then
- break
- end
- end
- end
- log.info("socket", "连接已断开,继续下一个循环")
- libnet.close(d1Name, 5000, netc)
- -- log.info(rtos.meminfo("sys"))
- sys.wait(50)
- end
- end
- sys.taskInit(ipv6test)
- -- 用户代码已结束---------------------------------------------
- -- 结尾总是这一句
- sys.run()
- -- sys.run()之后后面不要加任何语句!!!!!
|