openclaw-multi-auto 1.0.9 → 1.1.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (490) hide show
  1. package/dist/{accounts-C_lW3Ag9.js → accounts-BOzyfwW4.js} +2 -2
  2. package/dist/{accounts-BVgYdU9W.js → accounts-BtEMxtPK.js} +1 -1
  3. package/dist/{accounts-BU-CeDai.js → accounts-CUYZBSKh.js} +1 -1
  4. package/dist/{accounts-C9HcPI9h.js → accounts-L9ByEpnP.js} +2 -2
  5. package/dist/{accounts-DJaYqD2E.js → accounts-khXX75l1.js} +7 -7
  6. package/dist/{accounts-Tgelvk0C.js → accounts-yfBeCZtS.js} +17 -17
  7. package/dist/{acp-cli-oWFHnS7i.js → acp-cli-CA2oCCEA.js} +8 -8
  8. package/dist/{active-listener-BEdprTkn.js → active-listener-D1yqT1cw.js} +2 -2
  9. package/dist/{agent-scope-BRElciAf.js → agent-scope-NQ9CtXYN.js} +17 -17
  10. package/dist/{agents-CujsWz9d.js → agents-BWVlofyv.js} +14 -14
  11. package/dist/{agents.config-C42STger.js → agents.config-QVfqc4C-.js} +2 -2
  12. package/dist/{api-key-rotation-BJpKWXy0.js → api-key-rotation-DtsNS2Nb.js} +2 -2
  13. package/dist/{api-key-rotation-BbuLHl0_.js → api-key-rotation-Vix80AUw.js} +1 -1
  14. package/dist/{audio-preflight-Yg1vzPE1.js → audio-preflight-DUtdCXjJ.js} +34 -34
  15. package/dist/{audio-preflight-C58LeYZM.js → audio-preflight-DpxQCpsA.js} +32 -32
  16. package/dist/{audio-transcription-runner-B8jbozH5.js → audio-transcription-runner-28fcRNNi.js} +12 -12
  17. package/dist/{audio-transcription-runner-D1cvrZ6s.js → audio-transcription-runner-DYKvqK54.js} +23 -23
  18. package/dist/{audit-membership-runtime-Dntemq07.js → audit-membership-runtime-DWyHWAHM.js} +4 -4
  19. package/dist/{audit-DtlAv66L.js → audit-oC--RZy0.js} +29 -29
  20. package/dist/{auth-Mj21c_GN.js → auth-YB6m93-M.js} +1 -1
  21. package/dist/{auth-choice-CEycltU4.js → auth-choice-CsSQLEkP.js} +13 -13
  22. package/dist/{auth-choice-CGHVedXa.js → auth-choice-gp-h1aBd.js} +11 -11
  23. package/dist/{auth-choice.apply-helpers-D4YQXYfc.js → auth-choice.apply-helpers-CPpuynsT.js} +1 -1
  24. package/dist/{auth-profiles-CdLTlJLc.js → auth-profiles-15pq9j9V.js} +16 -16
  25. package/dist/{auth-token-7kzDLVhb.js → auth-token-Bwn8N6KA.js} +1 -1
  26. package/dist/{banner-CpzKVd4-.js → banner-hcZ0XNXv.js} +2 -2
  27. package/dist/{bonjour-discovery-BuS9wF_p.js → bonjour-discovery-2btw06uD.js} +1 -1
  28. package/dist/{browser-cli-BVtbvQ3Z.js → browser-cli-C4PdLYHy.js} +12 -12
  29. package/dist/build-info.json +3 -3
  30. package/dist/bundled/boot-md/handler.js +51 -51
  31. package/dist/bundled/bootstrap-extra-files/handler.js +6 -6
  32. package/dist/bundled/command-logger/handler.js +2 -2
  33. package/dist/bundled/session-memory/handler.js +51 -51
  34. package/dist/{call-Dp9SDkqK.js → call-CZgRbVYm.js} +10 -10
  35. package/dist/canvas-host/a2ui/.bundle.hash +1 -1
  36. package/dist/{channel-account-context-ESLVGdUf.js → channel-account-context-BYiOuxnf.js} +5 -5
  37. package/dist/{channel-activity-LfmEkdDN.js → channel-activity-D9K6yeu8.js} +1 -1
  38. package/dist/{channel-activity-BDnjYF7B.js → channel-activity-xHOMiarp.js} +3 -3
  39. package/dist/{channel-options-CsYaFB_4.js → channel-options-DJkf_8wi.js} +3 -3
  40. package/dist/{channel-selection-DFJsYFPo.js → channel-selection-BpIsCVV5.js} +1 -1
  41. package/dist/{channel-web-DzRsJEE4.js → channel-web-wb1My1Yi.js} +17 -17
  42. package/dist/{channels-cli-CpfeOXtJ.js → channels-cli-DlQONtY9.js} +93 -93
  43. package/dist/{channels-status-issues-6V9ktlZD.js → channels-status-issues-Hj-GOSVm.js} +1 -1
  44. package/dist/{chrome-CRBG2YP_.js → chrome-DwizpzOC.js} +26 -26
  45. package/dist/{chrome-CyM61Cn2.js → chrome-L_icBVLq.js} +4 -4
  46. package/dist/{clawbot-cli-COtnakIJ.js → clawbot-cli-CyTZUVIJ.js} +11 -11
  47. package/dist/cli/daemon-cli.js +1 -1
  48. package/dist/{cli-jnT1hYPO.js → cli-CNTrB-ni.js} +73 -73
  49. package/dist/{client-C-wA75vx.js → client-g0366Z73.js} +2 -2
  50. package/dist/{command-registry-g5tDqdAU.js → command-registry-VOmumlmD.js} +11 -11
  51. package/dist/{command-secret-targets-u3eWn-W3.js → command-secret-targets-CIMuIdbL.js} +4 -4
  52. package/dist/{commands-C8pWcFs_.js → commands-DU7At3rY.js} +1 -1
  53. package/dist/{commands-registry-J6nVL3BQ.js → commands-registry-B_WUa9KB.js} +3 -3
  54. package/dist/{commands-registry-D5qXbFJn.js → commands-registry-V1zZ5pPC.js} +4 -4
  55. package/dist/{completion-cli-DQsyli3B.js → completion-cli-pqdmKVTo.js} +13 -13
  56. package/dist/{config-cli-DUycVHoC.js → config-cli-BqznmzlM.js} +7 -7
  57. package/dist/{config-guard--Obk2MGi.js → config-guard-D8jiLjr7.js} +3 -3
  58. package/dist/{config-validation-iOxszdEd.js → config-validation-BngLiGoq.js} +3 -3
  59. package/dist/{configure-Al_pz_5z.js → configure-B--Umf_O.js} +17 -17
  60. package/dist/{control-ui-assets-DpjfNcTC.js → control-ui-assets-C9ewbI8z.js} +1 -1
  61. package/dist/{cron-cli-CQ5EZw_b.js → cron-cli-DCRPpI1m.js} +11 -11
  62. package/dist/{daemon-cli-Bb80Mzv5.js → daemon-cli-DtBzKM_d.js} +15 -15
  63. package/dist/{daemon-install-B09ifLcJ.js → daemon-install-ckPBUAjJ.js} +4 -4
  64. package/dist/{daemon-install-helpers-CYpECiEx.js → daemon-install-helpers-BWf-PrgA.js} +11 -11
  65. package/dist/{deliver-ChbFOwjw.js → deliver-D4o6VIur.js} +21 -21
  66. package/dist/{deliver-CxF9h9PE.js → deliver-b93aBgie.js} +7 -7
  67. package/dist/deliver-runtime-CdWd-JQI.js +61 -0
  68. package/dist/deliver-runtime-P-G3bPjW.js +36 -0
  69. package/dist/deps-send-discord.runtime-BltWu9GW.js +36 -0
  70. package/dist/deps-send-discord.runtime-DnbhTFX9.js +26 -0
  71. package/dist/deps-send-imessage.runtime-BOiQ6mDx.js +25 -0
  72. package/dist/deps-send-imessage.runtime-CdCif3t7.js +35 -0
  73. package/dist/deps-send-signal.runtime-CTcl388M.js +24 -0
  74. package/dist/deps-send-signal.runtime-fHqkSmMW.js +34 -0
  75. package/dist/deps-send-slack.runtime-CCqBz4Kg.js +22 -0
  76. package/dist/deps-send-slack.runtime-CkyZO7ln.js +32 -0
  77. package/dist/{deps-send-telegram.runtime-B6wp3kx8.js → deps-send-telegram.runtime-C--B69CN.js} +16 -16
  78. package/dist/deps-send-telegram.runtime-DGSKTCpH.js +27 -0
  79. package/dist/deps-send-whatsapp.runtime-BkoMLlCM.js +119 -0
  80. package/dist/deps-send-whatsapp.runtime-CJkTHkah.js +60 -0
  81. package/dist/{devices-cli-C0dUbeL4.js → devices-cli-D6V059Dh.js} +8 -8
  82. package/dist/{diagnostic-Co6Kghr-.js → diagnostic-Bn4PZjMZ.js} +2 -2
  83. package/dist/{diagnostic-BfiudAAN.js → diagnostic-BpP9UBCE.js} +1 -1
  84. package/dist/{diagnostics-DCWM_8Ur.js → diagnostics---0c2_jo.js} +5 -5
  85. package/dist/{directory-cli-Dlws13Di.js → directory-cli-RYC34-EU.js} +7 -7
  86. package/dist/{dns-cli-But7QdAw.js → dns-cli-CElNoV5S.js} +5 -5
  87. package/dist/{dock-BbUkruOF.js → dock-nA2DVPCV.js} +4 -4
  88. package/dist/{docs-cli-Ck2IKAtw.js → docs-cli-Bv7ltPvi.js} +4 -4
  89. package/dist/{doctor-completion-Nat1HXE1.js → doctor-completion-CQIiN7rY.js} +2 -2
  90. package/dist/{doctor-config-flow-WCCoUXeB.js → doctor-config-flow-BOAryh5P.js} +15 -15
  91. package/dist/{enable-C3H8BtN4.js → enable-688HYBNS.js} +1 -1
  92. package/dist/entry.js +2 -2
  93. package/dist/{errors-xt401nuk.js → errors-CCLeFWAg.js} +1 -1
  94. package/dist/{exec-approvals-allowlist-ySf2Yo5n.js → exec-approvals-allowlist-C0Eya3rT.js} +1 -1
  95. package/dist/{exec-approvals-cli-DhLLOys6.js → exec-approvals-cli-BYYerKp8.js} +16 -16
  96. package/dist/{exec-safe-bin-runtime-policy-DnXViOlF.js → exec-safe-bin-runtime-policy-BmJmfDRo.js} +2 -2
  97. package/dist/{fetch-D-SiVqBm.js → fetch-BhAHVdwx.js} +3 -3
  98. package/dist/{fetch-DuraYswo.js → fetch-BlJWzEP6.js} +5 -5
  99. package/dist/{fetch-guard-CKYBfJ66.js → fetch-guard-BbBdhWz_.js} +1 -1
  100. package/dist/{fetch-guard-DWr0d00H.js → fetch-guard-ChYBwfiy.js} +2 -2
  101. package/dist/{frontmatter-BkTfEZ93.js → frontmatter-CvaMP376.js} +3 -3
  102. package/dist/{fs-safe-CTYUrIgQ.js → fs-safe-0jAo_Whb.js} +4 -4
  103. package/dist/{fs-safe-jMDpsYew.js → fs-safe-CFLs-j60.js} +24 -24
  104. package/dist/{gateway-cli--atF6LYo.js → gateway-cli-DOYWSRqa.js} +153 -153
  105. package/dist/{gateway-rpc-DYyQQ1z9.js → gateway-rpc-w6t8Eq_1.js} +1 -1
  106. package/dist/{github-copilot-token-BDioPmd6.js → github-copilot-token-D13V9YBz.js} +7 -7
  107. package/dist/{health-BDJ72U4Z.js → health-C1EstYHd.js} +11 -11
  108. package/dist/{hooks-cli-r0UX8B9a.js → hooks-cli-BFMve5sG.js} +81 -81
  109. package/dist/{hooks-status-CxBdpBry.js → hooks-status-DBnEWOEP.js} +1 -1
  110. package/dist/{image-C7RmnYxa.js → image-Bbn53mzj.js} +6 -6
  111. package/dist/{image-CyHTO86Q.js → image-Bw71y73Q.js} +5 -5
  112. package/dist/{image-ops-pjs5W0CZ.js → image-ops-AJL9tN3_.js} +10 -10
  113. package/dist/{image-ops-BuUnEOE0.js → image-ops-CehkHxmW.js} +2 -2
  114. package/dist/image-runtime-6jhrqcle.js +55 -0
  115. package/dist/image-runtime-CVv2ra9J.js +29 -0
  116. package/dist/{inspect-BeU4yMp2.js → inspect-CUxeDA8c.js} +4 -4
  117. package/dist/{install-safe-path-BiL8OJvK.js → install-safe-path-BX58wFBl.js} +25 -25
  118. package/dist/{installs-BwKmG0Uy.js → installs-B-xr0Fzq.js} +9 -9
  119. package/dist/{ipv4-BoGwfnEw.js → ipv4-C4Yt-xid.js} +1 -1
  120. package/dist/{ir-B83looB-.js → ir-DAP-B-Xw.js} +8 -8
  121. package/dist/{ir-Da7Ahsqc.js → ir-otKVkb4a.js} +8 -8
  122. package/dist/{issue-format-CJ89_-9v.js → issue-format-DSksfKiV.js} +1 -1
  123. package/dist/{json-files-CuJjdF_0.js → json-files-rR19q30D.js} +8 -8
  124. package/dist/{legacy-names-DOC03BkU.js → legacy-names-TyzbVqa_.js} +1 -1
  125. package/dist/{lifecycle-core-BzuWBBm8.js → lifecycle-core-DVwen0ks.js} +5 -5
  126. package/dist/llm-slug-generator.js +51 -51
  127. package/dist/{logger-BfjWMCSD.js → logger-DMZQQtxK.js} +7 -7
  128. package/dist/{login-VkQ9MW3d.js → login-BcRaGQb6.js} +3 -3
  129. package/dist/{login-CrIwcrVI.js → login-DiCctRo1.js} +5 -5
  130. package/dist/{login-qr-BVMQ-xQz.js → login-qr-C9D47WEZ.js} +6 -6
  131. package/dist/{login-qr-BpPDZdl_.js → login-qr-MUbXgjtd.js} +10 -10
  132. package/dist/{logs-cli-DWHhSnWs.js → logs-cli-B3TG2lp8.js} +9 -9
  133. package/dist/{manager-1bvuGrNR.js → manager-BW_NSIMl.js} +13 -13
  134. package/dist/{manager-BUBwl6M6.js → manager-iwcRf3Xc.js} +14 -14
  135. package/dist/manager-runtime-BN6VevdC.js +18 -0
  136. package/dist/{manager-runtime-H9iQnw64.js → manager-runtime-D3KqpRT7.js} +9 -9
  137. package/dist/{manifest-registry-DVviqWVY.js → manifest-registry-CJb8odRF.js} +1 -1
  138. package/dist/{memory-cli-DarWCbU6.js → memory-cli-DGsJx7YM.js} +12 -12
  139. package/dist/{model-BLV8y_N-.js → model-B792l3Cn.js} +2 -2
  140. package/dist/{model-catalog-DTcBkUlX.js → model-catalog-DZ6kl2Ko.js} +3 -3
  141. package/dist/{model-picker-D-fTOOzb.js → model-picker-6fHys0wK.js} +4 -4
  142. package/dist/{model-selection-Dna0Gz1k.js → model-selection-idoqPmw0.js} +43 -43
  143. package/dist/{models-CmjutMmh.js → models-DerrMmwF.js} +17 -17
  144. package/dist/{models-cli-BZFQX22p.js → models-cli-BkTZg_4v.js} +78 -78
  145. package/dist/{models-config-DNcDbV_G.js → models-config-Dabs9Kdv.js} +6 -6
  146. package/dist/{net-D5fSREu4.js → net-BAIqYNz0.js} +2 -2
  147. package/dist/{node-cli-Cnq-PJTp.js → node-cli-DoEq4zeG.js} +33 -33
  148. package/dist/{node-command-policy-B5BMBBJP.js → node-command-policy-DRhSc90G.js} +1 -1
  149. package/dist/{node-service-BpYZAvpl.js → node-service-CcgtNIeT.js} +1 -1
  150. package/dist/{nodes-cli-CkrDOWpv.js → nodes-cli--d9RiCnK.js} +16 -16
  151. package/dist/{nodes-screen-DfsQohWd.js → nodes-screen-BTND5VDq.js} +7 -7
  152. package/dist/{npm-pack-install-E-mkrZ55.js → npm-pack-install-CGUDOSKz.js} +18 -18
  153. package/dist/{npm-resolution-BUUmg5ON.js → npm-resolution-CAoULCWM.js} +4 -4
  154. package/dist/{onboard-DINGSS9R.js → onboard-CMZRkPVc.js} +6 -6
  155. package/dist/{onboard-channels-Di4MjWDY.js → onboard-channels-SlJu6XWJ.js} +21 -21
  156. package/dist/{onboard-custom-DITBQuWQ.js → onboard-custom-EzfL89yz.js} +4 -4
  157. package/dist/{onboard-helpers-DO7u2ZOy.js → onboard-helpers-lGpaVp03.js} +10 -10
  158. package/dist/{onboard-hooks-K02_ZWDq.js → onboard-hooks-B7TMiJoY.js} +4 -4
  159. package/dist/{onboard-remote-5-ugiSN2.js → onboard-remote-nvWS7on7.js} +4 -4
  160. package/dist/{onboard-skills-CSzbC2KZ.js → onboard-skills-hz91W03u.js} +4 -4
  161. package/dist/{onboarding-BM4dvUK6.js → onboarding-CTPuMVQI.js} +14 -14
  162. package/dist/{onboarding.finalize-p_RDh7ET.js → onboarding.finalize-DLzl5ZOg.js} +87 -87
  163. package/dist/{onboarding.gateway-config-DldlRfms.js → onboarding.gateway-config-C4MoTr60.js} +18 -18
  164. package/dist/{onboarding.secret-input-BIRIJiCU.js → onboarding.secret-input-BFD0OW4X.js} +1 -1
  165. package/dist/{openai-model-default-CY2Nk4cn.js → openai-model-default-h4LbSR7f.js} +2 -2
  166. package/dist/{openclaw-root-BFfBQ6FD.js → openclaw-root-BU3lu8pM.js} +8 -8
  167. package/dist/{outbound-ChDjtuD6.js → outbound-C2kanETZ.js} +6 -6
  168. package/dist/{outbound-attachment-DfWsfe2N.js → outbound-attachment-CAJBGcna.js} +2 -2
  169. package/dist/{outbound-attachment-DqHlD21U.js → outbound-attachment-DBrYWX8h.js} +2 -2
  170. package/dist/{outbound-DPdJe7e1.js → outbound-kbHYt1JW.js} +3 -3
  171. package/dist/{pairing-cli-D2VZxWg3.js → pairing-cli-CXbmkz7Z.js} +8 -8
  172. package/dist/{pairing-labels-D4rnJ5pJ.js → pairing-labels-CaPLIhlh.js} +1 -1
  173. package/dist/{pairing-store-BXArq4hn.js → pairing-store-Brs9aNn_.js} +3 -3
  174. package/dist/{path-alias-guards-DbNvNQar.js → path-alias-guards-DhIwq92y.js} +3 -3
  175. package/dist/{path-alias-guards-BzvdLvTI.js → path-alias-guards-DqXRZmsL.js} +1 -1
  176. package/dist/{path-safety-ClQO4BB6.js → path-safety-BjIM4N4t.js} +1 -1
  177. package/dist/{paths-Cvc9EM8Y.js → paths-C6TxBCvO.js} +5 -5
  178. package/dist/{paths-Bkr-BCxW.js → paths-CCxysrzL.js} +4 -4
  179. package/dist/{paths-BB_1ZWOj.js → paths-Cv63xST_.js} +9 -9
  180. package/dist/{pi-embedded-BC_GWGuw.js → pi-embedded-BaGj07T0.js} +167 -167
  181. package/dist/{pi-embedded-helpers-oXDyXTD8.js → pi-embedded-helpers-DC2OtKrl.js} +6 -6
  182. package/dist/{pi-embedded-helpers-CbIShbOM.js → pi-embedded-helpers-wy0DZvx1.js} +52 -52
  183. package/dist/{pi-model-discovery-Dymwdjt0.js → pi-model-discovery-BGgOlX8N.js} +7 -7
  184. package/dist/{pi-model-discovery-CECkJMCt.js → pi-model-discovery-DOb5RTev.js} +1 -1
  185. package/dist/pi-model-discovery-runtime-Bwmi4Ev8.js +11 -0
  186. package/dist/{pi-model-discovery-runtime-g7UP-SFR.js → pi-model-discovery-runtime-DShjmiiF.js} +5 -5
  187. package/dist/{pi-tools.before-tool-call.runtime-Df7B-ggW.js → pi-tools.before-tool-call.runtime-B079pVah.js} +5 -5
  188. package/dist/{pi-tools.before-tool-call.runtime-Cwab_5W1.js → pi-tools.before-tool-call.runtime-BuLxSyx9.js} +9 -9
  189. package/dist/{pi-tools.policy-CydUEzFi.js → pi-tools.policy-DZ-X86Va.js} +5 -5
  190. package/dist/{plugin-auto-enable-BE4ZVjAL.js → plugin-auto-enable-BCMqEDjQ.js} +3 -3
  191. package/dist/{plugin-registry-DXW3eyib.js → plugin-registry-BPlCWMur.js} +3 -3
  192. package/dist/plugin-sdk/discord.js +6 -6
  193. package/dist/{plugins-4Rj4OjLY.js → plugins-CWkRQYDj.js} +11 -11
  194. package/dist/{plugins-VAZrrfgw.js → plugins-DCxT-37x.js} +2 -2
  195. package/dist/{plugins-cli-BQJOOVMx.js → plugins-cli-D9ILEfyb.js} +83 -83
  196. package/dist/{ports-DMkRSlnH.js → ports-BlCLhwbc.js} +1 -1
  197. package/dist/{ports-DogAV7pa.js → ports-CER5YPnN.js} +2 -2
  198. package/dist/{program-BA11qFx1.js → program-CV56xn7w.js} +81 -81
  199. package/dist/{prompt-select-styled-B0GS28ia.js → prompt-select-styled-BOa5I_PU.js} +40 -40
  200. package/dist/{provider-auth-helpers-CrGL-jik.js → provider-auth-helpers-c6mNCUXO.js} +5 -5
  201. package/dist/{proxy-env-wKO3g8Yv.js → proxy-env-BMrSVckF.js} +1 -1
  202. package/dist/{proxy-env-DlmzDx8x.js → proxy-env-Cq5gdrbj.js} +1 -1
  203. package/dist/{proxy-fetch-B2pEfjbR.js → proxy-fetch-CCjEYbFm.js} +1 -1
  204. package/dist/{push-apns-BEwBjZ0a.js → push-apns-FOkPD05E.js} +5 -5
  205. package/dist/{pw-ai-D2wlMJtN.js → pw-ai-Cl1Lc7RC.js} +14 -14
  206. package/dist/{pw-ai-CnbxziFP.js → pw-ai-CxBU3aK5.js} +18 -18
  207. package/dist/{qmd-manager-BtIKUaO9.js → qmd-manager-BsYsO9Ii.js} +10 -10
  208. package/dist/{qmd-manager-Dp6PJ8zQ.js → qmd-manager-DEJMqoGd.js} +20 -20
  209. package/dist/{qr-cli-CFz9kS5X.js → qr-cli-BD2jK4fg.js} +2 -2
  210. package/dist/{query-expansion-BrSWVbaE.js → query-expansion-1UTIWjP6.js} +12 -12
  211. package/dist/{query-expansion-CX-1fS52.js → query-expansion-DtLc3wjL.js} +6 -6
  212. package/dist/{redact-COik8ET1.js → redact-ClbcYG1J.js} +1 -1
  213. package/dist/{redact-snapshot-Bs4goggz.js → redact-snapshot-D_qQD4A-.js} +1 -1
  214. package/dist/{register.agent-D641ju8B.js → register.agent-tbPA5YAy.js} +94 -94
  215. package/dist/register.configure-DuwRrXc2.js +165 -0
  216. package/dist/{register.maintenance-BFkk8MEH.js → register.maintenance-DrqDlV5b.js} +95 -95
  217. package/dist/{register.message-D5uE_Hop.js → register.message-CoKXNaU0.js} +74 -74
  218. package/dist/{register.onboard-zHFvSwFr.js → register.onboard-CSWOSL9O.js} +18 -18
  219. package/dist/{register.setup-uTbv3_P1.js → register.setup-DBx5JX6h.js} +21 -21
  220. package/dist/{register.status-health-sessions-CB7t-JQx.js → register.status-health-sessions-D1bPtfep.js} +88 -88
  221. package/dist/{register.subclis-MEiNmuy5.js → register.subclis-QGJNmjss.js} +31 -31
  222. package/dist/{rpc-DVfuVmy9.js → rpc-DkMrTUww.js} +1 -1
  223. package/dist/{run-main-mrvunzuy.js → run-main-CPftxqTe.js} +92 -92
  224. package/dist/{run-with-concurrency-BgYfgkXT.js → run-with-concurrency-D_ZpbgEG.js} +4 -4
  225. package/dist/{runtime-C87FQrrv.js → runtime-MAH2Oph4.js} +3 -3
  226. package/dist/{runtime-config-collectors-BQaC477D.js → runtime-config-collectors-BJMV6Mt1.js} +1 -1
  227. package/dist/{runtime-whatsapp-login.runtime-DjdgScUI.js → runtime-whatsapp-login.runtime-CGYWl9eB.js} +7 -7
  228. package/dist/runtime-whatsapp-login.runtime-IeylZEl4.js +13 -0
  229. package/dist/runtime-whatsapp-outbound.runtime-ClBRuLsq.js +22 -0
  230. package/dist/{runtime-whatsapp-outbound.runtime-B74K7opl.js → runtime-whatsapp-outbound.runtime-hZEfYaRM.js} +15 -15
  231. package/dist/{sandbox-Ct-_lzi1.js → sandbox-0TbzPJaS.js} +18 -18
  232. package/dist/{sandbox-cli-C3RoXFRE.js → sandbox-cli-NLaxffXl.js} +25 -25
  233. package/dist/{secrets-cli-B3hqCxvs.js → secrets-cli-KLGbYet6.js} +11 -11
  234. package/dist/{security-cli-DcHIyXr7.js → security-cli-EIEkcYVb.js} +42 -42
  235. package/dist/{send-dfu6_rgf.js → send-BQERFNyo.js} +5 -5
  236. package/dist/{send-CnRP4P-G.js → send-BW-ZtYG3.js} +5 -5
  237. package/dist/{send-L7gRiwyd.js → send-Bj776ESJ.js} +7 -7
  238. package/dist/{send-2zKwf9NW.js → send-BtZAqquW.js} +11 -11
  239. package/dist/{send-DAQAKa9Z.js → send-CxgWxXZc.js} +6 -6
  240. package/dist/{send-5o2p_xjn.js → send-DAMtu9kK.js} +4 -4
  241. package/dist/{send-PE6cwoTe.js → send-DcxmcFi_.js} +8 -8
  242. package/dist/{send-6lz6rNVP.js → send-Dx2RkUOZ.js} +6 -6
  243. package/dist/{send-u1Bo4CSn.js → send-sC6ka831.js} +8 -8
  244. package/dist/{send-BHTiZcH3.js → send-vmONuVgL.js} +26 -26
  245. package/dist/{server-context-8pDe2iyd.js → server-context-HJVwPQYn.js} +12 -12
  246. package/dist/{server-Ci4xtuR9.js → server-kUElNhlY.js} +20 -20
  247. package/dist/{server-lifecycle-k5daSrde.js → server-lifecycle-BE32unpZ.js} +2 -2
  248. package/dist/{server-middleware-DMiFT9xU.js → server-middleware-AS2VOYkK.js} +1 -1
  249. package/dist/{server-node-events-BmnPjNXE.js → server-node-events-DkbZzI6P.js} +74 -74
  250. package/dist/{service-BhOFtHSw.js → service-CyStNr3d.js} +15 -15
  251. package/dist/{session-D8ImowSs.js → session-A4QhBRvH.js} +8 -8
  252. package/dist/{session-DkOjpX3_.js → session-CaCx4rPH.js} +1 -1
  253. package/dist/{session-utils-DKRmXD2l.js → session-utils-CGqb1oeq.js} +6 -6
  254. package/dist/{sessions-svLGrv0Z.js → sessions-BV6HNW4h.js} +4 -4
  255. package/dist/{sessions-BOWPuhe5.js → sessions-BmVDW-7q.js} +15 -15
  256. package/dist/{shared-CjuadLFV.js → shared-DUQavBtY.js} +3 -3
  257. package/dist/{shared-CbAkLNrg.js → shared-JW74idb0.js} +1 -1
  258. package/dist/{skill-commands-DNqJ-kwn.js → skill-commands-CMzBZKG2.js} +9 -9
  259. package/dist/{skill-commands-TpUsdjev.js → skill-commands-WtIJG0CI.js} +5 -5
  260. package/dist/{skill-scanner-BZvvItef.js → skill-scanner-DIFsGcqE.js} +6 -6
  261. package/dist/{skills-DR-vacol.js → skills-7T9PwwL6.js} +3 -3
  262. package/dist/{skills-7ODkHQYp.js → skills-CE_iqvM5.js} +22 -22
  263. package/dist/{skills-cli-oay0tY8Z.js → skills-cli-DPavvthL.js} +5 -5
  264. package/dist/{skills-install-Vmi7xYfa.js → skills-install-BoLfaoWv.js} +6 -6
  265. package/dist/{skills-status-BmN697ff.js → skills-status-CK5Gnf6i.js} +1 -1
  266. package/dist/{slash-commands.runtime-BfaheruW.js → slash-commands.runtime-CMGx2xHy.js} +11 -11
  267. package/dist/slash-commands.runtime-Cpn2tYW4.js +16 -0
  268. package/dist/slash-dispatch.runtime-BAeJXa56.js +114 -0
  269. package/dist/slash-dispatch.runtime-DoBAQBU5.js +56 -0
  270. package/dist/{slash-skill-commands.runtime-D34BKAN-.js → slash-skill-commands.runtime-ChU2tck2.js} +15 -15
  271. package/dist/slash-skill-commands.runtime-DKMvvdDW.js +20 -0
  272. package/dist/{status-BcQchPaC.js → status-Bp-K1BEf.js} +27 -27
  273. package/dist/{status.update-B20UBTDq.js → status.update-ZYUSggzS.js} +2 -2
  274. package/dist/{store-D89wDcz9.js → store--eR1R_UX.js} +2 -2
  275. package/dist/{store-DDkqo1sO.js → store-DeASfYEV.js} +5 -5
  276. package/dist/{subagent-registry-CPxHbyN5.js → subagent-registry-8qHIVhRq.js} +149 -149
  277. package/dist/subagent-registry-runtime-DVomlbm6.js +114 -0
  278. package/dist/subagent-registry-runtime-ppWS3tVu.js +56 -0
  279. package/dist/{subsystem-B45WV3qB.js → subsystem-Di1z8l0Z.js} +14 -14
  280. package/dist/{system-cli-D2yIJoKU.js → system-cli-C2xNfuQM.js} +9 -9
  281. package/dist/{system-run-command-BmhbnLTE.js → system-run-command-BPWZk7KI.js} +1 -1
  282. package/dist/{systemd-DjWVSbAG.js → systemd-Cf-0XKYu.js} +9 -9
  283. package/dist/{systemd-hints-Do-aQ9jw.js → systemd-hints-CYllYKO0.js} +6 -6
  284. package/dist/{systemd-linger-DpmnYgKU.js → systemd-linger-zllO90bD.js} +1 -1
  285. package/dist/{tables-BxyIF0w4.js → tables-CV7Afb0h.js} +1 -1
  286. package/dist/{tables-mE4cJBN2.js → tables-d739Y1xW.js} +1 -1
  287. package/dist/{tailnet-0_FsdHP-.js → tailnet-DJFUq7_R.js} +1 -1
  288. package/dist/{target-errors-KOHiT_JA.js → target-errors-CBI2Ga0y.js} +2 -2
  289. package/dist/{target-errors-mnlwhAjP.js → target-errors-iVxliVqA.js} +4 -4
  290. package/dist/{thinking-BeGmb5k6.js → thinking-DXYisHiZ.js} +7 -7
  291. package/dist/{tokens-q32vI39c.js → tokens-DxnY9ui_.js} +1 -1
  292. package/dist/{tool-images-RZdHiZcG.js → tool-images-2cBx1W8h.js} +2 -2
  293. package/dist/{tool-images-CNPfeCmU.js → tool-images-Bn6dB14u.js} +1 -1
  294. package/dist/{tui-DL6NZZEa.js → tui-DTVy-YhN.js} +6 -6
  295. package/dist/{tui-cli-BNAYhvpu.js → tui-cli-CexRLJ3a.js} +32 -32
  296. package/dist/{update-DlS-d52F.js → update-DijPxK0g.js} +3 -3
  297. package/dist/{update-cli-D8-DqIs2.js → update-cli-9NslG4yR.js} +104 -104
  298. package/dist/{update-runner-CzTQ7BJT.js → update-runner-B6_UqreW.js} +16 -16
  299. package/dist/{web-igmw_EhT.js → web-CzWRVmFt.js} +55 -55
  300. package/dist/web-DGoa03ue.js +118 -0
  301. package/dist/{webhooks-cli-QdaQhvbT.js → webhooks-cli-B4ZcXTtw.js} +6 -6
  302. package/dist/{whatsapp-actions-Bp8F0cOF.js → whatsapp-actions-Jm4VW1Ve.js} +17 -17
  303. package/dist/{whatsapp-actions-BHbJJyqw.js → whatsapp-actions-iEArE_Ez.js} +21 -21
  304. package/dist/{with-timeout-BMMWHlH3.js → with-timeout-C8-tY12i.js} +3 -3
  305. package/dist/{workspace-v76gFdZu.js → workspace-CIGzK2_w.js} +1 -1
  306. package/dist/{workspace-U-DyR64O.js → workspace-CUVC6GX1.js} +20 -20
  307. package/dist/{workspace-dirs-Cz_Zgtg2.js → workspace-dirs-D4SMysgC.js} +1 -1
  308. package/dist/{wsl-CvQfS6aU.js → wsl-CsGe5QCP.js} +2 -2
  309. package/package.json +7 -9
  310. package/scripts/auth-monitor.sh +89 -0
  311. package/scripts/bench-cli-startup.ts +200 -0
  312. package/scripts/bench-model.ts +146 -0
  313. package/scripts/build-and-run-mac.sh +18 -0
  314. package/scripts/build-docs-list.mjs +14 -0
  315. package/scripts/build_icon.sh +59 -0
  316. package/scripts/bundle-a2ui.sh +95 -0
  317. package/scripts/canvas-a2ui-copy.ts +40 -0
  318. package/scripts/changelog-to-html.sh +91 -0
  319. package/scripts/check-channel-agnostic-boundaries.mjs +343 -0
  320. package/scripts/check-composite-action-input-interpolation.py +81 -0
  321. package/scripts/check-ingress-agent-owner-context.mjs +45 -0
  322. package/scripts/check-no-monolithic-plugin-sdk-entry-imports.ts +103 -0
  323. package/scripts/check-no-pairing-store-group-auth.mjs +180 -0
  324. package/scripts/check-no-random-messaging-tmp.mjs +89 -0
  325. package/scripts/check-no-raw-channel-fetch.mjs +107 -0
  326. package/scripts/check-no-raw-window-open.mjs +86 -0
  327. package/scripts/check-no-register-http-handler.mjs +38 -0
  328. package/scripts/check-pairing-account-scope.mjs +100 -0
  329. package/scripts/check-plugin-sdk-exports.mjs +157 -0
  330. package/scripts/check-ts-max-loc.ts +80 -0
  331. package/scripts/check-webhook-auth-body-order.mjs +55 -0
  332. package/scripts/ci-changed-scope.d.mts +9 -0
  333. package/scripts/ci-changed-scope.mjs +141 -0
  334. package/scripts/claude-auth-status.sh +280 -0
  335. package/scripts/clawlog.sh +321 -0
  336. package/scripts/clawtributors-map.json +40 -0
  337. package/scripts/codesign-mac-app.sh +289 -0
  338. package/scripts/codespell-dictionary.txt +3 -0
  339. package/scripts/codespell-ignore.txt +9 -0
  340. package/scripts/committer +117 -0
  341. package/scripts/copy-export-html-templates.ts +59 -0
  342. package/scripts/copy-hook-metadata.ts +55 -0
  343. package/scripts/copy-plugin-sdk-root-alias.mjs +10 -0
  344. package/scripts/create-dmg.sh +176 -0
  345. package/scripts/create-instance.sh +4 -1
  346. package/scripts/cron_usage_report.ts +273 -0
  347. package/scripts/debug-claude-usage.ts +391 -0
  348. package/scripts/dev/discord-acp-plain-language-smoke.ts +868 -0
  349. package/scripts/dev/gateway-smoke.ts +75 -0
  350. package/scripts/dev/gateway-ws-client.ts +132 -0
  351. package/scripts/dev/ios-node-e2e.ts +283 -0
  352. package/scripts/dev/ios-pull-gateway-log.sh +17 -0
  353. package/scripts/dev/test-device-pair-telegram.ts +62 -0
  354. package/scripts/docker/cleanup-smoke/Dockerfile +19 -0
  355. package/scripts/docker/cleanup-smoke/run.sh +35 -0
  356. package/scripts/docker/install-sh-common/cli-verify.sh +47 -0
  357. package/scripts/docker/install-sh-e2e/Dockerfile +17 -0
  358. package/scripts/docker/install-sh-e2e/run.sh +535 -0
  359. package/scripts/docker/install-sh-nonroot/Dockerfile +33 -0
  360. package/scripts/docker/install-sh-nonroot/run.sh +36 -0
  361. package/scripts/docker/install-sh-smoke/Dockerfile +25 -0
  362. package/scripts/docker/install-sh-smoke/run.sh +63 -0
  363. package/scripts/docs-i18n/doc_mode.go +272 -0
  364. package/scripts/docs-i18n/glossary.go +29 -0
  365. package/scripts/docs-i18n/go.mod +10 -0
  366. package/scripts/docs-i18n/go.sum +10 -0
  367. package/scripts/docs-i18n/html_translate.go +160 -0
  368. package/scripts/docs-i18n/main.go +273 -0
  369. package/scripts/docs-i18n/markdown_segments.go +131 -0
  370. package/scripts/docs-i18n/masking.go +89 -0
  371. package/scripts/docs-i18n/order.go +37 -0
  372. package/scripts/docs-i18n/placeholders.go +30 -0
  373. package/scripts/docs-i18n/process.go +202 -0
  374. package/scripts/docs-i18n/prompt.go +146 -0
  375. package/scripts/docs-i18n/segment.go +11 -0
  376. package/scripts/docs-i18n/tm.go +132 -0
  377. package/scripts/docs-i18n/translator.go +247 -0
  378. package/scripts/docs-i18n/util.go +81 -0
  379. package/scripts/docs-link-audit.mjs +233 -0
  380. package/scripts/docs-list.js +173 -0
  381. package/scripts/docs-spellcheck.sh +44 -0
  382. package/scripts/e2e/Dockerfile +30 -0
  383. package/scripts/e2e/Dockerfile.qr-import +13 -0
  384. package/scripts/e2e/doctor-install-switch-docker.sh +160 -0
  385. package/scripts/e2e/gateway-network-docker.sh +145 -0
  386. package/scripts/e2e/onboard-docker.sh +570 -0
  387. package/scripts/e2e/plugins-docker.sh +224 -0
  388. package/scripts/e2e/qr-import-docker.sh +11 -0
  389. package/scripts/firecrawl-compare.ts +139 -0
  390. package/scripts/generate-host-env-security-policy-swift.mjs +74 -0
  391. package/scripts/generate-secretref-credential-matrix.ts +14 -0
  392. package/scripts/ghsa-patch.mjs +168 -0
  393. package/scripts/install-maca.sh +173 -0
  394. package/scripts/install.ps1 +329 -0
  395. package/scripts/ios-configure-signing.sh +100 -0
  396. package/scripts/ios-team-id.sh +207 -0
  397. package/scripts/label-open-issues.ts +893 -0
  398. package/scripts/lib/callsite-guard.mjs +45 -0
  399. package/scripts/lib/pairing-guard-context.mjs +13 -0
  400. package/scripts/lib/ts-guard-utils.mjs +157 -0
  401. package/scripts/make_appcast.sh +70 -0
  402. package/scripts/mobile-reauth.sh +84 -0
  403. package/scripts/notarize-mac-artifact.sh +65 -0
  404. package/scripts/npm_publish.sh +48 -0
  405. package/scripts/package-mac-app.sh +280 -0
  406. package/scripts/package-mac-dist.sh +93 -0
  407. package/scripts/podman/openclaw.container.in +28 -0
  408. package/scripts/pr +1708 -0
  409. package/scripts/pr-merge +44 -0
  410. package/scripts/pr-prepare +40 -0
  411. package/scripts/pr-review +13 -0
  412. package/scripts/pre-commit/filter-staged-files.mjs +39 -0
  413. package/scripts/pre-commit/run-node-tool.sh +31 -0
  414. package/scripts/protocol-gen-swift.ts +247 -0
  415. package/scripts/protocol-gen.ts +51 -0
  416. package/scripts/readability-basic-compare.ts +66 -0
  417. package/scripts/recover-orphaned-processes.sh +191 -0
  418. package/scripts/release-check.ts +362 -0
  419. package/scripts/repro/tsx-name-repro.ts +3 -0
  420. package/scripts/restart-mac.sh +269 -0
  421. package/scripts/run-node.d.mts +22 -0
  422. package/scripts/run-node.mjs +263 -0
  423. package/scripts/run-openclaw-podman.sh +213 -0
  424. package/scripts/sandbox-browser-entrypoint.sh +127 -0
  425. package/scripts/sandbox-browser-setup.sh +7 -0
  426. package/scripts/sandbox-common-setup.sh +40 -0
  427. package/scripts/sandbox-setup.sh +7 -0
  428. package/scripts/setup-auth-system.sh +119 -0
  429. package/scripts/shell-helpers/README.md +226 -0
  430. package/scripts/shell-helpers/clawdock-helpers.sh +417 -0
  431. package/scripts/sparkle-build.ts +76 -0
  432. package/scripts/sqlite-vec-smoke.mjs +38 -0
  433. package/scripts/sync-dist.sh +171 -0
  434. package/scripts/sync-labels.ts +100 -0
  435. package/scripts/sync-moonshot-docs.ts +125 -0
  436. package/scripts/sync-plugin-versions.ts +108 -0
  437. package/scripts/systemd/openclaw-auth-monitor.service +14 -0
  438. package/scripts/systemd/openclaw-auth-monitor.timer +10 -0
  439. package/scripts/termux-auth-widget.sh +81 -0
  440. package/scripts/termux-quick-auth.sh +30 -0
  441. package/scripts/termux-sync-widget.sh +25 -0
  442. package/scripts/test-cleanup-docker.sh +14 -0
  443. package/scripts/test-force.ts +59 -0
  444. package/scripts/test-hotspots.mjs +83 -0
  445. package/scripts/test-install-sh-docker.sh +72 -0
  446. package/scripts/test-install-sh-e2e-docker.sh +29 -0
  447. package/scripts/test-live-gateway-models-docker.sh +33 -0
  448. package/scripts/test-live-models-docker.sh +34 -0
  449. package/scripts/test-parallel.mjs +468 -0
  450. package/scripts/test-perf-budget.mjs +127 -0
  451. package/scripts/test-shell-completion.ts +223 -0
  452. package/scripts/ui.js +203 -0
  453. package/scripts/update-clawtributors.ts +573 -0
  454. package/scripts/update-clawtributors.types.ts +36 -0
  455. package/scripts/update-package-with-mirror.sh +67 -0
  456. package/scripts/watch-node.d.mts +14 -0
  457. package/scripts/watch-node.mjs +92 -0
  458. package/scripts/write-build-info.ts +47 -0
  459. package/scripts/write-cli-compat.ts +74 -0
  460. package/scripts/write-cli-startup-metadata.ts +93 -0
  461. package/scripts/write-plugin-sdk-entry-dts.ts +60 -0
  462. package/scripts/zai-fallback-repro.ts +168 -0
  463. package/dist/deliver-runtime-B80mXYrL.js +0 -36
  464. package/dist/deliver-runtime-DixuU_uB.js +0 -61
  465. package/dist/deps-send-discord.runtime-CDcLcDSt.js +0 -36
  466. package/dist/deps-send-discord.runtime-DZUccI6Z.js +0 -26
  467. package/dist/deps-send-imessage.runtime-BGPUVMIE.js +0 -35
  468. package/dist/deps-send-imessage.runtime-CF3OpoqY.js +0 -25
  469. package/dist/deps-send-signal.runtime-Cw4-ozeO.js +0 -24
  470. package/dist/deps-send-signal.runtime-DQDD44_O.js +0 -34
  471. package/dist/deps-send-slack.runtime-BDsDhS1P.js +0 -22
  472. package/dist/deps-send-slack.runtime-Cpy77gBG.js +0 -32
  473. package/dist/deps-send-telegram.runtime-D_4xVasO.js +0 -27
  474. package/dist/deps-send-whatsapp.runtime-D2sVAclS.js +0 -119
  475. package/dist/deps-send-whatsapp.runtime-UBgpoWg9.js +0 -60
  476. package/dist/image-runtime-B2qh5seQ.js +0 -55
  477. package/dist/image-runtime-DtMbm7AR.js +0 -29
  478. package/dist/manager-runtime-FO1Sx3W8.js +0 -18
  479. package/dist/pi-model-discovery-runtime-BeY4EUPp.js +0 -11
  480. package/dist/register.configure-BrDOSLIq.js +0 -165
  481. package/dist/runtime-whatsapp-login.runtime-DUb55byQ.js +0 -13
  482. package/dist/runtime-whatsapp-outbound.runtime-Bii_xSfI.js +0 -22
  483. package/dist/slash-commands.runtime-CVw6566g.js +0 -16
  484. package/dist/slash-dispatch.runtime-CMK_EPYG.js +0 -56
  485. package/dist/slash-dispatch.runtime-D0xInkf3.js +0 -114
  486. package/dist/slash-skill-commands.runtime-DxZ4z5h6.js +0 -20
  487. package/dist/subagent-registry-runtime-DdGrQBnQ.js +0 -56
  488. package/dist/subagent-registry-runtime-SkB2tTaE.js +0 -114
  489. package/dist/web-DWiOofzq.js +0 -118
  490. package/install.sh +0 -293
