customer_srv_fota.lua 6.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139
  1. --[[
  2. @module air_srv_fota
  3. @summary 使用自建服务器远程升级功能模块
  4. @version 1.0
  5. @date 2025.08.12
  6. @author 孟伟
  7. @usage
  8. 实现远程升级功能,具体流程如下:
  9. 1、接收 CUSTOMER_SRV_FOTA 系统消息,触发升级;
  10. 2、判断网卡是否连接成功;
  11. 3、初始化fota2模块;
  12. 4、调用fota2模块的升级函数;
  13. 5、根据升级结果进行处理;
  14. ]]
  15. --加在libfota2扩展库
  16. libfota2 = require "libfota2"
  17. -- 循环打印版本号, 方便看版本号变化, 非必须
  18. function get_version()
  19. log.info("降功耗 找合宙")
  20. log.info("fota", "脚本版本号", VERSION, "core版本号", rtos.version())
  21. end
  22. sys.timerLoopStart(get_version, 3000)
  23. -- fota升级标志:true 表示当前正有 FOTA 流程在跑
  24. local fota_running = false
  25. -- 升级结果的回调函数
  26. -- 功能:获取fota的回调函数
  27. -- 参数:
  28. -- result:number类型
  29. -- 0表示成功
  30. -- 1表示连接失败
  31. -- 2表示url错误
  32. -- 3表示服务器断开
  33. -- 4表示接收报文错误
  34. -- 5缺少必要的PROJECT_KEY参数
  35. local function fota_cb(ret)
  36. log.info("fota", ret)
  37. -- fota结束,无论成功还是失败,都释放fota_running标志
  38. fota_running = false
  39. if ret == 0 then
  40. log.info("升级包下载成功,重启模块")
  41. rtos.reboot()
  42. elseif ret == 1 then
  43. log.info("连接失败", "请检查url拼写或服务器配置(是否为内网)")
  44. elseif ret == 2 then
  45. log.info("url错误", "检查url拼写")
  46. elseif ret == 3 then
  47. log.info("服务器断开", "检查服务器白名单配置")
  48. elseif ret == 4 then
  49. log.error("FOTA 失败",
  50. "原因可能有:\n" ..
  51. "1) 服务器返回 200/206 但报文体为空(0 字节)—— 通常是升级包文件缺失或 URL 指向空文件;\n" ..
  52. "2) 服务器返回 4xx/5xx 等异常状态码 —— 请确认升级包已上传、URL 正确、鉴权信息有效;\n" ..
  53. "3) 已经是最新版本,无需升级")
  54. elseif ret == 5 then
  55. log.info("缺少必要的PROJECT_KEY参数")
  56. else
  57. log.info("不是上面几种情况 ret为", ret)
  58. end
  59. end
  60. -- 使用第三方服务器,配置ota_opts参数
  61. --[[
  62. -- opts参数说明, 所有参数都是可选的
  63. -- 1. opts.url string 升级所需要的URL, 若使用合宙iot平台,则不需要填
  64. -- 2. opts.version string 版本号, 默认是 BSP版本号.x.z格式
  65. -- 3. opts.timeout int 请求超时时间, 默认300000毫秒,单位毫秒
  66. -- 4. opts.project_key string 合宙IOT平台的项目key, 默认取全局变量PRODUCT_KEY. 自建服务器不用填
  67. -- 5. opts.imei string 设备识别码, 默认取IMEI(Cat.1模块)或WLAN MAC地址(wifi模块)或MCU唯一ID
  68. -- 6. opts.firmware_name string 固件名称,默认是 _G.PROJECT.. "_LuatOS-SoC_" .. rtos.bsp()
  69. -- 7. opts.server_cert string 服务器证书, 默认不使用
  70. -- 8. opts.client_cert string 客户端证书, 默认不使用
  71. -- 9. opts.client_key string 客户端私钥, 默认不使用
  72. -- 10. opts.client_password string 客户端私钥口令, 默认不使用
  73. -- 11. opts.method string 请求方法, 默认是GET
  74. -- 12. opts.headers table 额外添加的请求头,默认不需要
  75. -- 13. opts.body string 额外添加的请求body,默认不需要
  76. ]]
  77. local opts = {
  78. url = "",
  79. -- 合宙IOT平台的默认升级URL, 不填就是这个默认值
  80. -- 如果是自建的OTA服务器, 则需要填写正确的URL, 例如 http://192.168.1.5:8000/update
  81. -- 如果自建OTA服务器,且url包含全部参数,不需要额外添加参数, 请在url前面添加 ###
  82. -- 如果不加###,则默认会上传如下参数
  83. -- 1. opts.version string 版本号, 默认是 BSP版本号.x.z格式
  84. -- 2. opts.timeout int 请求超时时间, 默认300000毫秒,单位毫秒
  85. -- 3. opts.project_key string 合宙IOT平台的项目key, 默认取全局变量PRODUCT_KEY. 自建服务器不用填
  86. -- 4. opts.imei string 设备识别码, 默认取IMEI(Cat.1模块)或WLAN MAC地址(wifi模块)或MCU唯一ID
  87. -- 5. opts.firmware_name string 底层版本号
  88. -- 请求的版本号, 合宙IOT有一套版本号体系,不传就是合宙规则, 自建服务器的话当然是自行约定版本号了
  89. version = ""
  90. -- 其他更多参数, 请查阅libfota2的文档 https://docs.openluat.com/osapi/ext/libfota2/
  91. }
  92. local function air_fota_func(data)
  93. -- 如果当前时间点设置的默认网卡还没有连接成功,一直在这里循环等待
  94. while not socket.adapter(socket.dft()) do
  95. log.warn("fota_task_func", "wait IP_READY", socket.dft())
  96. -- 在此处阻塞等待默认网卡连接成功的消息"IP_READY"
  97. -- 或者等待1秒超时退出阻塞等待状态;
  98. -- 注意:此处的1000毫秒超时不要修改的更长;
  99. -- 因为当使用libnetif.set_priority_order配置多个网卡连接外网的优先级时,会隐式的修改默认使用的网卡
  100. -- 当libnetif.set_priority_order的调用时序和此处的socket.adapter(socket.dft())判断时序有可能不匹配
  101. -- 此处的1秒,能够保证,即使时序不匹配,也能1秒钟退出阻塞状态,再去判断socket.adapter(socket.dft())
  102. sys.waitUntil("IP_READY", 1000)
  103. end
  104. -- 检测到了IP_READY消息
  105. log.info("fota_task_func", "recv IP_READY", socket.dft())
  106. while true do
  107. -- 阻塞等待外部事件:"CUSTOMER_SRV_FOTA"
  108. local result, data = sys.waitUntil("CUSTOMER_SRV_FOTA")
  109. if result then
  110. log.info("接收到数据", "date", #data)
  111. if fota_running then
  112. log.warn("fota_task", "FOTA 正在运行,跳过本次请求")
  113. else
  114. -- 标记FOTA正在运行
  115. -- 注意:这里只是标记,实际的FOTA流程还没有开始
  116. opts.url = data.url
  117. opts.version = data.version
  118. fota_running = true
  119. log.info("开始检查升级")
  120. libfota2.request(fota_cb, opts)
  121. end
  122. end
  123. end
  124. end
  125. -- 初始化FOTA任务
  126. sys.taskInit(air_fota_func)