hot-update.js 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637
  1. import config from "@/utils/config";
  2. /**
  3. * 这里用本地存储做一个非常轻量的“启动期热更新锁”。
  4. *
  5. * 为什么不用纯内存变量:
  6. * 1. App.vue 的 onLaunch 和页面组件 mounted 不在同一个文件里,直接共享内存状态不直观。
  7. * 2. 现有项目已经在登录页、首页挂了旧的整包升级弹窗组件,我们需要一个跨文件、最小改动的互斥标记。
  8. * 3. 用 storage 的好处是组件侧只要读一个 key 就能知道“启动阶段是否已经有热更新流程在跑”。
  9. *
  10. * 注意:
  11. * 这个锁只是为了解决“启动期并发弹窗”的问题,不是分布式锁,也不是强一致锁。
  12. * 对当前这个 uni-app 项目来说,这样的复杂度已经足够,而且侵入性最低。
  13. */
  14. export const HOT_UPDATE_LOCK_KEY = "__APP_WGT_HOT_UPDATE_RUNNING__";
  15. /**
  16. * 当用户点击“稍后再说”时,我们把当前版本记下来。
  17. *
  18. * 这样做的原因:
  19. * 1. 热更新检查放在 onLaunch,意味着每次冷启动都会检查一次。
  20. * 2. 如果用户已经明确拒绝了某个版本,下一次冷启动继续弹同一个版本,体验会比较差。
  21. * 3. 只记录“被跳过的目标版本”,新版本来了仍然会重新提示。
  22. */
  23. const HOT_UPDATE_SKIP_VERSION_KEY = "__APP_WGT_HOT_UPDATE_SKIP_VERSION__";
  24. /**
  25. * 这里单独声明接口地址,而不是直接写死在 uni.request 里。
  26. *
  27. * 这样做的原因:
  28. * 1. 现有项目已经有 config 环境配置,说明项目本身就是按环境切换接口域名的。
  29. * 2. 热更新检查应该沿用现有配置习惯,避免你上线后再去改很多地方。
  30. * 3. 这个路径是本次方案“约定”的服务端接口,你可以直接让后端按这个接口实现,
  31. * 也可以只改这里一处,把路径替换成你们实际的地址。
  32. */
  33. const HOT_UPDATE_API_URL =
  34. config.default.apiUrl + config.default.apiUrlSuffix + "/rq/iot-app/newWgt";
  35. /**
  36. * 读取启动期热更新锁。
  37. *
  38. * 对页面组件的意义:
  39. * 旧的整包升级逻辑挂在页面里,页面 mounted 时如果发现启动热更新还在跑,
  40. * 就应该先让路,避免同时弹两个升级框。
  41. */
  42. export const isHotUpdateRunning = () => {
  43. return uni.getStorageSync(HOT_UPDATE_LOCK_KEY) === "1";
  44. };
  45. /**
  46. * 设置/清理热更新锁。
  47. *
  48. * 输入:
  49. * - running: Boolean,true 表示流程开始,false 表示流程结束。
  50. *
  51. * 输出:
  52. * - 无返回值,副作用是修改 storage 中的互斥标记。
  53. */
  54. const setHotUpdateRunning = (running) => {
  55. if (running) {
  56. uni.setStorageSync(HOT_UPDATE_LOCK_KEY, "1");
  57. } else {
  58. uni.removeStorageSync(HOT_UPDATE_LOCK_KEY);
  59. }
  60. };
  61. /**
  62. * 等待 plus 对象可用。
  63. *
  64. * 为什么即便在 APP-PLUS 里也还要做这一步:
  65. * 1. 你要求热更新接在启动阶段,启动阶段最容易踩到“plus 尚未完全就绪”的时序问题。
  66. * 2. 现有 App.vue 的 onLaunch 里已经直接用了 plus.globalEvent,这说明项目默认运行在 App-Plus 环境。
  67. * 3. 但热更新比普通日志更敏感,因为它要马上调用 plus.runtime.getProperty / install / restart,
  68. * 所以这里显式兜底一次,会比“直接假设 plus 一定存在”更稳。
  69. *
  70. * 成功分支:
  71. * - plus 已就绪,resolve,继续后续热更新流程。
  72. *
  73. * 失败分支:
  74. * - 理论上这里没有主动 reject,超时场景也继续等待;
  75. * 如果 plus 始终不可用,后续逻辑不会开始,也就不会影响原有启动逻辑。
  76. */
  77. const waitForPlusReady = () => {
  78. return new Promise((resolve) => {
  79. if (typeof plus !== "undefined") {
  80. resolve();
  81. return;
  82. }
  83. document.addEventListener(
  84. "plusready",
  85. () => {
  86. resolve();
  87. },
  88. { once: true }
  89. );
  90. });
  91. };
  92. /**
  93. * 使用 plus.runtime.getProperty 读取当前资源包信息。
  94. *
  95. * 这是这次方案里最关键的一步,也是你特别强调不能替换掉的地方。
  96. *
  97. * 为什么必须用它:
  98. * 1. 你要做的是 .wgt 资源热更新,比较的是“当前资源包版本”和“服务端资源包版本”。
  99. * 2. plus.runtime.getProperty(plus.runtime.appid) 返回的是当前应用资源信息,
  100. * 其中 wgtinfo.version 才是最贴近“资源版本”的值。
  101. * 3. plus.runtime.version 更偏运行时/客户端信息,uni.getSystemInfo 也不是这个场景最合适的来源,
  102. * 所以这里严格按你的要求使用 getProperty。
  103. *
  104. * 输出:
  105. * - resolve(wgtInfo)
  106. * 其中最常用的字段包括:
  107. * - appid: 当前应用的 DCloud appid
  108. * - version: 当前资源版本
  109. * - name: 当前应用名称
  110. */
  111. export const getCurrentWgtInfo = () => {
  112. return new Promise((resolve, reject) => {
  113. try {
  114. plus.runtime.getProperty(plus.runtime.appid, (wgtInfo) => {
  115. if (!wgtInfo || !wgtInfo.version) {
  116. reject(new Error("未能获取当前应用资源版本信息"));
  117. return;
  118. }
  119. resolve(wgtInfo);
  120. });
  121. } catch (error) {
  122. reject(error);
  123. }
  124. });
  125. };
  126. /**
  127. * 一个简单的版本比较函数。
  128. *
  129. * 为什么这里仍然保留本地比较,而不是完全信任服务端的 update 字段:
  130. * 1. 服务端当然应该负责判断版本,但客户端再做一次兜底,可以避免后端配置错误导致重复更新。
  131. * 2. 这对“启动期自动更新”尤其重要,因为一旦后端误配,用户会在每次启动都被弹框打扰。
  132. *
  133. * 比较规则:
  134. * - a > b 返回 1
  135. * - a = b 返回 0
  136. * - a < b 返回 -1
  137. */
  138. const compareVersion = (a = "", b = "") => {
  139. const aList = String(a)
  140. .split(".")
  141. .map((item) => Number(item || 0));
  142. const bList = String(b)
  143. .split(".")
  144. .map((item) => Number(item || 0));
  145. const length = Math.max(aList.length, bList.length);
  146. for (let i = 0; i < length; i += 1) {
  147. const aNum = aList[i] || 0;
  148. const bNum = bList[i] || 0;
  149. if (aNum > bNum) return 1;
  150. if (aNum < bNum) return -1;
  151. }
  152. return 0;
  153. };
  154. /**
  155. * 请求服务端检查是否存在新的 .wgt 资源包。
  156. *
  157. * 为什么这里直接使用 uni.request,而不是复用现有 utils/request.js:
  158. * 1. 现有 request 封装默认会带 token、tenant-id、loading、401 刷新令牌等业务逻辑。
  159. * 2. 热更新检查发生在 App 启动最早阶段,不应该依赖登录态,也不应该因为 token 过期影响更新检查。
  160. * 3. 启动期代码越“去业务化”,越不容易被后续鉴权改造牵连。
  161. *
  162. * 请求参数说明:
  163. * - appid: 用来确保服务端下发的是当前应用对应的更新包。
  164. * - version: 当前资源版本,服务端据此判断是否需要更新。
  165. * - platform: 当前平台,便于服务端做 Android / iOS 区分。
  166. * - name: 可选,便于后端日志排查。
  167. *
  168. * 服务端返回格式约定:
  169. * {
  170. * code: 0,
  171. * data: {
  172. * update: true,
  173. * packageType: "wgt",
  174. * version: "1.3.6",
  175. * wgtUrl: "https://example.com/app/1.3.6/update.wgt",
  176. * note: "1. 修复巡检表单保存异常\\n2. 优化首页加载速度",
  177. * force: false
  178. * },
  179. * message: "success"
  180. * }
  181. *
  182. * 字段用途:
  183. * - code: 业务状态码,0 表示接口业务成功。
  184. * - data.update: 服务端是否认为当前客户端需要更新。
  185. * - data.packageType: 更新类型。这里约定 wgt 表示资源热更新,native 表示整包更新。
  186. * - data.version: 服务端最新资源版本。
  187. * - data.wgtUrl: .wgt 下载地址。
  188. * - data.note: 更新说明,弹窗里给用户看。
  189. * - data.force: 是否强制更新。true 时不展示取消按钮。
  190. * - message: 服务端通用提示。
  191. */
  192. const requestHotUpdateInfo = () => {
  193. console.log("11", 11);
  194. return new Promise((resolve, reject) => {
  195. uni.request({
  196. url: HOT_UPDATE_API_URL,
  197. method: "GET",
  198. header: {
  199. "Content-Type": "application/json",
  200. "tenant-id": "1",
  201. },
  202. timeout: 10000,
  203. success: (response) => {
  204. if (response.statusCode !== 200) {
  205. reject(
  206. new Error(
  207. `热更新检查接口请求失败,HTTP 状态码:${response.statusCode}`
  208. )
  209. );
  210. return;
  211. }
  212. resolve(response.data || {});
  213. },
  214. fail: (error) => {
  215. reject(
  216. new Error(
  217. error?.errMsg || "热更新检查接口请求失败,请检查网络或服务端状态"
  218. )
  219. );
  220. },
  221. complete: () => {
  222. /**
  223. * 这里故意不做 UI 处理,只保留 complete 生命周期位置。
  224. *
  225. * 原因:
  226. * 启动期的检查更新是“后台检查”,不是用户手动点按钮触发的请求。
  227. * 如果这里也弹 loading,会和现有登录页/首页首屏体验打架。
  228. */
  229. },
  230. });
  231. });
  232. };
  233. /**
  234. * 规范化服务端返回,统一后续分支判断。
  235. *
  236. * 为什么要单独做这一层:
  237. * 1. 启动流程最怕到处写 if/else,后面维护时很难看出到底在哪个条件提前返回。
  238. * 2. 统一成“shouldUpdate + reason + payload”的结构,后面 orchestration 会清晰很多。
  239. *
  240. * 成功输出示例:
  241. * {
  242. * shouldUpdate: true,
  243. * reason: "has-update",
  244. * payload: { ...服务端 data... }
  245. * }
  246. */
  247. const normalizeUpdateResult = (response, currentVersion) => {
  248. console.log("response", response);
  249. const code = response?.code;
  250. const message = response?.message || response?.msg || "";
  251. const data = response?.data || {};
  252. if (code !== 0) {
  253. return {
  254. shouldUpdate: false,
  255. reason: "server-error",
  256. message: message || "热更新接口返回了业务错误",
  257. };
  258. }
  259. // if (!data.update) {
  260. // return {
  261. // shouldUpdate: false,
  262. // reason: "no-update",
  263. // message: "服务端确认当前版本无需更新",
  264. // };
  265. // }
  266. // if (data.packageType && data.packageType !== "wgt") {
  267. // return {
  268. // shouldUpdate: false,
  269. // reason: "not-wgt",
  270. // message: `服务端返回的是 ${data.packageType} 更新,不属于本次 .wgt 热更新流程`,
  271. // payload: data,
  272. // };
  273. // }
  274. if (!data.version || !data.wgtUrl) {
  275. return {
  276. shouldUpdate: false,
  277. reason: "invalid-data",
  278. message: "服务端返回缺少 version 或 wgtUrl,无法继续热更新",
  279. };
  280. }
  281. if (compareVersion(data.version, currentVersion) <= 0) {
  282. return {
  283. shouldUpdate: false,
  284. reason: "not-newer",
  285. message: "服务端返回的版本不高于当前版本,跳过本次热更新",
  286. payload: data,
  287. };
  288. }
  289. const skippedVersion = uni.getStorageSync(HOT_UPDATE_SKIP_VERSION_KEY);
  290. if (!data.force && skippedVersion === data.version) {
  291. return {
  292. shouldUpdate: false,
  293. reason: "skipped-same-version",
  294. message: "用户之前已经跳过该版本,本次启动不再重复提示",
  295. payload: data,
  296. };
  297. }
  298. return {
  299. shouldUpdate: true,
  300. reason: "has-update",
  301. payload: data,
  302. };
  303. };
  304. /**
  305. * 组装展示给用户的更新文案。
  306. *
  307. * 这里没有引入 i18n,也没有复用页面组件里的 UI。
  308. *
  309. * 原因:
  310. * 1. 热更新发生在 App 启动阶段,应该尽量减少对页面层、全局实例、组件状态的依赖。
  311. * 2. 现有项目的旧升级逻辑是在组件里拿 $t,这条链路不适合直接照搬到启动期。
  312. * 3. 启动期用系统级的 uni.showModal 更稳,也更符合“最小侵入式改造”。
  313. */
  314. const buildUpdateContent = (payload) => {
  315. const note = payload.note
  316. ? String(payload.note)
  317. : "修复已知问题,优化使用体验";
  318. const content = [`发现新的资源版本:${payload.version}`, "", note];
  319. if (payload.force) {
  320. content.push("", "该更新为强制更新,请完成安装后继续使用。");
  321. }
  322. return content.join("\n");
  323. };
  324. /**
  325. * 提示用户是否开始热更新。
  326. *
  327. * 成功分支:
  328. * - 用户点击确认,resolve(true),继续下载。
  329. *
  330. * 失败/取消分支:
  331. * - 非强更时,用户点击取消,记录本次跳过的版本,resolve(false)。
  332. * - 强更时,没有取消按钮,只能确认。
  333. */
  334. const confirmHotUpdate = (payload) => {
  335. return new Promise((resolve) => {
  336. uni.showModal({
  337. title: payload.force ? "发现重要更新" : "发现新版本",
  338. content: buildUpdateContent(payload),
  339. showCancel: !payload.force,
  340. confirmText: "立即更新",
  341. cancelText: "稍后再说",
  342. success: (result) => {
  343. if (result.confirm) {
  344. uni.removeStorageSync(HOT_UPDATE_SKIP_VERSION_KEY);
  345. resolve(true);
  346. return;
  347. }
  348. if (!payload.force && result.cancel) {
  349. uni.setStorageSync(HOT_UPDATE_SKIP_VERSION_KEY, payload.version);
  350. }
  351. resolve(false);
  352. },
  353. fail: () => {
  354. resolve(false);
  355. },
  356. });
  357. });
  358. };
  359. /**
  360. * 下载 .wgt 文件。
  361. *
  362. * 为什么这里要单独拆出来:
  363. * 1. 下载是一个明显独立的异步阶段,最需要看清楚成功和失败分支。
  364. * 2. 下载是启动期流程里耗时较长的一步,单独封装后更方便控制提示方式。
  365. *
  366. * 用户可见表现:
  367. * - 下载中:顶部展示固定 loading 提示,不展示实时百分比。
  368. * - 下载成功:loading 自动关闭,进入安装阶段。
  369. * - 下载失败:loading 关闭,并在上层统一提示“热更新失败”。
  370. *
  371. * 输出:
  372. * - resolve(tempFilePath) 返回下载到本地后的临时文件路径。
  373. */
  374. const downloadWgtPackage = (wgtUrl) => {
  375. return new Promise((resolve, reject) => {
  376. uni.showLoading({
  377. title: "下载更新中",
  378. mask: true,
  379. });
  380. uni.downloadFile({
  381. url: wgtUrl,
  382. timeout: 600000,
  383. success: (response) => {
  384. if (response.statusCode === 200 && response.tempFilePath) {
  385. resolve(response.tempFilePath);
  386. return;
  387. }
  388. reject(
  389. new Error(
  390. `热更新包下载失败,HTTP 状态码:${response.statusCode || "unknown"}`
  391. )
  392. );
  393. },
  394. fail: (error) => {
  395. reject(new Error(error?.errMsg || "热更新包下载失败"));
  396. },
  397. complete: () => {
  398. uni.hideLoading();
  399. },
  400. });
  401. });
  402. };
  403. /**
  404. * 安装下载完成后的 .wgt 包。
  405. *
  406. * 为什么安装时要带 appid:
  407. * 1. 官方文档说明可以传入 appid 做校验。
  408. * 2. 这可以避免服务端配置错误时,把不属于当前应用的资源包安装进来。
  409. *
  410. * 为什么 force 设为 false:
  411. * 1. force=true 会跳过版本校验,风险更高。
  412. * 2. 现在我们已经在服务端和客户端各做了一层版本判断,没有必要强制覆盖旧包。
  413. */
  414. const installWgtPackage = (filePath, appid) => {
  415. return new Promise((resolve, reject) => {
  416. try {
  417. plus.runtime.install(
  418. filePath,
  419. {
  420. appid,
  421. force: false,
  422. },
  423. () => {
  424. resolve();
  425. },
  426. (error) => {
  427. reject(
  428. new Error(
  429. error?.message || `热更新安装失败,错误码:${error?.code || ""}`
  430. )
  431. );
  432. }
  433. );
  434. } catch (error) {
  435. reject(error);
  436. }
  437. });
  438. };
  439. /**
  440. * 安装完成后提示用户并重启应用。
  441. *
  442. * 为什么这里显式调用 plus.runtime.restart:
  443. * 1. .wgt 安装完成只是把新资源放到了运行环境,想让用户真正跑到新代码,还需要重启应用。
  444. * 2. 这一步和旧的页面级整包升级不同,旧逻辑 Android 是直接安装 apk。
  445. * 3. 热更新的目标是“尽快让新前端资源生效”,因此安装完成后立即重启是最直接的方案。
  446. *
  447. * 注意这里在 restart 前先清掉锁:
  448. * - 因为 storage 会跨重启保留。
  449. * - 如果不先清,应用重启后旧组件会误以为“热更新一直在运行中”,造成后续检查失效。
  450. */
  451. const restartAppAfterInstall = (targetVersion) => {
  452. return new Promise((resolve) => {
  453. uni.showModal({
  454. title: "更新完成",
  455. content: `资源版本 ${targetVersion} 已安装完成,点击确定后将立即重启应用使更新生效。`,
  456. showCancel: false,
  457. success: () => {
  458. setHotUpdateRunning(false);
  459. plus.runtime.restart();
  460. resolve();
  461. },
  462. fail: () => {
  463. /**
  464. * 即便弹窗失败,我们也仍然执行重启。
  465. *
  466. * 原因:
  467. * - 到这里说明安装已经完成。
  468. * - 如果不重启,用户还会继续运行旧资源,和“安装成功”的状态不一致。
  469. */
  470. setHotUpdateRunning(false);
  471. plus.runtime.restart();
  472. resolve();
  473. },
  474. });
  475. });
  476. };
  477. /**
  478. * 统一展示“热更新失败”提示。
  479. *
  480. * 为什么不把所有失败都提示给用户:
  481. * 1. 启动阶段的“检查接口失败 / 无网络”很常见,如果每次启动都弹错,会严重影响体验。
  482. * 2. 所以我们只对“用户已经明确点击开始更新之后”的失败给出弹窗提示。
  483. * 3. 纯检查阶段失败,记录日志后继续进入原有流程即可。
  484. */
  485. const showHotUpdateError = (error, title = "更新失败") => {
  486. uni.showModal({
  487. title,
  488. content: error?.message || "热更新执行失败,请稍后重试",
  489. showCancel: false,
  490. });
  491. };
  492. /**
  493. * 对外暴露的启动期热更新入口。
  494. *
  495. * 这是你后续在 App.vue onLaunch 里真正调用的函数。
  496. *
  497. * 整体执行顺序:
  498. * 1. 等待 plus 就绪
  499. * 2. 获取当前资源版本
  500. * 3. 请求服务端检查更新
  501. * 4. 判断是否存在新的 .wgt 包
  502. * 5. 提示用户确认
  503. * 6. 下载 .wgt
  504. * 7. 安装 .wgt
  505. * 8. 重启应用
  506. *
  507. * 返回值:
  508. * - Promise<{ status: string, ... }>
  509. * 主要用于日志或后续扩展,目前 App.vue 不需要依赖它的返回值。
  510. */
  511. export const runWgtHotUpdate = async () => {
  512. // #ifdef APP-PLUS
  513. if (isHotUpdateRunning()) {
  514. return {
  515. status: "locked",
  516. message: "已有热更新流程在执行,跳过重复检查",
  517. };
  518. }
  519. let currentStage = "prepare";
  520. setHotUpdateRunning(true);
  521. try {
  522. currentStage = "plus-ready";
  523. await waitForPlusReady();
  524. currentStage = "read-local-version";
  525. const wgtInfo = await getCurrentWgtInfo();
  526. currentStage = "request-server";
  527. const response = await requestHotUpdateInfo();
  528. const updateResult = normalizeUpdateResult(response, wgtInfo.version);
  529. if (!updateResult.shouldUpdate) {
  530. return {
  531. status: updateResult.reason,
  532. message: updateResult.message,
  533. data: updateResult.payload || null,
  534. };
  535. }
  536. currentStage = "confirm";
  537. const confirmed = await confirmHotUpdate(updateResult.payload);
  538. if (!confirmed) {
  539. return {
  540. status: "cancelled",
  541. message: "用户取消本次热更新",
  542. };
  543. }
  544. currentStage = "download";
  545. const filePath = await downloadWgtPackage(updateResult.payload.wgtUrl);
  546. currentStage = "install";
  547. await installWgtPackage(filePath, wgtInfo.appid);
  548. currentStage = "restart";
  549. await restartAppAfterInstall(updateResult.payload.version);
  550. return {
  551. status: "restarted",
  552. message: "热更新安装完成,应用已触发重启",
  553. };
  554. } catch (error) {
  555. console.error("[hot-update] 执行失败:", currentStage, error);
  556. if (currentStage === "download") {
  557. showHotUpdateError(error, "下载失败");
  558. } else if (currentStage === "install" || currentStage === "restart") {
  559. showHotUpdateError(error, "安装失败");
  560. } else {
  561. /**
  562. * 检查阶段失败只打日志,不主动打扰用户。
  563. *
  564. * 为什么这么处理:
  565. * - 当前项目 onLaunch 里还有数据库初始化、url scheme 参数接收等逻辑。
  566. * - 热更新检查只是“加分项”,不应该因为检查失败阻断原有启动流程。
  567. * - 这样最符合“最小侵入式改造”的目标。
  568. */
  569. }
  570. return {
  571. status: "error",
  572. stage: currentStage,
  573. error,
  574. };
  575. } finally {
  576. /**
  577. * 正常情况下:
  578. * - 无更新 / 用户取消 / 下载失败 / 安装失败,都会走到这里清锁。
  579. * - 安装成功后会在 restart 之前先清锁,所以这里再次清理也没有副作用。
  580. */
  581. setHotUpdateRunning(false);
  582. }
  583. // #endif
  584. return {
  585. status: "skip-non-app-plus",
  586. message: "当前不是 APP-PLUS 环境,跳过 .wgt 热更新检查",
  587. };
  588. };