@@ -0,0 +1,893 @@
1
+ import { execFileSync } from "node:child_process";
2
+ import { existsSync, mkdirSync, readFileSync, writeFileSync } from "node:fs";
3
+ import { homedir } from "node:os";
4
+ import { dirname, join } from "node:path";
5
+
6
+ const BUG_LABEL = "bug";
7
+ const ENHANCEMENT_LABEL = "enhancement";
8
+ const SUPPORT_LABEL = "r: support";
9
+ const SKILL_LABEL = "r: skill";
10
+ const DEFAULT_MODEL = "gpt-5.2-codex";
11
+ const MAX_BODY_CHARS = 6000;
12
+ const GH_MAX_BUFFER = 50 * 1024 * 1024;
13
+ const PAGE_SIZE = 50;
14
+ const WORK_BATCH_SIZE = 500;
15
+ const STATE_VERSION = 1;
16
+ const STATE_FILE_NAME = "issue-labeler-state.json";
17
+ const CONFIG_BASE_DIR = process.env.XDG_CONFIG_HOME ?? join(homedir(), ".config");
18
+ const STATE_FILE_PATH = join(CONFIG_BASE_DIR, "openclaw", STATE_FILE_NAME);
19
+
20
+ const ISSUE_QUERY = `
21
+ query($owner: String!, $name: String!, $after: String, $pageSize: Int!) {
22
+ repository(owner: $owner, name: $name) {
23
+ issues(states: OPEN, first: $pageSize, after: $after, orderBy: { field: CREATED_AT, direction: DESC }) {
24
+ nodes {
25
+ number
26
+ title
27
+ body
28
+ labels(first: 100) {
29
+ nodes {
30
+ name
31
+ }
32
+ }
33
+ }
34
+ pageInfo {
35
+ hasNextPage
36
+ endCursor
37
+ }
38
+ totalCount
39
+ }
40
+ }
41
+ }
42
+ `;
43
+
44
+ const PULL_REQUEST_QUERY = `
45
+ query($owner: String!, $name: String!, $after: String, $pageSize: Int!) {
46
+ repository(owner: $owner, name: $name) {
47
+ pullRequests(states: OPEN, first: $pageSize, after: $after, orderBy: { field: CREATED_AT, direction: DESC }) {
48
+ nodes {
49
+ number
50
+ title
51
+ body
52
+ labels(first: 100) {
53
+ nodes {
54
+ name
55
+ }
56
+ }
57
+ }
58
+ pageInfo {
59
+ hasNextPage
60
+ endCursor
61
+ }
62
+ totalCount
63
+ }
64
+ }
65
+ }
66
+ `;
67
+
68
+ type IssueLabel = { name: string };
69
+
70
+ type LabelItem = {
71
+ number: number;
72
+ title: string;
73
+ body?: string | null;
74
+ labels: IssueLabel[];
75
+ };
76
+
77
+ type Issue = LabelItem;
78
+
79
+ type PullRequest = LabelItem;
80
+
81
+ type Classification = {
82
+ category: "bug" | "enhancement";
83
+ isSupport: boolean;
84
+ isSkillOnly: boolean;
85
+ };
86
+
87
+ type ScriptOptions = {
88
+ limit: number;
89
+ dryRun: boolean;
90
+ model: string;
91
+ };
92
+
93
+ type OpenAIResponse = {
94
+ output_text?: string;
95
+ output?: OpenAIResponseOutput[];
96
+ };
97
+
98
+ type OpenAIResponseOutput = {
99
+ type?: string;
100
+ content?: OpenAIResponseContent[];
101
+ };
102
+
103
+ type OpenAIResponseContent = {
104
+ type?: string;
105
+ text?: string;
106
+ };
107
+
108
+ type RepoInfo = {
109
+ owner: string;
110
+ name: string;
111
+ };
112
+
113
+ type IssuePageInfo = {
114
+ hasNextPage: boolean;
115
+ endCursor?: string | null;
116
+ };
117
+
118
+ type IssuePage = {
119
+ nodes: Array<{
120
+ number: number;
121
+ title: string;
122
+ body?: string | null;
123
+ labels?: { nodes?: IssueLabel[] | null } | null;
124
+ }>;
125
+ pageInfo: IssuePageInfo;
126
+ totalCount: number;
127
+ };
128
+
129
+ type IssueQueryResponse = {
130
+ data?: {
131
+ repository?: {
132
+ issues?: IssuePage | null;
133
+ } | null;
134
+ };
135
+ errors?: Array<{ message?: string }>;
136
+ };
137
+
138
+ type PullRequestPage = {
139
+ nodes: Array<{
140
+ number: number;
141
+ title: string;
142
+ body?: string | null;
143
+ labels?: { nodes?: IssueLabel[] | null } | null;
144
+ }>;
145
+ pageInfo: IssuePageInfo;
146
+ totalCount: number;
147
+ };
148
+
149
+ type PullRequestQueryResponse = {
150
+ data?: {
151
+ repository?: {
152
+ pullRequests?: PullRequestPage | null;
153
+ } | null;
154
+ };
155
+ errors?: Array<{ message?: string }>;
156
+ };
157
+
158
+ type IssueBatch = {
159
+ batchIndex: number;
160
+ issues: Issue[];
161
+ totalCount: number;
162
+ fetchedCount: number;
163
+ };
164
+
165
+ type PullRequestBatch = {
166
+ batchIndex: number;
167
+ pullRequests: PullRequest[];
168
+ totalCount: number;
169
+ fetchedCount: number;
170
+ };
171
+
172
+ type ScriptState = {
173
+ version: number;
174
+ issues: number[];
175
+ pullRequests: number[];
176
+ };
177
+
178
+ type LoadedState = {
179
+ state: ScriptState;
180
+ issueSet: Set<number>;
181
+ pullRequestSet: Set<number>;
182
+ };
183
+
184
+ type LabelTarget = "issue" | "pr";
185
+ type LabelItemBatch = {
186
+ batchIndex: number;
187
+ items: LabelItem[];
188
+ totalCount: number;
189
+ fetchedCount: number;
190
+ };
191
+
192
+ function parseArgs(argv: string[]): ScriptOptions {
193
+ let limit = Number.POSITIVE_INFINITY;
194
+ let dryRun = false;
195
+ let model = DEFAULT_MODEL;
196
+
197
+ for (let index = 0; index < argv.length; index++) {
198
+ const arg = argv[index];
199
+
200
+ if (arg === "--dry-run") {
201
+ dryRun = true;
202
+ continue;
203
+ }
204
+
205
+ if (arg === "--limit") {
206
+ const next = argv[index + 1];
207
+ if (!next || Number.isNaN(Number(next))) {
208
+ throw new Error("Missing/invalid --limit value");
209
+ }
210
+ const parsed = Number(next);
211
+ if (parsed <= 0) {
212
+ throw new Error("--limit must be greater than 0");
213
+ }
214
+ limit = parsed;
215
+ index++;
216
+ continue;
217
+ }
218
+
219
+ if (arg === "--model") {
220
+ const next = argv[index + 1];
221
+ if (!next) {
222
+ throw new Error("Missing --model value");
223
+ }
224
+ model = next;
225
+ index++;
226
+ continue;
227
+ }
228
+ }
229
+
230
+ return { limit, dryRun, model };
231
+ }
232
+
233
+ function logHeader(title: string) {
234
+ // eslint-disable-next-line no-console
235
+ console.log(`\n${title}`);
236
+ // eslint-disable-next-line no-console
237
+ console.log("=".repeat(title.length));
238
+ }
239
+
240
+ function logStep(message: string) {
241
+ // eslint-disable-next-line no-console
242
+ console.log(`• ${message}`);
243
+ }
244
+
245
+ function logSuccess(message: string) {
246
+ // eslint-disable-next-line no-console
247
+ console.log(`✓ ${message}`);
248
+ }
249
+
250
+ function logInfo(message: string) {
251
+ // eslint-disable-next-line no-console
252
+ console.log(` ${message}`);
253
+ }
254
+
255
+ function createEmptyState(): LoadedState {
256
+ const state: ScriptState = {
257
+ version: STATE_VERSION,
258
+ issues: [],
259
+ pullRequests: [],
260
+ };
261
+ return {
262
+ state,
263
+ issueSet: new Set(),
264
+ pullRequestSet: new Set(),
265
+ };
266
+ }
267
+
268
+ function loadState(statePath: string): LoadedState {
269
+ if (!existsSync(statePath)) {
270
+ return createEmptyState();
271
+ }
272
+
273
+ const raw = readFileSync(statePath, "utf8");
274
+ const parsed = JSON.parse(raw) as Partial<ScriptState>;
275
+ const issues = Array.isArray(parsed.issues)
276
+ ? parsed.issues.filter(
277
+ (value): value is number => typeof value === "number" && Number.isFinite(value),
278
+ )
279
+ : [];
280
+ const pullRequests = Array.isArray(parsed.pullRequests)
281
+ ? parsed.pullRequests.filter(
282
+ (value): value is number => typeof value === "number" && Number.isFinite(value),
283
+ )
284
+ : [];
285
+
286
+ const state: ScriptState = {
287
+ version: STATE_VERSION,
288
+ issues,
289
+ pullRequests,
290
+ };
291
+
292
+ return {
293
+ state,
294
+ issueSet: new Set(issues),
295
+ pullRequestSet: new Set(pullRequests),
296
+ };
297
+ }
298
+
299
+ function saveState(statePath: string, state: ScriptState): void {
300
+ mkdirSync(dirname(statePath), { recursive: true });
301
+ writeFileSync(statePath, `${JSON.stringify(state, null, 2)}\n`);
302
+ }
303
+
304
+ function buildStateSnapshot(issueSet: Set<number>, pullRequestSet: Set<number>): ScriptState {
305
+ return {
306
+ version: STATE_VERSION,
307
+ issues: Array.from(issueSet).toSorted((a, b) => a - b),
308
+ pullRequests: Array.from(pullRequestSet).toSorted((a, b) => a - b),
309
+ };
310
+ }
311
+
312
+ function runGh(args: string[]): string {
313
+ return execFileSync("gh", args, {
314
+ encoding: "utf8",
315
+ maxBuffer: GH_MAX_BUFFER,
316
+ });
317
+ }
318
+
319
+ function resolveRepo(): RepoInfo {
320
+ const remote = execFileSync("git", ["config", "--get", "remote.origin.url"], {
321
+ encoding: "utf8",
322
+ }).trim();
323
+
324
+ if (!remote) {
325
+ throw new Error("Unable to determine repository from git remote.");
326
+ }
327
+
328
+ const normalized = remote.replace(/\.git$/, "");
329
+
330
+ if (normalized.startsWith("git@github.com:")) {
331
+ const slug = normalized.replace("git@github.com:", "");
332
+ const [owner, name] = slug.split("/");
333
+ if (owner && name) {
334
+ return { owner, name };
335
+ }
336
+ }
337
+
338
+ if (normalized.startsWith("https://github.com/")) {
339
+ const slug = normalized.replace("https://github.com/", "");
340
+ const [owner, name] = slug.split("/");
341
+ if (owner && name) {
342
+ return { owner, name };
343
+ }
344
+ }
345
+
346
+ throw new Error(`Unsupported GitHub remote: ${remote}`);
347
+ }
348
+
349
+ function fetchIssuePage(repo: RepoInfo, after: string | null): IssuePage {
350
+ const args = [
351
+ "api",
352
+ "graphql",
353
+ "-f",
354
+ `query=${ISSUE_QUERY}`,
355
+ "-f",
356
+ `owner=${repo.owner}`,
357
+ "-f",
358
+ `name=${repo.name}`,
359
+ ];
360
+
361
+ if (after) {
362
+ args.push("-f", `after=${after}`);
363
+ }
364
+
365
+ args.push("-F", `pageSize=${PAGE_SIZE}`);
366
+
367
+ const stdout = runGh(args);
368
+ const payload = JSON.parse(stdout) as IssueQueryResponse;
369
+
370
+ if (payload.errors?.length) {
371
+ const message = payload.errors.map((error) => error.message ?? "Unknown error").join("; ");
372
+ throw new Error(`GitHub API error: ${message}`);
373
+ }
374
+
375
+ const issues = payload.data?.repository?.issues;
376
+ if (!issues) {
377
+ throw new Error("GitHub API response missing issues data.");
378
+ }
379
+
380
+ return issues;
381
+ }
382
+
383
+ function fetchPullRequestPage(repo: RepoInfo, after: string | null): PullRequestPage {
384
+ const args = [
385
+ "api",
386
+ "graphql",
387
+ "-f",
388
+ `query=${PULL_REQUEST_QUERY}`,
389
+ "-f",
390
+ `owner=${repo.owner}`,
391
+ "-f",
392
+ `name=${repo.name}`,
393
+ ];
394
+
395
+ if (after) {
396
+ args.push("-f", `after=${after}`);
397
+ }
398
+
399
+ args.push("-F", `pageSize=${PAGE_SIZE}`);
400
+
401
+ const stdout = runGh(args);
402
+ const payload = JSON.parse(stdout) as PullRequestQueryResponse;
403
+
404
+ if (payload.errors?.length) {
405
+ const message = payload.errors.map((error) => error.message ?? "Unknown error").join("; ");
406
+ throw new Error(`GitHub API error: ${message}`);
407
+ }
408
+
409
+ const pullRequests = payload.data?.repository?.pullRequests;
410
+ if (!pullRequests) {
411
+ throw new Error("GitHub API response missing pull request data.");
412
+ }
413
+
414
+ return pullRequests;
415
+ }
416
+
417
+ function mapNodeToLabelItem(node: IssuePage["nodes"][number]): LabelItem {
418
+ return {
419
+ number: node.number,
420
+ title: node.title,
421
+ body: node.body ?? "",
422
+ labels: node.labels?.nodes ?? [],
423
+ };
424
+ }
425
+
426
+ function* fetchOpenLabelItemBatches(params: {
427
+ limit: number;
428
+ kindPlural: "issues" | "pull requests";
429
+ fetchPage: (repo: RepoInfo, after: string | null) => IssuePage | PullRequestPage;
430
+ }): Generator<LabelItemBatch> {
431
+ const repo = resolveRepo();
432
+ const results: LabelItem[] = [];
433
+ let page = 1;
434
+ let after: string | null = null;
435
+ let totalCount = 0;
436
+ let fetchedCount = 0;
437
+ let batchIndex = 1;
438
+
439
+ logStep(`Repository: ${repo.owner}/${repo.name}`);
440
+
441
+ while (fetchedCount < params.limit) {
442
+ const pageData = params.fetchPage(repo, after);
443
+ const nodes = pageData.nodes ?? [];
444
+ totalCount = pageData.totalCount ?? totalCount;
445
+
446
+ if (page === 1) {
447
+ logSuccess(`Found ${totalCount} open ${params.kindPlural}.`);
448
+ }
449
+
450
+ logInfo(`Fetched page ${page} (${nodes.length} ${params.kindPlural}).`);
451
+
452
+ for (const node of nodes) {
453
+ if (fetchedCount >= params.limit) {
454
+ break;
455
+ }
456
+ results.push(mapNodeToLabelItem(node));
457
+ fetchedCount += 1;
458
+
459
+ if (results.length >= WORK_BATCH_SIZE) {
460
+ yield {
461
+ batchIndex,
462
+ items: results.splice(0, results.length),
463
+ totalCount,
464
+ fetchedCount,
465
+ };
466
+ batchIndex += 1;
467
+ }
468
+ }
469
+
470
+ if (!pageData.pageInfo.hasNextPage) {
471
+ break;
472
+ }
473
+
474
+ after = pageData.pageInfo.endCursor ?? null;
475
+ page += 1;
476
+ }
477
+
478
+ if (results.length) {
479
+ yield {
480
+ batchIndex,
481
+ items: results,
482
+ totalCount,
483
+ fetchedCount,
484
+ };
485
+ }
486
+ }
487
+
488
+ function* fetchOpenIssueBatches(limit: number): Generator<IssueBatch> {
489
+ for (const batch of fetchOpenLabelItemBatches({
490
+ limit,
491
+ kindPlural: "issues",
492
+ fetchPage: fetchIssuePage,
493
+ })) {
494
+ yield {
495
+ batchIndex: batch.batchIndex,
496
+ issues: batch.items,
497
+ totalCount: batch.totalCount,
498
+ fetchedCount: batch.fetchedCount,
499
+ };
500
+ }
501
+ }
502
+
503
+ function* fetchOpenPullRequestBatches(limit: number): Generator<PullRequestBatch> {
504
+ for (const batch of fetchOpenLabelItemBatches({
505
+ limit,
506
+ kindPlural: "pull requests",
507
+ fetchPage: fetchPullRequestPage,
508
+ })) {
509
+ yield {
510
+ batchIndex: batch.batchIndex,
511
+ pullRequests: batch.items,
512
+ totalCount: batch.totalCount,
513
+ fetchedCount: batch.fetchedCount,
514
+ };
515
+ }
516
+ }
517
+
518
+ function truncateBody(body: string): string {
519
+ if (body.length <= MAX_BODY_CHARS) {
520
+ return body;
521
+ }
522
+ return `${body.slice(0, MAX_BODY_CHARS)}\n\n[truncated]`;
523
+ }
524
+
525
+ function buildItemPrompt(item: LabelItem, kind: "issue" | "pull request"): string {
526
+ const body = truncateBody(item.body?.trim() ?? "");
527
+ return `Type: ${kind}\nTitle:\n${item.title.trim()}\n\nBody:\n${body}`;
528
+ }
529
+
530
+ function extractResponseText(payload: OpenAIResponse): string {
531
+ if (payload.output_text && payload.output_text.trim()) {
532
+ return payload.output_text.trim();
533
+ }
534
+
535
+ const chunks: string[] = [];
536
+ for (const item of payload.output ?? []) {
537
+ if (item.type !== "message") {
538
+ continue;
539
+ }
540
+ for (const content of item.content ?? []) {
541
+ if (content.type === "output_text" && typeof content.text === "string") {
542
+ chunks.push(content.text);
543
+ }
544
+ }
545
+ }
546
+
547
+ return chunks.join("\n").trim();
548
+ }
549
+
550
+ function isRecord(value: unknown): value is Record<string, unknown> {
551
+ return typeof value === "object" && value !== null;
552
+ }
553
+
554
+ function fallbackCategory(issueText: string): "bug" | "enhancement" {
555
+ const lower = issueText.toLowerCase();
556
+ const bugSignals = [
557
+ "bug",
558
+ "error",
559
+ "crash",
560
+ "broken",
561
+ "regression",
562
+ "fails",
563
+ "failure",
564
+ "incorrect",
565
+ ];
566
+ return bugSignals.some((signal) => lower.includes(signal)) ? "bug" : "enhancement";
567
+ }
568
+
569
+ function normalizeClassification(raw: unknown, issueText: string): Classification {
570
+ const fallback = fallbackCategory(issueText);
571
+
572
+ if (!isRecord(raw)) {
573
+ return { category: fallback, isSupport: false, isSkillOnly: false };
574
+ }
575
+
576
+ const categoryRaw = raw.category;
577
+ const category = categoryRaw === "bug" || categoryRaw === "enhancement" ? categoryRaw : fallback;
578
+
579
+ const isSupport = raw.isSupport === true;
580
+ const isSkillOnly = raw.isSkillOnly === true;
581
+
582
+ return { category, isSupport, isSkillOnly };
583
+ }
584
+
585
+ async function classifyItem(
586
+ item: LabelItem,
587
+ kind: "issue" | "pull request",
588
+ options: { apiKey: string; model: string },
589
+ ): Promise<Classification> {
590
+ const itemText = buildItemPrompt(item, kind);
591
+ const response = await fetch("https://api.openai.com/v1/responses", {
592
+ method: "POST",
593
+ headers: {
594
+ Authorization: `Bearer ${options.apiKey}`,
595
+ "Content-Type": "application/json",
596
+ },
597
+ body: JSON.stringify({
598
+ model: options.model,
599
+ max_output_tokens: 200,
600
+ text: {
601
+ format: {
602
+ type: "json_schema",
603
+ name: "issue_classification",
604
+ schema: {
605
+ type: "object",
606
+ additionalProperties: false,
607
+ properties: {
608
+ category: { type: "string", enum: ["bug", "enhancement"] },
609
+ isSupport: { type: "boolean" },
610
+ isSkillOnly: { type: "boolean" },
611
+ },
612
+ required: ["category", "isSupport", "isSkillOnly"],
613
+ },
614
+ },
615
+ },
616
+ input: [
617
+ {
618
+ role: "system",
619
+ content:
620
+ "You classify GitHub issues and pull requests for OpenClaw. Respond with JSON only, no extra text.",
621
+ },
622
+ {
623
+ role: "user",
624
+ content: [
625
+ "Determine classification:\n",
626
+ "- category: 'bug' if the item reports incorrect behavior, errors, crashes, or regressions; otherwise 'enhancement'.\n",
627
+ "- isSupport: true if the item is primarily a support request or troubleshooting/how-to question, not a change request.\n",
628
+ "- isSkillOnly: true if the item solely requests or delivers adding/updating skills (no other feature/bug work).\n\n",
629
+ itemText,
630
+ "\n\nReturn JSON with keys: category, isSupport, isSkillOnly.",
631
+ ].join(""),
632
+ },
633
+ ],
634
+ }),
635
+ });
636
+
637
+ if (!response.ok) {
638
+ const text = await response.text();
639
+ throw new Error(`OpenAI request failed (${response.status}): ${text}`);
640
+ }
641
+
642
+ const payload = (await response.json()) as OpenAIResponse;
643
+ const rawText = extractResponseText(payload);
644
+ let parsed: unknown = undefined;
645
+
646
+ if (rawText) {
647
+ try {
648
+ parsed = JSON.parse(rawText);
649
+ } catch (error) {
650
+ throw new Error(`Failed to parse OpenAI response: ${String(error)} (raw: ${rawText})`, {
651
+ cause: error,
652
+ });
653
+ }
654
+ }
655
+
656
+ return normalizeClassification(parsed, itemText);
657
+ }
658
+
659
+ function applyLabels(
660
+ target: LabelTarget,
661
+ item: LabelItem,
662
+ labelsToAdd: string[],
663
+ dryRun: boolean,
664
+ ): boolean {
665
+ if (!labelsToAdd.length) {
666
+ return false;
667
+ }
668
+
669
+ if (dryRun) {
670
+ logInfo(`Would add labels: ${labelsToAdd.join(", ")}`);
671
+ return true;
672
+ }
673
+
674
+ const ghTarget = target === "issue" ? "issue" : "pr";
675
+
676
+ execFileSync(
677
+ "gh",
678
+ [ghTarget, "edit", String(item.number), "--add-label", labelsToAdd.join(",")],
679
+ { stdio: "inherit" },
680
+ );
681
+ return true;
682
+ }
683
+
684
+ async function main() {
685
+ // Makes `... | head` safe.
686
+ process.stdout.on("error", (error: NodeJS.ErrnoException) => {
687
+ if (error.code === "EPIPE") {
688
+ process.exit(0);
689
+ }
690
+ throw error;
691
+ });
692
+
693
+ const { limit, dryRun, model } = parseArgs(process.argv.slice(2));
694
+ const apiKey = process.env.OPENAI_API_KEY;
695
+ if (!apiKey) {
696
+ throw new Error("OPENAI_API_KEY is required to classify issues and pull requests.");
697
+ }
698
+
699
+ logHeader("OpenClaw Issue Label Audit");
700
+ logStep(`Mode: ${dryRun ? "dry-run" : "apply labels"}`);
701
+ logStep(`Model: ${model}`);
702
+ logStep(`Issue limit: ${Number.isFinite(limit) ? limit : "unlimited"}`);
703
+ logStep(`PR limit: ${Number.isFinite(limit) ? limit : "unlimited"}`);
704
+ logStep(`Batch size: ${WORK_BATCH_SIZE}`);
705
+ logStep(`State file: ${STATE_FILE_PATH}`);
706
+ if (dryRun) {
707
+ logInfo("Dry-run enabled: state file will not be updated.");
708
+ }
709
+
710
+ let loadedState: LoadedState;
711
+ try {
712
+ loadedState = loadState(STATE_FILE_PATH);
713
+ } catch (error) {
714
+ logInfo(`State file unreadable (${String(error)}); starting fresh.`);
715
+ loadedState = createEmptyState();
716
+ }
717
+
718
+ logInfo(
719
+ `State entries: ${loadedState.issueSet.size} issues, ${loadedState.pullRequestSet.size} pull requests.`,
720
+ );
721
+
722
+ const issueState = loadedState.issueSet;
723
+ const pullRequestState = loadedState.pullRequestSet;
724
+
725
+ logHeader("Issues");
726
+
727
+ let updatedCount = 0;
728
+ let supportCount = 0;
729
+ let skillCount = 0;
730
+ let categoryAddedCount = 0;
731
+ let scannedCount = 0;
732
+ let processedCount = 0;
733
+ let skippedCount = 0;
734
+ let totalCount = 0;
735
+ let batches = 0;
736
+
737
+ for (const batch of fetchOpenIssueBatches(limit)) {
738
+ batches += 1;
739
+ scannedCount += batch.issues.length;
740
+ totalCount = batch.totalCount ?? totalCount;
741
+
742
+ const pendingIssues = batch.issues.filter((issue) => !issueState.has(issue.number));
743
+ const skippedInBatch = batch.issues.length - pendingIssues.length;
744
+ skippedCount += skippedInBatch;
745
+
746
+ logHeader(`Issue Batch ${batch.batchIndex}`);
747
+ logInfo(`Fetched ${batch.issues.length} issues (${skippedInBatch} already processed).`);
748
+ logInfo(`Processing ${pendingIssues.length} issues (scanned so far: ${scannedCount}).`);
749
+
750
+ for (const issue of pendingIssues) {
751
+ // eslint-disable-next-line no-console
752
+ console.log(`\n#${issue.number} — ${issue.title}`);
753
+
754
+ const labels = new Set(issue.labels.map((label) => label.name));
755
+ logInfo(`Existing labels: ${Array.from(labels).toSorted().join(", ") || "none"}`);
756
+
757
+ const classification = await classifyItem(issue, "issue", { apiKey, model });
758
+ logInfo(
759
+ `Classification: category=${classification.category}, support=${classification.isSupport ? "yes" : "no"}, skill-only=${classification.isSkillOnly ? "yes" : "no"}.`,
760
+ );
761
+
762
+ const toAdd: string[] = [];
763
+
764
+ if (!labels.has(BUG_LABEL) && !labels.has(ENHANCEMENT_LABEL)) {
765
+ toAdd.push(classification.category);
766
+ categoryAddedCount += 1;
767
+ }
768
+
769
+ if (classification.isSupport && !labels.has(SUPPORT_LABEL)) {
770
+ toAdd.push(SUPPORT_LABEL);
771
+ supportCount += 1;
772
+ }
773
+
774
+ if (classification.isSkillOnly && !labels.has(SKILL_LABEL)) {
775
+ toAdd.push(SKILL_LABEL);
776
+ skillCount += 1;
777
+ }
778
+
779
+ const changed = applyLabels("issue", issue, toAdd, dryRun);
780
+ if (changed) {
781
+ updatedCount += 1;
782
+ logSuccess(`Labels added: ${toAdd.join(", ")}`);
783
+ } else {
784
+ logInfo("No label changes needed.");
785
+ }
786
+
787
+ issueState.add(issue.number);
788
+ processedCount += 1;
789
+ }
790
+
791
+ if (!dryRun && pendingIssues.length > 0) {
792
+ saveState(STATE_FILE_PATH, buildStateSnapshot(issueState, pullRequestState));
793
+ logInfo("State checkpoint saved.");
794
+ }
795
+ }
796
+
797
+ logHeader("Pull Requests");
798
+
799
+ let prUpdatedCount = 0;
800
+ let prSkillCount = 0;
801
+ let prScannedCount = 0;
802
+ let prProcessedCount = 0;
803
+ let prSkippedCount = 0;
804
+ let prTotalCount = 0;
805
+ let prBatches = 0;
806
+
807
+ for (const batch of fetchOpenPullRequestBatches(limit)) {
808
+ prBatches += 1;
809
+ prScannedCount += batch.pullRequests.length;
810
+ prTotalCount = batch.totalCount ?? prTotalCount;
811
+
812
+ const pendingPullRequests = batch.pullRequests.filter(
813
+ (pullRequest) => !pullRequestState.has(pullRequest.number),
814
+ );
815
+ const skippedInBatch = batch.pullRequests.length - pendingPullRequests.length;
816
+ prSkippedCount += skippedInBatch;
817
+
818
+ logHeader(`PR Batch ${batch.batchIndex}`);
819
+ logInfo(
820
+ `Fetched ${batch.pullRequests.length} pull requests (${skippedInBatch} already processed).`,
821
+ );
822
+ logInfo(
823
+ `Processing ${pendingPullRequests.length} pull requests (scanned so far: ${prScannedCount}).`,
824
+ );
825
+
826
+ for (const pullRequest of pendingPullRequests) {
827
+ // eslint-disable-next-line no-console
828
+ console.log(`\n#${pullRequest.number} — ${pullRequest.title}`);
829
+
830
+ const labels = new Set(pullRequest.labels.map((label) => label.name));
831
+ logInfo(`Existing labels: ${Array.from(labels).toSorted().join(", ") || "none"}`);
832
+
833
+ if (labels.has(SKILL_LABEL)) {
834
+ logInfo("Skill label already present; skipping classification.");
835
+ pullRequestState.add(pullRequest.number);
836
+ prProcessedCount += 1;
837
+ continue;
838
+ }
839
+
840
+ const classification = await classifyItem(pullRequest, "pull request", { apiKey, model });
841
+ logInfo(
842
+ `Classification: category=${classification.category}, support=${classification.isSupport ? "yes" : "no"}, skill-only=${classification.isSkillOnly ? "yes" : "no"}.`,
843
+ );
844
+
845
+ const toAdd: string[] = [];
846
+
847
+ if (classification.isSkillOnly && !labels.has(SKILL_LABEL)) {
848
+ toAdd.push(SKILL_LABEL);
849
+ prSkillCount += 1;
850
+ }
851
+
852
+ const changed = applyLabels("pr", pullRequest, toAdd, dryRun);
853
+ if (changed) {
854
+ prUpdatedCount += 1;
855
+ logSuccess(`Labels added: ${toAdd.join(", ")}`);
856
+ } else {
857
+ logInfo("No label changes needed.");
858
+ }
859
+
860
+ pullRequestState.add(pullRequest.number);
861
+ prProcessedCount += 1;
862
+ }
863
+
864
+ if (!dryRun && pendingPullRequests.length > 0) {
865
+ saveState(STATE_FILE_PATH, buildStateSnapshot(issueState, pullRequestState));
866
+ logInfo("State checkpoint saved.");
867
+ }
868
+ }
869
+
870
+ logHeader("Summary");
871
+ logInfo(`Issues scanned: ${scannedCount}`);
872
+ if (totalCount) {
873
+ logInfo(`Total open issues: ${totalCount}`);
874
+ }
875
+ logInfo(`Issue batches processed: ${batches}`);
876
+ logInfo(`Issues processed: ${processedCount}`);
877
+ logInfo(`Issues skipped (state): ${skippedCount}`);
878
+ logInfo(`Issues updated: ${updatedCount}`);
879
+ logInfo(`Added bug/enhancement labels: ${categoryAddedCount}`);
880
+ logInfo(`Added r: support labels: ${supportCount}`);
881
+ logInfo(`Added r: skill labels (issues): ${skillCount}`);
882
+ logInfo(`Pull requests scanned: ${prScannedCount}`);
883
+ if (prTotalCount) {
884
+ logInfo(`Total open pull requests: ${prTotalCount}`);
885
+ }
886
+ logInfo(`PR batches processed: ${prBatches}`);
887
+ logInfo(`Pull requests processed: ${prProcessedCount}`);
888
+ logInfo(`Pull requests skipped (state): ${prSkippedCount}`);
889
+ logInfo(`Pull requests updated: ${prUpdatedCount}`);
890
+ logInfo(`Added r: skill labels (PRs): ${prSkillCount}`);
891
+ }
892
+
893
+ await main();