exfotawifi.lua 7.3 KB

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