testALiYun.lua 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185
  1. --- 模块功能:阿里云功能测试.
  2. -- 支持数据传输和OTA功能
  3. -- @author openLuat
  4. -- @module aLiYun.testALiYun
  5. -- @license MIT
  6. -- @copyright openLuat
  7. -- @release 2018.04.14
  8. module(...,package.seeall)
  9. require"aLiYun"
  10. require"misc"
  11. require"pm"
  12. --地域和可用区,详情参考:https://help.aliyun.com/document_detail/40654.html?spm=a2c4g.11186623.2.22.797d7c80uIGAZ7
  13. --根据自己的产品所在地域修改
  14. local REGION_ID = "cn-shanghai"
  15. --三元组信息,根据实际值自行修改
  16. --注意:这里默认的三元组无法连接上阿里云
  17. local PRODUCT_KEY = "yourProductKey"
  18. local PRODUCE_SECRET = "8vOb5miTNRLrFecG"
  19. local DEVICE_NAME = "862991234567890"
  20. --[[
  21. 函数名:getDeviceName
  22. 功能 :获取设备名称
  23. 参数 :无
  24. 返回值:设备名称
  25. ]]
  26. local function getDeviceName()
  27. return DEVICE_NAME
  28. end
  29. --[[
  30. 函数名:getDeviceSecret
  31. 功能 :获取设备密钥
  32. 参数 :无
  33. 返回值:设备密钥
  34. ]]
  35. local function getDeviceSecret()
  36. --默认使用设备的SN作为设备密钥,用户可以根据项目需求自行修改
  37. return misc.getSn()
  38. end
  39. --[[
  40. 函数名:setDeviceSecret
  41. 功能 :修改设备密钥
  42. 参数 :设备密钥
  43. 返回值:无
  44. ]]
  45. local function setDeviceSecret(s)
  46. --默认使用设备的SN作为设备密钥,用户可以根据项目需求自行修改
  47. misc.setSn(s)
  48. end
  49. --阿里云客户端是否处于连接状态
  50. local sConnected
  51. local publishCnt = 1
  52. --[[
  53. 函数名:pubqos1testackcb
  54. 功能 :发布1条qos为1的消息后收到PUBACK的回调函数
  55. 参数 :
  56. usertag:调用mqttclient:publish时传入的usertag
  57. result:true表示发布成功,false或者nil表示失败
  58. 返回值:无
  59. ]]
  60. local function publishTestCb(result,para)
  61. log.info("testALiYun.publishTestCb",result,para)
  62. sys.timerStart(publishTest,20000)
  63. publishCnt = publishCnt+1
  64. end
  65. --发布一条QOS为1的消息
  66. function publishTest()
  67. if sConnected then
  68. --注意:在此处自己去控制payload的内容编码,aLiYun库中不会对payload的内容做任何编码转换
  69. aLiYun.publish("/"..PRODUCT_KEY.."/"..getDeviceName().."/update","qos1data",1,publishTestCb,"publishTest_"..publishCnt)
  70. end
  71. end
  72. ---数据接收的处理函数
  73. -- @string topic,UTF8编码的消息主题
  74. -- @number qos,消息质量等级
  75. -- @string payload,原始编码的消息负载
  76. local function rcvCbFnc(topic,qos,payload)
  77. log.info("testALiYun.rcvCbFnc",topic,qos,payload)
  78. end
  79. --- 连接结果的处理函数
  80. -- @bool result,连接结果,true表示连接成功,false或者nil表示连接失败
  81. local function connectCbFnc(result)
  82. log.info("testALiYun.connectCbFnc",result)
  83. sConnected = result
  84. if result then
  85. --订阅主题,不需要考虑订阅结果,如果订阅失败,aLiYun库中会自动重连
  86. --根据自己的项目需要订阅主题,下面注释掉的一行代码中的主题是非法的,所以不能打开,一旦打开,会导致订阅失败
  87. --aLiYun.subscribe({["/"..PRODUCT_KEY.."/"..getDeviceName().."/get"]=0, ["/"..PRODUCT_KEY.."/"..getDeviceName().."/get"]=1})
  88. --注册数据接收的处理函数
  89. aLiYun.on("receive",rcvCbFnc)
  90. --PUBLISH消息测试
  91. publishTest()
  92. end
  93. end
  94. --rrpcUseCustomTopic接口不是必须的,在设置rrpc自定义topic时调用
  95. --aLiyun.rrpcUseCustomTopic(true)
  96. aLiYun.on("connect",connectCbFnc)
  97. --setMqtt接口不是必须的,aLiYun.lua中有这个接口设置的参数默认值,如果默认值满足不了需求,参考下面注释掉的代码,去设置参数
  98. --aLiYun.setMqtt(0)
  99. aLiYun.setRegion(REGION_ID)
  100. aLiYun.setConnectMode("direct",PRODUCT_KEY..".iot-as-mqtt."..REGION_ID..".aliyuncs.com",1883)
  101. aLiYun.setup(PRODUCT_KEY,PRODUCE_SECRET,getDeviceName,getDeviceSecret,setDeviceSecret)
  102. --要使用阿里云OTA功能,必须参考本文件124或者126行aLiYun.setup去配置参数
  103. --然后加载阿里云OTA功能模块(打开下面的代码注释)
  104. require"aLiYunOta"
  105. --如果利用阿里云OTA功能去下载升级合宙模块的新固件,默认的固件版本号格式为:_G.PROJECT.."_".._G.VERSION.."_"..sys.getcorever(),下载结束后,直接重启,则到此为止,不需要再看下文说明
  106. --如果下载升级合宙模块的新固件,下载结束后,自己控制是否重启
  107. --如果利用阿里云OTA功能去下载其他升级包,例如模块外接的MCU升级包,则根据实际情况,打开下面的代码注释,调用设置接口进行配置和处理
  108. --设置MCU当前运行的固件版本号
  109. --aLiYunOta.setVer("MCU_VERSION_1.0.0")
  110. --设置新固件下载后保存的文件名
  111. --aLiYunOta.setName("MCU_FIRMWARE.bin")
  112. --[[
  113. 函数名:otaCb
  114. 功能 :新固件文件下载结束后的回调函数
  115. 通过uart1(115200,8,uart.PAR_NONE,uart.STOP_1)把下载成功的文件,发送到MCU,发送成功后,删除此文件
  116. 参数 :
  117. result:下载结果,true为成功,false为失败
  118. filePath:新固件文件保存的完整路径,只有result为true时,此参数才有意义
  119. 返回值:无
  120. ]]
  121. local function otaCb(result,filePath)
  122. log.info("testALiYun.otaCb",result,filePath)
  123. if result then
  124. local uartID = 1
  125. sys.taskInit(
  126. function()
  127. local fileHandle = io.open(filePath,"rb")
  128. if not fileHandle then
  129. log.error("testALiYun.otaCb open file error")
  130. if filePath then os.remove(filePath) end
  131. return
  132. end
  133. pm.wake("UART_SENT2MCU")
  134. uart.on(uartID,"sent",function() sys.publish("UART_SENT2MCU_OK") end)
  135. uart.setup(uartID,115200,8,uart.PAR_NONE,uart.STOP_1,nil,1)
  136. while true do
  137. local data = fileHandle:read(1460)
  138. if not data then break end
  139. uart.write(uartID,data)
  140. sys.waitUntil("UART_SENT2MCU_OK")
  141. end
  142. --此处上报新固件版本号(仅供测试使用)
  143. --用户开发自己的程序时,根据下载下来的新固件,执行升级动作
  144. --升级成功后,调用aLiYunOta.setVer上报新固件版本号
  145. --如果升级失败,调用aLiYunOta.setVer上报旧固件版本号
  146. aLiYunOta.setVer("MCU_VERSION_1.0.1")
  147. uart.close(uartID)
  148. pm.sleep("UART_SENT2MCU")
  149. fileHandle:close()
  150. if filePath then os.remove(filePath) end
  151. end
  152. )
  153. else
  154. --文件使用完之后,如果以后不再需求,需要自行删除
  155. if filePath then os.remove(filePath) end
  156. end
  157. end
  158. --设置新固件下载结果的回调函数
  159. --aLiYunOta.setCb(otaCb)