| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209 |
- --- 模块功能:阿里云功能测试.
- -- 支持数据传输和OTA功能
- -- @author openLuat
- -- @module aLiYun.testALiYun
- -- @license MIT
- -- @copyright openLuat
- -- @release 2018.04.14
- module(...,package.seeall)
- require"aLiYun"
- require"misc"
- require"pm"
- --采用一机一密认证方案时:
- --PRODUCT_KEY为阿里云华东2站点上创建的产品的ProductKey,用户根据实际值自行修改
- local PRODUCT_KEY = "b0FMK1Ga5cp"
- --除了上面的PRODUCT_KEY外,还需要提供获取DeviceName的函数、获取DeviceSecret的函数
- --设备名称使用函数getDeviceName的返回值,默认为设备的IMEI
- --设备密钥使用函数getDeviceSecret的返回值,默认为设备的SN
- --单体测试时,可以直接修改getDeviceName和getDeviceSecret的返回值
- --批量量产时,使用设备的IMEI和SN;合宙生产的模块,都有唯一的IMEI,用户可以在自己的产线批量写入跟IMEI(设备名称)对应的SN(设备密钥)
- --或者用户自建一个服务器,设备上报IMEI给服务器,服务器返回对应的设备密钥,然后调用misc.setSn接口写到设备的SN中
- --采用一型一密认证方案时:
- --PRODUCT_KEY和PRODUCE_SECRET为阿里云华东2站点上创建的产品的ProductKey和ProductSecret,用户根据实际值自行修改
- --local PRODUCT_KEY = "b1KCi45LcCP"
- --local PRODUCE_SECRET = "VWll9fiYWKiwraBk"
- --除了上面的PRODUCT_KEY和PRODUCE_SECRET外,还需要提供获取DeviceName的函数、获取DeviceSecret的函数、设置DeviceSecret的函数
- --设备第一次在某个product下使用时,会先去云端动态注册,获取到DeviceSecret后,调用设置DeviceSecret的函数保存DeviceSecret
- --[[
- 函数名:getDeviceName
- 功能 :获取设备名称
- 参数 :无
- 返回值:设备名称
- ]]
- local function getDeviceName()
- --默认使用设备的IMEI作为设备名称,用户可以根据项目需求自行修改
- return misc.getImei()
-
- --用户单体测试时,可以在此处直接返回阿里云的iot控制台上注册的设备名称,例如return "862991419835241"
- --return "862991419835241"
- end
- --[[
- 函数名:setDeviceSecret
- 功能 :修改设备密钥
- 参数 :设备密钥
- 返回值:无
- ]]
- local function setDeviceSecret(s)
- --默认使用设备的SN作为设备密钥,用户可以根据项目需求自行修改
- misc.setSn(s)
- end
- --[[
- 函数名:getDeviceSecret
- 功能 :获取设备密钥
- 参数 :无
- 返回值:设备密钥
- ]]
- local function getDeviceSecret()
- --默认使用设备的SN作为设备密钥,用户可以根据项目需求自行修改
- return misc.getSn()
-
- --用户单体测试时,可以在此处直接返回阿里云的iot控制台上生成的设备密钥,例如return "y7MTCG6Gk33Ux26bbWSpANl4OaI0bg5Q"
- --return "y7MTCG6Gk33Ux26bbWSpANl4OaI0bg5Q"
- end
- --阿里云客户端是否处于连接状态
- local sConnected
- local publishCnt = 1
- --[[
- 函数名:pubqos1testackcb
- 功能 :发布1条qos为1的消息后收到PUBACK的回调函数
- 参数 :
- usertag:调用mqttclient:publish时传入的usertag
- result:true表示发布成功,false或者nil表示失败
- 返回值:无
- ]]
- local function publishTestCb(result,para)
- log.info("testALiYun.publishTestCb",result,para)
- sys.timerStart(publishTest,20000)
- publishCnt = publishCnt+1
- end
- --发布一条QOS为1的消息
- function publishTest()
- if sConnected then
- --注意:在此处自己去控制payload的内容编码,aLiYun库中不会对payload的内容做任何编码转换
- aLiYun.publish("/"..PRODUCT_KEY.."/"..getDeviceName().."/update","qos1data",1,publishTestCb,"publishTest_"..publishCnt)
- end
- end
- ---数据接收的处理函数
- -- @string topic,UTF8编码的消息主题
- -- @number qos,消息质量等级
- -- @string payload,原始编码的消息负载
- local function rcvCbFnc(topic,qos,payload)
- log.info("testALiYun.rcvCbFnc",topic,qos,payload)
- end
- --- 连接结果的处理函数
- -- @bool result,连接结果,true表示连接成功,false或者nil表示连接失败
- local function connectCbFnc(result)
- log.info("testALiYun.connectCbFnc",result)
- sConnected = result
- if result then
- --订阅主题,不需要考虑订阅结果,如果订阅失败,aLiYun库中会自动重连
- aLiYun.subscribe({["/"..PRODUCT_KEY.."/"..getDeviceName().."/get"]=0, ["/"..PRODUCT_KEY.."/"..getDeviceName().."/get"]=1})
- --注册数据接收的处理函数
- aLiYun.on("receive",rcvCbFnc)
- --PUBLISH消息测试
- publishTest()
- end
- end
- -- 认证结果的处理函数
- -- @bool result,认证结果,true表示认证成功,false或者nil表示认证失败
- local function authCbFnc(result)
- log.info("testALiYun.authCbFnc",result)
- end
- --采用一机一密认证方案时:
- --配置:ProductKey、获取DeviceName的函数、获取DeviceSecret的函数;其中aLiYun.setup中的第二个参数必须传入nil
- aLiYun.setup(PRODUCT_KEY,nil,getDeviceName,getDeviceSecret)
- --采用一型一密认证方案时:
- --配置:ProductKey、ProductSecret、获取DeviceName的函数、获取DeviceSecret的函数、设置DeviceSecret的函数
- --aLiYun.setup(PRODUCT_KEY,PRODUCE_SECRET,getDeviceName,getDeviceSecret,setDeviceSecret)
- --setMqtt接口不是必须的,aLiYun.lua中有这个接口设置的参数默认值,如果默认值满足不了需求,参考下面注释掉的代码,去设置参数
- --aLiYun.setMqtt(0)
- aLiYun.on("auth",authCbFnc)
- aLiYun.on("connect",connectCbFnc)
- --要使用阿里云OTA功能,必须参考本文件124或者126行aLiYun.setup去配置参数
- --然后加载阿里云OTA功能模块(打开下面的代码注释)
- require"aLiYunOta"
- --如果利用阿里云OTA功能去下载升级合宙模块的新固件,默认的固件版本号格式为:_G.PROJECT.."_".._G.VERSION.."_"..sys.getcorever(),下载结束后,直接重启,则到此为止,不需要再看下文说明
- --如果下载升级合宙模块的新固件,下载结束后,自己控制是否重启
- --如果利用阿里云OTA功能去下载其他升级包,例如模块外接的MCU升级包,则根据实际情况,打开下面的代码注释,调用设置接口进行配置和处理
- --设置MCU当前运行的固件版本号
- --aLiYunOta.setVer("MCU_VERSION_1.0.0")
- --设置新固件下载后保存的文件名
- --aLiYunOta.setName("MCU_FIRMWARE.bin")
- --[[
- 函数名:otaCb
- 功能 :新固件文件下载结束后的回调函数
- 通过uart1(115200,8,uart.PAR_NONE,uart.STOP_1)把下载成功的文件,发送到MCU,发送成功后,删除此文件
- 参数 :
- result:下载结果,true为成功,false为失败
- filePath:新固件文件保存的完整路径,只有result为true时,此参数才有意义
- 返回值:无
- ]]
- local function otaCb(result,filePath)
- log.info("testALiYun.otaCb",result,filePath)
- if result then
- local uartID = 1
- sys.taskInit(
- function()
- local fileHandle = io.open(filePath,"rb")
- if not fileHandle then
- log.error("testALiYun.otaCb open file error")
- if filePath then os.remove(filePath) end
- return
- end
-
- pm.wake("UART_SENT2MCU")
- uart.on(uartID,"sent",function() sys.publish("UART_SENT2MCU_OK") end)
- uart.setup(uartID,115200,8,uart.PAR_NONE,uart.STOP_1,nil,1)
- while true do
- local data = fileHandle:read(1460)
- if not data then break end
- uart.write(uartID,data)
- sys.waitUntil("UART_SENT2MCU_OK")
- end
- --此处上报新固件版本号(仅供测试使用)
- --用户开发自己的程序时,根据下载下来的新固件,执行升级动作
- --升级成功后,调用aLiYunOta.setVer上报新固件版本号
- --如果升级失败,调用aLiYunOta.setVer上报旧固件版本号
- aLiYunOta.setVer("MCU_VERSION_1.0.1")
-
- uart.close(uartID)
- pm.sleep("UART_SENT2MCU")
- fileHandle:close()
- if filePath then os.remove(filePath) end
- end
- )
-
- else
- --文件使用完之后,如果以后不再需求,需要自行删除
- if filePath then os.remove(filePath) end
- end
- end
- --设置新固件下载结果的回调函数
- --aLiYunOta.setCb(otaCb)
|