exfotawifi.lua 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221
  1. --[[
  2. @module exfotawifi
  3. @summary 用于Air8000/8000A/8000W型号模组自动升级WIFI
  4. @version 1.0.3
  5. @date 2025.9.23
  6. @author 拓毅恒
  7. @usage
  8. 注:使用时在创建的一个task处理函数中直接调用exfotawifi.request()即可开始执行WiFi升级任务
  9. 升级完毕后最好取消调用,防止后期版本升级过高导致程序使用不稳定
  10. -- 用法实例
  11. local exfotawifi = require("exfotawifi")
  12. local function fota_wifi_task()
  13. -- ...此处省略很多代码
  14. local result = exfotawifi.request()
  15. if result then
  16. log.info("exfotawifi", "升级任务执行成功")
  17. else
  18. log.info("exfotawifi", "升级任务执行失败")
  19. end
  20. -- ...此处省略很多代码
  21. end
  22. -- 启动WiFi自动更新任务
  23. sys.taskInit(fota_wifi_task)
  24. ]]
  25. local exfotawifi = {}
  26. local is_request = false -- 标记是否正在执行request任务
  27. local fota_result = false -- 记录fota任务的执行结果
  28. -- 判断是否为空
  29. local function is_nil(s)
  30. return s == nil or s == ""
  31. end
  32. -- 判断json是否合法
  33. local function is_json(str)
  34. local success, result = pcall(json.decode, str)
  35. return success and type(result) == "table"
  36. end
  37. -- 解析服务器响应的json数据
  38. local function parse_response(body)
  39. if not body or body == "" then
  40. log.error("exfotawifi", "返回的body为空")
  41. return nil
  42. end
  43. local success, json_body = pcall(json.decode, body)
  44. if success and type(json_body) == "table" then
  45. log.info("exfotawifi", "解析服务器响应成功")
  46. return json_body
  47. else
  48. log.error("exfotawifi", "解析服务器响应失败,body内容:", body)
  49. return nil
  50. end
  51. end
  52. -- 判断是否需要升级,返回true或false
  53. local function need_fota(version, server_version)
  54. local version_num = tonumber(version)
  55. local server_version_num = tonumber(server_version)
  56. if version_num < server_version_num then
  57. return true
  58. end
  59. return false
  60. end
  61. -- 下载升级文件,支持断点续传
  62. local function download_file(url)
  63. local file_path = "/luadb/fotawifi.bin"
  64. local downloaded_size = 0
  65. -- 检查文件是否存在,获取已下载的大小
  66. if io.exists(file_path) then
  67. downloaded_size = io.fileSize(file_path)
  68. log.info("exfotawifi", "检测到未完成的下载,已下载大小:", downloaded_size)
  69. end
  70. -- 设置请求头,支持断点续传
  71. local headers = {}
  72. if downloaded_size > 0 then
  73. headers["Range"] = "bytes=" .. downloaded_size .. "-"
  74. end
  75. local code, headers, body = http.request("GET", url, headers, nil, nil).wait()
  76. if code == 200 or code == 206 then
  77. -- 开始写入文件
  78. local file_mode = downloaded_size > 0 and "a+" or "w+"
  79. local file = io.open(file_path, file_mode)
  80. if file then
  81. file:seek("end", downloaded_size)
  82. file:write(body)
  83. file:close()
  84. -- 判断文件是否下载完整
  85. local file_size = io.fileSize(file_path)
  86. local content_length = tonumber(headers["content-length"] or headers["Content-Length"])
  87. if file_size >= (content_length or file_size) then
  88. log.info("exfotawifi", "下载升级文件成功,文件路径:", file_path)
  89. return file_path
  90. else
  91. log.info("exfotawifi", "下载中...当前大小:", file_size, "目标大小:", content_length)
  92. end
  93. else
  94. log.error("exfotawifi", "无法创建文件")
  95. -- 删除不完整的文件
  96. os.remove(file_path)
  97. end
  98. else
  99. log.error("exfotawifi", "下载失败,状态码:", code)
  100. -- 删除不完整的文件
  101. if io.exists(file_path) then
  102. os.remove(file_path)
  103. end
  104. end
  105. return nil
  106. end
  107. -- 执行升级操作
  108. local function fota_start(file_path)
  109. -- 检查文件是否存在
  110. if not io.exists(file_path) then
  111. log.error("exfotawifi", "升级文件不存在")
  112. return false
  113. end
  114. -- 检查文件大小是否超过256K (256 * 1024 Bytes)
  115. local file_size = io.fileSize(file_path)
  116. if file_size < 256 * 1024 then
  117. log.error("exfotawifi", "升级文件大小不足256K,文件大小:", file_size)
  118. return false
  119. end
  120. -- 执行airlink.sfota操作
  121. local result = airlink.sfota(file_path)
  122. if result then
  123. log.info("exfotawifi", "升级成功")
  124. -- 释放文件占用的空间
  125. -- 因为sfota是异步执行的,所以这里不能用os.remove()删除文件
  126. file_path = nil
  127. return true
  128. else
  129. log.error("exfotawifi", "升级失败")
  130. os.remove(file_path)
  131. return false
  132. end
  133. end
  134. function exfotawifi.request()
  135. local result, ip, adapter = sys.waitUntil("IP_READY", 30000)
  136. if result then
  137. log.info("exfotawifi", "开始执行升级任务")
  138. if is_request then
  139. log.warn("exfotawifi", "升级任务正在执行中,请勿重复调用")
  140. return false
  141. end
  142. is_request = true
  143. fota_result = false
  144. -- 构建请求URL
  145. local url = "http://wififota.openluat.com/air8000/update.json"
  146. local imei = is_nil(mobile.imei()) and "未知imei" or mobile.imei()
  147. local version = is_nil(airlink.sver()) and "未知版本" or airlink.sver()
  148. local muid = is_nil(mobile.muid()) and "未知muid" or mobile.muid()
  149. local hw = is_nil(hmeta.hwver()) and "未知硬件版本" or hmeta.hwver()
  150. local request_url = string.format("%s?imei=%s&version=%s&muid=%s&hw=%s", url, imei, version, muid, hw)
  151. log.info("exfotawifi", "正在请求升级信息, URL:", request_url)
  152. -- 发送HTTP请求获取服务器响应
  153. local code, headers, body = http.request("GET", request_url, {}, nil, {timeout = 30000}).wait()
  154. if code == 200 then
  155. log.info("exfotawifi", "获取服务器响应成功")
  156. -- 打印返回的body内容
  157. -- log.info("exfotawifi", "body:", body)
  158. -- 解析服务器响应的json数据
  159. local response = parse_response(body)
  160. if response then
  161. -- 获取服务器返回的版本号和下载链接
  162. local server_version = response.version
  163. local download_url = response.url
  164. -- 获取本地版本号
  165. local local_version = airlink.sver()
  166. -- 判断是否需要升级
  167. if need_fota(local_version, server_version) then
  168. log.info("exfotawifi", "需要升级, 本地版本:", local_version, "服务器版本:", server_version)
  169. -- 下载升级文件
  170. local file_path = download_file(download_url)
  171. if file_path then
  172. -- 开始升级
  173. fota_result = fota_start(file_path)
  174. end
  175. else
  176. log.info("exfotawifi", "当前已是最新WIFI固件")
  177. fota_result = true
  178. end
  179. else
  180. log.error("exfotawifi", "解析服务器响应失败")
  181. end
  182. else
  183. log.error("exfotawifi", "获取服务器响应失败,状态码:", code)
  184. end
  185. else
  186. log.error("当前正在升级WIFI&蓝牙固件,请插入可以上网的SIM卡并重新启动")
  187. end
  188. -- 释放请求标记
  189. is_request = false
  190. return fota_result
  191. end
  192. return exfotawifi