threadwell 0.0.0 → 0.0.2

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 (861) hide show
  1. package/CHANGELOG.md +4170 -0
  2. package/README.md +393 -4
  3. package/dist/bun/cli.d.ts +3 -0
  4. package/dist/bun/cli.d.ts.map +1 -0
  5. package/dist/bun/cli.js +9 -0
  6. package/dist/bun/cli.js.map +1 -0
  7. package/dist/bun/register-bedrock.d.ts +2 -0
  8. package/dist/bun/register-bedrock.d.ts.map +1 -0
  9. package/dist/bun/register-bedrock.js +4 -0
  10. package/dist/bun/register-bedrock.js.map +1 -0
  11. package/dist/bun/restore-sandbox-env.d.ts +13 -0
  12. package/dist/bun/restore-sandbox-env.d.ts.map +1 -0
  13. package/dist/bun/restore-sandbox-env.js +32 -0
  14. package/dist/bun/restore-sandbox-env.js.map +1 -0
  15. package/dist/cli/args.d.ts +53 -0
  16. package/dist/cli/args.d.ts.map +1 -0
  17. package/dist/cli/args.js +341 -0
  18. package/dist/cli/args.js.map +1 -0
  19. package/dist/cli/config-selector.d.ts +14 -0
  20. package/dist/cli/config-selector.d.ts.map +1 -0
  21. package/dist/cli/config-selector.js +31 -0
  22. package/dist/cli/config-selector.js.map +1 -0
  23. package/dist/cli/file-processor.d.ts +15 -0
  24. package/dist/cli/file-processor.d.ts.map +1 -0
  25. package/dist/cli/file-processor.js +83 -0
  26. package/dist/cli/file-processor.js.map +1 -0
  27. package/dist/cli/initial-message.d.ts +18 -0
  28. package/dist/cli/initial-message.d.ts.map +1 -0
  29. package/dist/cli/initial-message.js +22 -0
  30. package/dist/cli/initial-message.js.map +1 -0
  31. package/dist/cli/list-models.d.ts +9 -0
  32. package/dist/cli/list-models.d.ts.map +1 -0
  33. package/dist/cli/list-models.js +98 -0
  34. package/dist/cli/list-models.js.map +1 -0
  35. package/dist/cli/session-picker.d.ts +9 -0
  36. package/dist/cli/session-picker.d.ts.map +1 -0
  37. package/dist/cli/session-picker.js +35 -0
  38. package/dist/cli/session-picker.js.map +1 -0
  39. package/dist/cli.d.ts +3 -0
  40. package/dist/cli.d.ts.map +1 -0
  41. package/dist/cli.js +19 -3
  42. package/dist/cli.js.map +1 -0
  43. package/dist/config.d.ts +93 -0
  44. package/dist/config.d.ts.map +1 -0
  45. package/dist/config.js +404 -0
  46. package/dist/config.js.map +1 -0
  47. package/dist/core/agent-session-runtime.d.ts +117 -0
  48. package/dist/core/agent-session-runtime.d.ts.map +1 -0
  49. package/dist/core/agent-session-runtime.js +300 -0
  50. package/dist/core/agent-session-runtime.js.map +1 -0
  51. package/dist/core/agent-session-services.d.ts +86 -0
  52. package/dist/core/agent-session-services.d.ts.map +1 -0
  53. package/dist/core/agent-session-services.js +117 -0
  54. package/dist/core/agent-session-services.js.map +1 -0
  55. package/dist/core/agent-session.d.ts +659 -0
  56. package/dist/core/agent-session.d.ts.map +1 -0
  57. package/dist/core/agent-session.js +3210 -0
  58. package/dist/core/agent-session.js.map +1 -0
  59. package/dist/core/auth-guidance.d.ts +5 -0
  60. package/dist/core/auth-guidance.d.ts.map +1 -0
  61. package/dist/core/auth-guidance.js +21 -0
  62. package/dist/core/auth-guidance.js.map +1 -0
  63. package/dist/core/auth-storage.d.ts +141 -0
  64. package/dist/core/auth-storage.d.ts.map +1 -0
  65. package/dist/core/auth-storage.js +441 -0
  66. package/dist/core/auth-storage.js.map +1 -0
  67. package/dist/core/bash-executor.d.ts +32 -0
  68. package/dist/core/bash-executor.d.ts.map +1 -0
  69. package/dist/core/bash-executor.js +111 -0
  70. package/dist/core/bash-executor.js.map +1 -0
  71. package/dist/core/compaction/branch-summarization.d.ts +88 -0
  72. package/dist/core/compaction/branch-summarization.d.ts.map +1 -0
  73. package/dist/core/compaction/branch-summarization.js +243 -0
  74. package/dist/core/compaction/branch-summarization.js.map +1 -0
  75. package/dist/core/compaction/compaction.d.ts +121 -0
  76. package/dist/core/compaction/compaction.d.ts.map +1 -0
  77. package/dist/core/compaction/compaction.js +615 -0
  78. package/dist/core/compaction/compaction.js.map +1 -0
  79. package/dist/core/compaction/index.d.ts +7 -0
  80. package/dist/core/compaction/index.d.ts.map +1 -0
  81. package/dist/core/compaction/index.js +7 -0
  82. package/dist/core/compaction/index.js.map +1 -0
  83. package/dist/core/compaction/utils.d.ts +38 -0
  84. package/dist/core/compaction/utils.d.ts.map +1 -0
  85. package/dist/core/compaction/utils.js +153 -0
  86. package/dist/core/compaction/utils.js.map +1 -0
  87. package/dist/core/continuity/continuity-files.d.ts +79 -0
  88. package/dist/core/continuity/continuity-files.d.ts.map +1 -0
  89. package/dist/core/continuity/continuity-files.js +326 -0
  90. package/dist/core/continuity/continuity-files.js.map +1 -0
  91. package/dist/core/continuity/continuity-intent-router.d.ts +9 -0
  92. package/dist/core/continuity/continuity-intent-router.d.ts.map +1 -0
  93. package/dist/core/continuity/continuity-intent-router.js +96 -0
  94. package/dist/core/continuity/continuity-intent-router.js.map +1 -0
  95. package/dist/core/continuity/continuity-manager.d.ts +28 -0
  96. package/dist/core/continuity/continuity-manager.d.ts.map +1 -0
  97. package/dist/core/continuity/continuity-manager.js +288 -0
  98. package/dist/core/continuity/continuity-manager.js.map +1 -0
  99. package/dist/core/continuity/continuity-service.d.ts +22 -0
  100. package/dist/core/continuity/continuity-service.d.ts.map +1 -0
  101. package/dist/core/continuity/continuity-service.js +34 -0
  102. package/dist/core/continuity/continuity-service.js.map +1 -0
  103. package/dist/core/continuity/index.d.ts +5 -0
  104. package/dist/core/continuity/index.d.ts.map +1 -0
  105. package/dist/core/continuity/index.js +5 -0
  106. package/dist/core/continuity/index.js.map +1 -0
  107. package/dist/core/defaults.d.ts +3 -0
  108. package/dist/core/defaults.d.ts.map +1 -0
  109. package/dist/core/defaults.js +2 -0
  110. package/dist/core/defaults.js.map +1 -0
  111. package/dist/core/diagnostics.d.ts +15 -0
  112. package/dist/core/diagnostics.d.ts.map +1 -0
  113. package/dist/core/diagnostics.js +2 -0
  114. package/dist/core/diagnostics.js.map +1 -0
  115. package/dist/core/event-bus.d.ts +9 -0
  116. package/dist/core/event-bus.d.ts.map +1 -0
  117. package/dist/core/event-bus.js +25 -0
  118. package/dist/core/event-bus.js.map +1 -0
  119. package/dist/core/exec.d.ts +29 -0
  120. package/dist/core/exec.d.ts.map +1 -0
  121. package/dist/core/exec.js +75 -0
  122. package/dist/core/exec.js.map +1 -0
  123. package/dist/core/export-html/ansi-to-html.d.ts +22 -0
  124. package/dist/core/export-html/ansi-to-html.d.ts.map +1 -0
  125. package/dist/core/export-html/ansi-to-html.js +249 -0
  126. package/dist/core/export-html/ansi-to-html.js.map +1 -0
  127. package/dist/core/export-html/index.d.ts +37 -0
  128. package/dist/core/export-html/index.d.ts.map +1 -0
  129. package/dist/core/export-html/index.js +224 -0
  130. package/dist/core/export-html/index.js.map +1 -0
  131. package/dist/core/export-html/template.css +1066 -0
  132. package/dist/core/export-html/template.html +55 -0
  133. package/dist/core/export-html/template.js +1834 -0
  134. package/dist/core/export-html/tool-renderer.d.ts +34 -0
  135. package/dist/core/export-html/tool-renderer.d.ts.map +1 -0
  136. package/dist/core/export-html/tool-renderer.js +108 -0
  137. package/dist/core/export-html/tool-renderer.js.map +1 -0
  138. package/dist/core/export-html/vendor/highlight.min.js +1213 -0
  139. package/dist/core/export-html/vendor/marked.min.js +6 -0
  140. package/dist/core/extensions/index.d.ts +12 -0
  141. package/dist/core/extensions/index.d.ts.map +1 -0
  142. package/dist/core/extensions/index.js +9 -0
  143. package/dist/core/extensions/index.js.map +1 -0
  144. package/dist/core/extensions/loader.d.ts +24 -0
  145. package/dist/core/extensions/loader.d.ts.map +1 -0
  146. package/dist/core/extensions/loader.js +485 -0
  147. package/dist/core/extensions/loader.js.map +1 -0
  148. package/dist/core/extensions/runner.d.ts +159 -0
  149. package/dist/core/extensions/runner.d.ts.map +1 -0
  150. package/dist/core/extensions/runner.js +824 -0
  151. package/dist/core/extensions/runner.js.map +1 -0
  152. package/dist/core/extensions/types.d.ts +1173 -0
  153. package/dist/core/extensions/types.d.ts.map +1 -0
  154. package/dist/core/extensions/types.js +45 -0
  155. package/dist/core/extensions/types.js.map +1 -0
  156. package/dist/core/extensions/wrapper.d.ts +20 -0
  157. package/dist/core/extensions/wrapper.d.ts.map +1 -0
  158. package/dist/core/extensions/wrapper.js +22 -0
  159. package/dist/core/extensions/wrapper.js.map +1 -0
  160. package/dist/core/footer-data-provider.d.ts +72 -0
  161. package/dist/core/footer-data-provider.d.ts.map +1 -0
  162. package/dist/core/footer-data-provider.js +374 -0
  163. package/dist/core/footer-data-provider.js.map +1 -0
  164. package/dist/core/identity.d.ts +8 -0
  165. package/dist/core/identity.d.ts.map +1 -0
  166. package/dist/core/identity.js +8 -0
  167. package/dist/core/identity.js.map +1 -0
  168. package/dist/core/index.d.ts +12 -0
  169. package/dist/core/index.d.ts.map +1 -0
  170. package/dist/core/index.js +12 -0
  171. package/dist/core/index.js.map +1 -0
  172. package/dist/core/keybindings.d.ts +353 -0
  173. package/dist/core/keybindings.d.ts.map +1 -0
  174. package/dist/core/keybindings.js +295 -0
  175. package/dist/core/keybindings.js.map +1 -0
  176. package/dist/core/memory/adr/adr-candidates.d.ts +15 -0
  177. package/dist/core/memory/adr/adr-candidates.d.ts.map +1 -0
  178. package/dist/core/memory/adr/adr-candidates.js +63 -0
  179. package/dist/core/memory/adr/adr-candidates.js.map +1 -0
  180. package/dist/core/memory/adr/adr-format.d.ts +31 -0
  181. package/dist/core/memory/adr/adr-format.d.ts.map +1 -0
  182. package/dist/core/memory/adr/adr-format.js +138 -0
  183. package/dist/core/memory/adr/adr-format.js.map +1 -0
  184. package/dist/core/memory/adr/adr-service.d.ts +19 -0
  185. package/dist/core/memory/adr/adr-service.d.ts.map +1 -0
  186. package/dist/core/memory/adr/adr-service.js +63 -0
  187. package/dist/core/memory/adr/adr-service.js.map +1 -0
  188. package/dist/core/memory/adr/index.d.ts +4 -0
  189. package/dist/core/memory/adr/index.d.ts.map +1 -0
  190. package/dist/core/memory/adr/index.js +4 -0
  191. package/dist/core/memory/adr/index.js.map +1 -0
  192. package/dist/core/memory/content-ref.d.ts +22 -0
  193. package/dist/core/memory/content-ref.d.ts.map +1 -0
  194. package/dist/core/memory/content-ref.js +34 -0
  195. package/dist/core/memory/content-ref.js.map +1 -0
  196. package/dist/core/memory/context/context-candidates.d.ts +9 -0
  197. package/dist/core/memory/context/context-candidates.d.ts.map +1 -0
  198. package/dist/core/memory/context/context-candidates.js +64 -0
  199. package/dist/core/memory/context/context-candidates.js.map +1 -0
  200. package/dist/core/memory/context/context-diagnostics.d.ts +12 -0
  201. package/dist/core/memory/context/context-diagnostics.d.ts.map +1 -0
  202. package/dist/core/memory/context/context-diagnostics.js +50 -0
  203. package/dist/core/memory/context/context-diagnostics.js.map +1 -0
  204. package/dist/core/memory/context/context-format.d.ts +41 -0
  205. package/dist/core/memory/context/context-format.d.ts.map +1 -0
  206. package/dist/core/memory/context/context-format.js +199 -0
  207. package/dist/core/memory/context/context-format.js.map +1 -0
  208. package/dist/core/memory/context/index.d.ts +4 -0
  209. package/dist/core/memory/context/index.d.ts.map +1 -0
  210. package/dist/core/memory/context/index.js +4 -0
  211. package/dist/core/memory/context/index.js.map +1 -0
  212. package/dist/core/memory/context-pressure.d.ts +15 -0
  213. package/dist/core/memory/context-pressure.d.ts.map +1 -0
  214. package/dist/core/memory/context-pressure.js +36 -0
  215. package/dist/core/memory/context-pressure.js.map +1 -0
  216. package/dist/core/memory/continuity-import.d.ts +6 -0
  217. package/dist/core/memory/continuity-import.d.ts.map +1 -0
  218. package/dist/core/memory/continuity-import.js +63 -0
  219. package/dist/core/memory/continuity-import.js.map +1 -0
  220. package/dist/core/memory/dedupe.d.ts +10 -0
  221. package/dist/core/memory/dedupe.d.ts.map +1 -0
  222. package/dist/core/memory/dedupe.js +36 -0
  223. package/dist/core/memory/dedupe.js.map +1 -0
  224. package/dist/core/memory/embedding-adapter.d.ts +20 -0
  225. package/dist/core/memory/embedding-adapter.d.ts.map +1 -0
  226. package/dist/core/memory/embedding-adapter.js +11 -0
  227. package/dist/core/memory/embedding-adapter.js.map +1 -0
  228. package/dist/core/memory/event-summary.d.ts +7 -0
  229. package/dist/core/memory/event-summary.d.ts.map +1 -0
  230. package/dist/core/memory/event-summary.js +23 -0
  231. package/dist/core/memory/event-summary.js.map +1 -0
  232. package/dist/core/memory/feature/feature-format.d.ts +30 -0
  233. package/dist/core/memory/feature/feature-format.d.ts.map +1 -0
  234. package/dist/core/memory/feature/feature-format.js +121 -0
  235. package/dist/core/memory/feature/feature-format.js.map +1 -0
  236. package/dist/core/memory/feature/index.d.ts +2 -0
  237. package/dist/core/memory/feature/index.d.ts.map +1 -0
  238. package/dist/core/memory/feature/index.js +2 -0
  239. package/dist/core/memory/feature/index.js.map +1 -0
  240. package/dist/core/memory/ignore.d.ts +5 -0
  241. package/dist/core/memory/ignore.d.ts.map +1 -0
  242. package/dist/core/memory/ignore.js +26 -0
  243. package/dist/core/memory/ignore.js.map +1 -0
  244. package/dist/core/memory/index.d.ts +21 -0
  245. package/dist/core/memory/index.d.ts.map +1 -0
  246. package/dist/core/memory/index.js +20 -0
  247. package/dist/core/memory/index.js.map +1 -0
  248. package/dist/core/memory/memory-anchors.d.ts +11 -0
  249. package/dist/core/memory/memory-anchors.d.ts.map +1 -0
  250. package/dist/core/memory/memory-anchors.js +35 -0
  251. package/dist/core/memory/memory-anchors.js.map +1 -0
  252. package/dist/core/memory/memory-identity.d.ts +11 -0
  253. package/dist/core/memory/memory-identity.d.ts.map +1 -0
  254. package/dist/core/memory/memory-identity.js +19 -0
  255. package/dist/core/memory/memory-identity.js.map +1 -0
  256. package/dist/core/memory/memory-retrieval.d.ts +55 -0
  257. package/dist/core/memory/memory-retrieval.d.ts.map +1 -0
  258. package/dist/core/memory/memory-retrieval.js +128 -0
  259. package/dist/core/memory/memory-retrieval.js.map +1 -0
  260. package/dist/core/memory/memory-schema.d.ts +11 -0
  261. package/dist/core/memory/memory-schema.d.ts.map +1 -0
  262. package/dist/core/memory/memory-schema.js +111 -0
  263. package/dist/core/memory/memory-schema.js.map +1 -0
  264. package/dist/core/memory/memory-service.d.ts +35 -0
  265. package/dist/core/memory/memory-service.d.ts.map +1 -0
  266. package/dist/core/memory/memory-service.js +125 -0
  267. package/dist/core/memory/memory-service.js.map +1 -0
  268. package/dist/core/memory/memory-store.d.ts +37 -0
  269. package/dist/core/memory/memory-store.d.ts.map +1 -0
  270. package/dist/core/memory/memory-store.js +109 -0
  271. package/dist/core/memory/memory-store.js.map +1 -0
  272. package/dist/core/memory/memory-types.d.ts +55 -0
  273. package/dist/core/memory/memory-types.d.ts.map +1 -0
  274. package/dist/core/memory/memory-types.js +2 -0
  275. package/dist/core/memory/memory-types.js.map +1 -0
  276. package/dist/core/memory/redaction.d.ts +4 -0
  277. package/dist/core/memory/redaction.d.ts.map +1 -0
  278. package/dist/core/memory/redaction.js +49 -0
  279. package/dist/core/memory/redaction.js.map +1 -0
  280. package/dist/core/memory/sqlite-memory-store.d.ts +30 -0
  281. package/dist/core/memory/sqlite-memory-store.d.ts.map +1 -0
  282. package/dist/core/memory/sqlite-memory-store.js +300 -0
  283. package/dist/core/memory/sqlite-memory-store.js.map +1 -0
  284. package/dist/core/memory/startup-order.d.ts +9 -0
  285. package/dist/core/memory/startup-order.d.ts.map +1 -0
  286. package/dist/core/memory/startup-order.js +39 -0
  287. package/dist/core/memory/startup-order.js.map +1 -0
  288. package/dist/core/messages.d.ts +77 -0
  289. package/dist/core/messages.d.ts.map +1 -0
  290. package/dist/core/messages.js +123 -0
  291. package/dist/core/messages.js.map +1 -0
  292. package/dist/core/model-registry.d.ts +150 -0
  293. package/dist/core/model-registry.d.ts.map +1 -0
  294. package/dist/core/model-registry.js +727 -0
  295. package/dist/core/model-registry.js.map +1 -0
  296. package/dist/core/model-resolver.d.ts +110 -0
  297. package/dist/core/model-resolver.d.ts.map +1 -0
  298. package/dist/core/model-resolver.js +494 -0
  299. package/dist/core/model-resolver.js.map +1 -0
  300. package/dist/core/output-guard.d.ts +6 -0
  301. package/dist/core/output-guard.d.ts.map +1 -0
  302. package/dist/core/output-guard.js +59 -0
  303. package/dist/core/output-guard.js.map +1 -0
  304. package/dist/core/package-manager.d.ts +198 -0
  305. package/dist/core/package-manager.d.ts.map +1 -0
  306. package/dist/core/package-manager.js +1961 -0
  307. package/dist/core/package-manager.js.map +1 -0
  308. package/dist/core/prompt-templates.d.ts +52 -0
  309. package/dist/core/prompt-templates.d.ts.map +1 -0
  310. package/dist/core/prompt-templates.js +250 -0
  311. package/dist/core/prompt-templates.js.map +1 -0
  312. package/dist/core/provider-display-names.d.ts +2 -0
  313. package/dist/core/provider-display-names.d.ts.map +1 -0
  314. package/dist/core/provider-display-names.js +32 -0
  315. package/dist/core/provider-display-names.js.map +1 -0
  316. package/dist/core/resolve-config-value.d.ts +23 -0
  317. package/dist/core/resolve-config-value.d.ts.map +1 -0
  318. package/dist/core/resolve-config-value.js +126 -0
  319. package/dist/core/resolve-config-value.js.map +1 -0
  320. package/dist/core/resource-loader.d.ts +194 -0
  321. package/dist/core/resource-loader.d.ts.map +1 -0
  322. package/dist/core/resource-loader.js +731 -0
  323. package/dist/core/resource-loader.js.map +1 -0
  324. package/dist/core/sdk.d.ts +107 -0
  325. package/dist/core/sdk.d.ts.map +1 -0
  326. package/dist/core/sdk.js +282 -0
  327. package/dist/core/sdk.js.map +1 -0
  328. package/dist/core/session-cwd.d.ts +19 -0
  329. package/dist/core/session-cwd.d.ts.map +1 -0
  330. package/dist/core/session-cwd.js +38 -0
  331. package/dist/core/session-cwd.js.map +1 -0
  332. package/dist/core/session-manager.d.ts +333 -0
  333. package/dist/core/session-manager.d.ts.map +1 -0
  334. package/dist/core/session-manager.js +1109 -0
  335. package/dist/core/session-manager.js.map +1 -0
  336. package/dist/core/settings-manager.d.ts +295 -0
  337. package/dist/core/settings-manager.d.ts.map +1 -0
  338. package/dist/core/settings-manager.js +842 -0
  339. package/dist/core/settings-manager.js.map +1 -0
  340. package/dist/core/skills.d.ts +60 -0
  341. package/dist/core/skills.d.ts.map +1 -0
  342. package/dist/core/skills.js +404 -0
  343. package/dist/core/skills.js.map +1 -0
  344. package/dist/core/slash-commands.d.ts +14 -0
  345. package/dist/core/slash-commands.d.ts.map +1 -0
  346. package/dist/core/slash-commands.js +32 -0
  347. package/dist/core/slash-commands.js.map +1 -0
  348. package/dist/core/source-info.d.ts +18 -0
  349. package/dist/core/source-info.d.ts.map +1 -0
  350. package/dist/core/source-info.js +19 -0
  351. package/dist/core/source-info.js.map +1 -0
  352. package/dist/core/system-prompt.d.ts +28 -0
  353. package/dist/core/system-prompt.d.ts.map +1 -0
  354. package/dist/core/system-prompt.js +120 -0
  355. package/dist/core/system-prompt.js.map +1 -0
  356. package/dist/core/telemetry.d.ts +3 -0
  357. package/dist/core/telemetry.d.ts.map +1 -0
  358. package/dist/core/telemetry.js +9 -0
  359. package/dist/core/telemetry.js.map +1 -0
  360. package/dist/core/timings.d.ts +8 -0
  361. package/dist/core/timings.d.ts.map +1 -0
  362. package/dist/core/timings.js +31 -0
  363. package/dist/core/timings.js.map +1 -0
  364. package/dist/core/tools/bash.d.ts +68 -0
  365. package/dist/core/tools/bash.d.ts.map +1 -0
  366. package/dist/core/tools/bash.js +335 -0
  367. package/dist/core/tools/bash.js.map +1 -0
  368. package/dist/core/tools/edit-diff.d.ts +85 -0
  369. package/dist/core/tools/edit-diff.d.ts.map +1 -0
  370. package/dist/core/tools/edit-diff.js +338 -0
  371. package/dist/core/tools/edit-diff.js.map +1 -0
  372. package/dist/core/tools/edit.d.ts +49 -0
  373. package/dist/core/tools/edit.d.ts.map +1 -0
  374. package/dist/core/tools/edit.js +324 -0
  375. package/dist/core/tools/edit.js.map +1 -0
  376. package/dist/core/tools/file-mutation-queue.d.ts +6 -0
  377. package/dist/core/tools/file-mutation-queue.d.ts.map +1 -0
  378. package/dist/core/tools/file-mutation-queue.js +37 -0
  379. package/dist/core/tools/file-mutation-queue.js.map +1 -0
  380. package/dist/core/tools/find.d.ts +35 -0
  381. package/dist/core/tools/find.d.ts.map +1 -0
  382. package/dist/core/tools/find.js +298 -0
  383. package/dist/core/tools/find.js.map +1 -0
  384. package/dist/core/tools/grep.d.ts +37 -0
  385. package/dist/core/tools/grep.d.ts.map +1 -0
  386. package/dist/core/tools/grep.js +304 -0
  387. package/dist/core/tools/grep.js.map +1 -0
  388. package/dist/core/tools/index.d.ts +40 -0
  389. package/dist/core/tools/index.d.ts.map +1 -0
  390. package/dist/core/tools/index.js +112 -0
  391. package/dist/core/tools/index.js.map +1 -0
  392. package/dist/core/tools/ls.d.ts +37 -0
  393. package/dist/core/tools/ls.d.ts.map +1 -0
  394. package/dist/core/tools/ls.js +169 -0
  395. package/dist/core/tools/ls.js.map +1 -0
  396. package/dist/core/tools/output-accumulator.d.ts +50 -0
  397. package/dist/core/tools/output-accumulator.d.ts.map +1 -0
  398. package/dist/core/tools/output-accumulator.js +178 -0
  399. package/dist/core/tools/output-accumulator.js.map +1 -0
  400. package/dist/core/tools/path-utils.d.ts +8 -0
  401. package/dist/core/tools/path-utils.d.ts.map +1 -0
  402. package/dist/core/tools/path-utils.js +81 -0
  403. package/dist/core/tools/path-utils.js.map +1 -0
  404. package/dist/core/tools/read.d.ts +35 -0
  405. package/dist/core/tools/read.d.ts.map +1 -0
  406. package/dist/core/tools/read.js +288 -0
  407. package/dist/core/tools/read.js.map +1 -0
  408. package/dist/core/tools/render-utils.d.ts +21 -0
  409. package/dist/core/tools/render-utils.d.ts.map +1 -0
  410. package/dist/core/tools/render-utils.js +49 -0
  411. package/dist/core/tools/render-utils.js.map +1 -0
  412. package/dist/core/tools/tool-definition-wrapper.d.ts +14 -0
  413. package/dist/core/tools/tool-definition-wrapper.d.ts.map +1 -0
  414. package/dist/core/tools/tool-definition-wrapper.js +34 -0
  415. package/dist/core/tools/tool-definition-wrapper.js.map +1 -0
  416. package/dist/core/tools/truncate.d.ts +70 -0
  417. package/dist/core/tools/truncate.d.ts.map +1 -0
  418. package/dist/core/tools/truncate.js +205 -0
  419. package/dist/core/tools/truncate.js.map +1 -0
  420. package/dist/core/tools/write.d.ts +26 -0
  421. package/dist/core/tools/write.d.ts.map +1 -0
  422. package/dist/core/tools/write.js +213 -0
  423. package/dist/core/tools/write.js.map +1 -0
  424. package/dist/index.d.ts +28 -0
  425. package/dist/index.d.ts.map +1 -0
  426. package/dist/index.js +41 -0
  427. package/dist/index.js.map +1 -0
  428. package/dist/main.d.ts +12 -0
  429. package/dist/main.d.ts.map +1 -0
  430. package/dist/main.js +583 -0
  431. package/dist/main.js.map +1 -0
  432. package/dist/migrations.d.ts +31 -0
  433. package/dist/migrations.d.ts.map +1 -0
  434. package/dist/migrations.js +279 -0
  435. package/dist/migrations.js.map +1 -0
  436. package/dist/modes/index.d.ts +9 -0
  437. package/dist/modes/index.d.ts.map +1 -0
  438. package/dist/modes/index.js +8 -0
  439. package/dist/modes/index.js.map +1 -0
  440. package/dist/modes/interactive/assets/clankolas.png +0 -0
  441. package/dist/modes/interactive/components/armin.d.ts +34 -0
  442. package/dist/modes/interactive/components/armin.d.ts.map +1 -0
  443. package/dist/modes/interactive/components/armin.js +333 -0
  444. package/dist/modes/interactive/components/armin.js.map +1 -0
  445. package/dist/modes/interactive/components/assistant-message.d.ts +20 -0
  446. package/dist/modes/interactive/components/assistant-message.d.ts.map +1 -0
  447. package/dist/modes/interactive/components/assistant-message.js +122 -0
  448. package/dist/modes/interactive/components/assistant-message.js.map +1 -0
  449. package/dist/modes/interactive/components/bash-execution.d.ts +36 -0
  450. package/dist/modes/interactive/components/bash-execution.d.ts.map +1 -0
  451. package/dist/modes/interactive/components/bash-execution.js +182 -0
  452. package/dist/modes/interactive/components/bash-execution.js.map +1 -0
  453. package/dist/modes/interactive/components/bordered-loader.d.ts +16 -0
  454. package/dist/modes/interactive/components/bordered-loader.d.ts.map +1 -0
  455. package/dist/modes/interactive/components/bordered-loader.js +54 -0
  456. package/dist/modes/interactive/components/bordered-loader.js.map +1 -0
  457. package/dist/modes/interactive/components/branch-summary-message.d.ts +16 -0
  458. package/dist/modes/interactive/components/branch-summary-message.d.ts.map +1 -0
  459. package/dist/modes/interactive/components/branch-summary-message.js +44 -0
  460. package/dist/modes/interactive/components/branch-summary-message.js.map +1 -0
  461. package/dist/modes/interactive/components/compaction-summary-message.d.ts +16 -0
  462. package/dist/modes/interactive/components/compaction-summary-message.d.ts.map +1 -0
  463. package/dist/modes/interactive/components/compaction-summary-message.js +45 -0
  464. package/dist/modes/interactive/components/compaction-summary-message.js.map +1 -0
  465. package/dist/modes/interactive/components/config-selector.d.ts +71 -0
  466. package/dist/modes/interactive/components/config-selector.d.ts.map +1 -0
  467. package/dist/modes/interactive/components/config-selector.js +481 -0
  468. package/dist/modes/interactive/components/config-selector.js.map +1 -0
  469. package/dist/modes/interactive/components/countdown-timer.d.ts +14 -0
  470. package/dist/modes/interactive/components/countdown-timer.d.ts.map +1 -0
  471. package/dist/modes/interactive/components/countdown-timer.js +33 -0
  472. package/dist/modes/interactive/components/countdown-timer.js.map +1 -0
  473. package/dist/modes/interactive/components/custom-editor.d.ts +21 -0
  474. package/dist/modes/interactive/components/custom-editor.d.ts.map +1 -0
  475. package/dist/modes/interactive/components/custom-editor.js +70 -0
  476. package/dist/modes/interactive/components/custom-editor.js.map +1 -0
  477. package/dist/modes/interactive/components/custom-message.d.ts +20 -0
  478. package/dist/modes/interactive/components/custom-message.d.ts.map +1 -0
  479. package/dist/modes/interactive/components/custom-message.js +79 -0
  480. package/dist/modes/interactive/components/custom-message.js.map +1 -0
  481. package/dist/modes/interactive/components/daxnuts.d.ts +23 -0
  482. package/dist/modes/interactive/components/daxnuts.d.ts.map +1 -0
  483. package/dist/modes/interactive/components/daxnuts.js +140 -0
  484. package/dist/modes/interactive/components/daxnuts.js.map +1 -0
  485. package/dist/modes/interactive/components/diff.d.ts +12 -0
  486. package/dist/modes/interactive/components/diff.d.ts.map +1 -0
  487. package/dist/modes/interactive/components/diff.js +133 -0
  488. package/dist/modes/interactive/components/diff.js.map +1 -0
  489. package/dist/modes/interactive/components/dynamic-border.d.ts +26 -0
  490. package/dist/modes/interactive/components/dynamic-border.d.ts.map +1 -0
  491. package/dist/modes/interactive/components/dynamic-border.js +58 -0
  492. package/dist/modes/interactive/components/dynamic-border.js.map +1 -0
  493. package/dist/modes/interactive/components/earendil-announcement.d.ts +5 -0
  494. package/dist/modes/interactive/components/earendil-announcement.d.ts.map +1 -0
  495. package/dist/modes/interactive/components/earendil-announcement.js +40 -0
  496. package/dist/modes/interactive/components/earendil-announcement.js.map +1 -0
  497. package/dist/modes/interactive/components/extension-editor.d.ts +20 -0
  498. package/dist/modes/interactive/components/extension-editor.d.ts.map +1 -0
  499. package/dist/modes/interactive/components/extension-editor.js +111 -0
  500. package/dist/modes/interactive/components/extension-editor.js.map +1 -0
  501. package/dist/modes/interactive/components/extension-input.d.ts +23 -0
  502. package/dist/modes/interactive/components/extension-input.d.ts.map +1 -0
  503. package/dist/modes/interactive/components/extension-input.js +61 -0
  504. package/dist/modes/interactive/components/extension-input.js.map +1 -0
  505. package/dist/modes/interactive/components/extension-selector.d.ts +24 -0
  506. package/dist/modes/interactive/components/extension-selector.d.ts.map +1 -0
  507. package/dist/modes/interactive/components/extension-selector.js +78 -0
  508. package/dist/modes/interactive/components/extension-selector.js.map +1 -0
  509. package/dist/modes/interactive/components/flexible-viewport-spacer.d.ts +10 -0
  510. package/dist/modes/interactive/components/flexible-viewport-spacer.d.ts.map +1 -0
  511. package/dist/modes/interactive/components/flexible-viewport-spacer.js +20 -0
  512. package/dist/modes/interactive/components/flexible-viewport-spacer.js.map +1 -0
  513. package/dist/modes/interactive/components/footer.d.ts +27 -0
  514. package/dist/modes/interactive/components/footer.d.ts.map +1 -0
  515. package/dist/modes/interactive/components/footer.js +272 -0
  516. package/dist/modes/interactive/components/footer.js.map +1 -0
  517. package/dist/modes/interactive/components/index.d.ts +32 -0
  518. package/dist/modes/interactive/components/index.d.ts.map +1 -0
  519. package/dist/modes/interactive/components/index.js +33 -0
  520. package/dist/modes/interactive/components/index.js.map +1 -0
  521. package/dist/modes/interactive/components/keybinding-hints.d.ts +8 -0
  522. package/dist/modes/interactive/components/keybinding-hints.d.ts.map +1 -0
  523. package/dist/modes/interactive/components/keybinding-hints.js +22 -0
  524. package/dist/modes/interactive/components/keybinding-hints.js.map +1 -0
  525. package/dist/modes/interactive/components/login-dialog.d.ts +46 -0
  526. package/dist/modes/interactive/components/login-dialog.d.ts.map +1 -0
  527. package/dist/modes/interactive/components/login-dialog.js +160 -0
  528. package/dist/modes/interactive/components/login-dialog.js.map +1 -0
  529. package/dist/modes/interactive/components/model-selector.d.ts +47 -0
  530. package/dist/modes/interactive/components/model-selector.d.ts.map +1 -0
  531. package/dist/modes/interactive/components/model-selector.js +271 -0
  532. package/dist/modes/interactive/components/model-selector.js.map +1 -0
  533. package/dist/modes/interactive/components/oauth-selector.d.ts +31 -0
  534. package/dist/modes/interactive/components/oauth-selector.d.ts.map +1 -0
  535. package/dist/modes/interactive/components/oauth-selector.js +165 -0
  536. package/dist/modes/interactive/components/oauth-selector.js.map +1 -0
  537. package/dist/modes/interactive/components/scoped-models-selector.d.ts +42 -0
  538. package/dist/modes/interactive/components/scoped-models-selector.d.ts.map +1 -0
  539. package/dist/modes/interactive/components/scoped-models-selector.js +290 -0
  540. package/dist/modes/interactive/components/scoped-models-selector.js.map +1 -0
  541. package/dist/modes/interactive/components/session-selector-search.d.ts +23 -0
  542. package/dist/modes/interactive/components/session-selector-search.d.ts.map +1 -0
  543. package/dist/modes/interactive/components/session-selector-search.js +155 -0
  544. package/dist/modes/interactive/components/session-selector-search.js.map +1 -0
  545. package/dist/modes/interactive/components/session-selector.d.ts +96 -0
  546. package/dist/modes/interactive/components/session-selector.d.ts.map +1 -0
  547. package/dist/modes/interactive/components/session-selector.js +861 -0
  548. package/dist/modes/interactive/components/session-selector.js.map +1 -0
  549. package/dist/modes/interactive/components/settings-selector.d.ts +73 -0
  550. package/dist/modes/interactive/components/settings-selector.d.ts.map +1 -0
  551. package/dist/modes/interactive/components/settings-selector.js +531 -0
  552. package/dist/modes/interactive/components/settings-selector.js.map +1 -0
  553. package/dist/modes/interactive/components/show-images-selector.d.ts +10 -0
  554. package/dist/modes/interactive/components/show-images-selector.d.ts.map +1 -0
  555. package/dist/modes/interactive/components/show-images-selector.js +39 -0
  556. package/dist/modes/interactive/components/show-images-selector.js.map +1 -0
  557. package/dist/modes/interactive/components/skill-invocation-message.d.ts +17 -0
  558. package/dist/modes/interactive/components/skill-invocation-message.d.ts.map +1 -0
  559. package/dist/modes/interactive/components/skill-invocation-message.js +47 -0
  560. package/dist/modes/interactive/components/skill-invocation-message.js.map +1 -0
  561. package/dist/modes/interactive/components/split-info-card.d.ts +18 -0
  562. package/dist/modes/interactive/components/split-info-card.d.ts.map +1 -0
  563. package/dist/modes/interactive/components/split-info-card.js +82 -0
  564. package/dist/modes/interactive/components/split-info-card.js.map +1 -0
  565. package/dist/modes/interactive/components/theme-selector.d.ts +11 -0
  566. package/dist/modes/interactive/components/theme-selector.d.ts.map +1 -0
  567. package/dist/modes/interactive/components/theme-selector.js +50 -0
  568. package/dist/modes/interactive/components/theme-selector.js.map +1 -0
  569. package/dist/modes/interactive/components/thinking-selector.d.ts +11 -0
  570. package/dist/modes/interactive/components/thinking-selector.d.ts.map +1 -0
  571. package/dist/modes/interactive/components/thinking-selector.js +51 -0
  572. package/dist/modes/interactive/components/thinking-selector.js.map +1 -0
  573. package/dist/modes/interactive/components/tool-execution.d.ts +66 -0
  574. package/dist/modes/interactive/components/tool-execution.d.ts.map +1 -0
  575. package/dist/modes/interactive/components/tool-execution.js +313 -0
  576. package/dist/modes/interactive/components/tool-execution.js.map +1 -0
  577. package/dist/modes/interactive/components/tree-selector.d.ts +89 -0
  578. package/dist/modes/interactive/components/tree-selector.d.ts.map +1 -0
  579. package/dist/modes/interactive/components/tree-selector.js +1092 -0
  580. package/dist/modes/interactive/components/tree-selector.js.map +1 -0
  581. package/dist/modes/interactive/components/user-message-selector.d.ts +30 -0
  582. package/dist/modes/interactive/components/user-message-selector.d.ts.map +1 -0
  583. package/dist/modes/interactive/components/user-message-selector.js +114 -0
  584. package/dist/modes/interactive/components/user-message-selector.js.map +1 -0
  585. package/dist/modes/interactive/components/user-message.d.ts +10 -0
  586. package/dist/modes/interactive/components/user-message.d.ts.map +1 -0
  587. package/dist/modes/interactive/components/user-message.js +29 -0
  588. package/dist/modes/interactive/components/user-message.js.map +1 -0
  589. package/dist/modes/interactive/components/visual-truncate.d.ts +24 -0
  590. package/dist/modes/interactive/components/visual-truncate.d.ts.map +1 -0
  591. package/dist/modes/interactive/components/visual-truncate.js +33 -0
  592. package/dist/modes/interactive/components/visual-truncate.js.map +1 -0
  593. package/dist/modes/interactive/interactive-mode.d.ts +369 -0
  594. package/dist/modes/interactive/interactive-mode.d.ts.map +1 -0
  595. package/dist/modes/interactive/interactive-mode.js +4899 -0
  596. package/dist/modes/interactive/interactive-mode.js.map +1 -0
  597. package/dist/modes/interactive/theme/dark.json +85 -0
  598. package/dist/modes/interactive/theme/light.json +84 -0
  599. package/dist/modes/interactive/theme/theme-schema.json +335 -0
  600. package/dist/modes/interactive/theme/theme.d.ts +81 -0
  601. package/dist/modes/interactive/theme/theme.d.ts.map +1 -0
  602. package/dist/modes/interactive/theme/theme.js +971 -0
  603. package/dist/modes/interactive/theme/theme.js.map +1 -0
  604. package/dist/modes/print-mode.d.ts +28 -0
  605. package/dist/modes/print-mode.d.ts.map +1 -0
  606. package/dist/modes/print-mode.js +131 -0
  607. package/dist/modes/print-mode.js.map +1 -0
  608. package/dist/modes/rpc/jsonl.d.ts +17 -0
  609. package/dist/modes/rpc/jsonl.d.ts.map +1 -0
  610. package/dist/modes/rpc/jsonl.js +49 -0
  611. package/dist/modes/rpc/jsonl.js.map +1 -0
  612. package/dist/modes/rpc/rpc-client.d.ts +224 -0
  613. package/dist/modes/rpc/rpc-client.d.ts.map +1 -0
  614. package/dist/modes/rpc/rpc-client.js +410 -0
  615. package/dist/modes/rpc/rpc-client.js.map +1 -0
  616. package/dist/modes/rpc/rpc-mode.d.ts +20 -0
  617. package/dist/modes/rpc/rpc-mode.d.ts.map +1 -0
  618. package/dist/modes/rpc/rpc-mode.js +601 -0
  619. package/dist/modes/rpc/rpc-mode.js.map +1 -0
  620. package/dist/modes/rpc/rpc-types.d.ts +419 -0
  621. package/dist/modes/rpc/rpc-types.d.ts.map +1 -0
  622. package/dist/modes/rpc/rpc-types.js +8 -0
  623. package/dist/modes/rpc/rpc-types.js.map +1 -0
  624. package/dist/package-manager-cli.d.ts +4 -0
  625. package/dist/package-manager-cli.d.ts.map +1 -0
  626. package/dist/package-manager-cli.js +460 -0
  627. package/dist/package-manager-cli.js.map +1 -0
  628. package/dist/utils/changelog.d.ts +21 -0
  629. package/dist/utils/changelog.d.ts.map +1 -0
  630. package/dist/utils/changelog.js +87 -0
  631. package/dist/utils/changelog.js.map +1 -0
  632. package/dist/utils/child-process.d.ts +12 -0
  633. package/dist/utils/child-process.d.ts.map +1 -0
  634. package/dist/utils/child-process.js +86 -0
  635. package/dist/utils/child-process.js.map +1 -0
  636. package/dist/utils/clipboard-image.d.ts +11 -0
  637. package/dist/utils/clipboard-image.d.ts.map +1 -0
  638. package/dist/utils/clipboard-image.js +245 -0
  639. package/dist/utils/clipboard-image.js.map +1 -0
  640. package/dist/utils/clipboard-native.d.ts +8 -0
  641. package/dist/utils/clipboard-native.d.ts.map +1 -0
  642. package/dist/utils/clipboard-native.js +14 -0
  643. package/dist/utils/clipboard-native.js.map +1 -0
  644. package/dist/utils/clipboard-text.d.ts +4 -0
  645. package/dist/utils/clipboard-text.d.ts.map +1 -0
  646. package/dist/utils/clipboard-text.js +37 -0
  647. package/dist/utils/clipboard-text.js.map +1 -0
  648. package/dist/utils/clipboard.d.ts +2 -0
  649. package/dist/utils/clipboard.d.ts.map +1 -0
  650. package/dist/utils/clipboard.js +117 -0
  651. package/dist/utils/clipboard.js.map +1 -0
  652. package/dist/utils/exif-orientation.d.ts +5 -0
  653. package/dist/utils/exif-orientation.d.ts.map +1 -0
  654. package/dist/utils/exif-orientation.js +158 -0
  655. package/dist/utils/exif-orientation.js.map +1 -0
  656. package/dist/utils/frontmatter.d.ts +8 -0
  657. package/dist/utils/frontmatter.d.ts.map +1 -0
  658. package/dist/utils/frontmatter.js +26 -0
  659. package/dist/utils/frontmatter.js.map +1 -0
  660. package/dist/utils/fs-watch.d.ts +5 -0
  661. package/dist/utils/fs-watch.d.ts.map +1 -0
  662. package/dist/utils/fs-watch.js +25 -0
  663. package/dist/utils/fs-watch.js.map +1 -0
  664. package/dist/utils/git.d.ts +26 -0
  665. package/dist/utils/git.d.ts.map +1 -0
  666. package/dist/utils/git.js +163 -0
  667. package/dist/utils/git.js.map +1 -0
  668. package/dist/utils/image-convert.d.ts +9 -0
  669. package/dist/utils/image-convert.d.ts.map +1 -0
  670. package/dist/utils/image-convert.js +39 -0
  671. package/dist/utils/image-convert.js.map +1 -0
  672. package/dist/utils/image-resize.d.ts +36 -0
  673. package/dist/utils/image-resize.d.ts.map +1 -0
  674. package/dist/utils/image-resize.js +137 -0
  675. package/dist/utils/image-resize.js.map +1 -0
  676. package/dist/utils/mime.d.ts +2 -0
  677. package/dist/utils/mime.d.ts.map +1 -0
  678. package/dist/utils/mime.js +26 -0
  679. package/dist/utils/mime.js.map +1 -0
  680. package/dist/utils/paths.d.ts +14 -0
  681. package/dist/utils/paths.d.ts.map +1 -0
  682. package/dist/utils/paths.js +34 -0
  683. package/dist/utils/paths.js.map +1 -0
  684. package/dist/utils/photon.d.ts +21 -0
  685. package/dist/utils/photon.d.ts.map +1 -0
  686. package/dist/utils/photon.js +121 -0
  687. package/dist/utils/photon.js.map +1 -0
  688. package/dist/utils/shell.d.ts +30 -0
  689. package/dist/utils/shell.d.ts.map +1 -0
  690. package/dist/utils/shell.js +190 -0
  691. package/dist/utils/shell.js.map +1 -0
  692. package/dist/utils/sleep.d.ts +5 -0
  693. package/dist/utils/sleep.d.ts.map +1 -0
  694. package/dist/utils/sleep.js +17 -0
  695. package/dist/utils/sleep.js.map +1 -0
  696. package/dist/utils/threadwell-user-agent.d.ts +2 -0
  697. package/dist/utils/threadwell-user-agent.d.ts.map +1 -0
  698. package/dist/utils/threadwell-user-agent.js +5 -0
  699. package/dist/utils/threadwell-user-agent.js.map +1 -0
  700. package/dist/utils/tools-manager.d.ts +3 -0
  701. package/dist/utils/tools-manager.d.ts.map +1 -0
  702. package/dist/utils/tools-manager.js +256 -0
  703. package/dist/utils/tools-manager.js.map +1 -0
  704. package/dist/utils/version-check.d.ts +14 -0
  705. package/dist/utils/version-check.d.ts.map +1 -0
  706. package/dist/utils/version-check.js +77 -0
  707. package/dist/utils/version-check.js.map +1 -0
  708. package/docs/compaction.md +394 -0
  709. package/docs/custom-provider.md +646 -0
  710. package/docs/development.md +71 -0
  711. package/docs/docs.json +148 -0
  712. package/docs/extensions.md +2596 -0
  713. package/docs/images/doom-extension.png +0 -0
  714. package/docs/images/exy.png +0 -0
  715. package/docs/images/interactive-mode.png +0 -0
  716. package/docs/images/tree-view.png +0 -0
  717. package/docs/index.md +64 -0
  718. package/docs/json.md +82 -0
  719. package/docs/keybindings.md +197 -0
  720. package/docs/memory.md +484 -0
  721. package/docs/models.md +474 -0
  722. package/docs/packages.md +223 -0
  723. package/docs/prompt-templates.md +88 -0
  724. package/docs/providers.md +241 -0
  725. package/docs/quickstart.md +142 -0
  726. package/docs/rpc.md +1407 -0
  727. package/docs/sdk.md +1149 -0
  728. package/docs/session-format.md +412 -0
  729. package/docs/sessions.md +137 -0
  730. package/docs/settings.md +378 -0
  731. package/docs/shell-aliases.md +13 -0
  732. package/docs/skills.md +232 -0
  733. package/docs/terminal-setup.md +106 -0
  734. package/docs/termux.md +127 -0
  735. package/docs/themes.md +295 -0
  736. package/docs/tmux.md +61 -0
  737. package/docs/tui.md +918 -0
  738. package/docs/usage.md +277 -0
  739. package/docs/windows.md +17 -0
  740. package/examples/README.md +25 -0
  741. package/examples/extensions/README.md +208 -0
  742. package/examples/extensions/auto-commit-on-exit.ts +49 -0
  743. package/examples/extensions/bash-spawn-hook.ts +30 -0
  744. package/examples/extensions/bookmark.ts +50 -0
  745. package/examples/extensions/border-status-editor.ts +147 -0
  746. package/examples/extensions/built-in-tool-renderer.ts +249 -0
  747. package/examples/extensions/claude-rules.ts +86 -0
  748. package/examples/extensions/commands.ts +72 -0
  749. package/examples/extensions/confirm-destructive.ts +59 -0
  750. package/examples/extensions/custom-compaction.ts +127 -0
  751. package/examples/extensions/custom-footer.ts +64 -0
  752. package/examples/extensions/custom-header.ts +73 -0
  753. package/examples/extensions/custom-provider-anthropic/index.ts +604 -0
  754. package/examples/extensions/custom-provider-anthropic/package-lock.json +24 -0
  755. package/examples/extensions/custom-provider-anthropic/package.json +19 -0
  756. package/examples/extensions/custom-provider-gitlab-duo/index.ts +349 -0
  757. package/examples/extensions/custom-provider-gitlab-duo/package.json +16 -0
  758. package/examples/extensions/custom-provider-gitlab-duo/test.ts +82 -0
  759. package/examples/extensions/dirty-repo-guard.ts +56 -0
  760. package/examples/extensions/doom-overlay/README.md +46 -0
  761. package/examples/extensions/doom-overlay/doom/build/doom.js +21 -0
  762. package/examples/extensions/doom-overlay/doom/build/doom.wasm +0 -0
  763. package/examples/extensions/doom-overlay/doom/build.sh +152 -0
  764. package/examples/extensions/doom-overlay/doom/doomgeneric_pi.c +72 -0
  765. package/examples/extensions/doom-overlay/doom-component.ts +132 -0
  766. package/examples/extensions/doom-overlay/doom-engine.ts +173 -0
  767. package/examples/extensions/doom-overlay/doom-keys.ts +104 -0
  768. package/examples/extensions/doom-overlay/index.ts +74 -0
  769. package/examples/extensions/doom-overlay/wad-finder.ts +51 -0
  770. package/examples/extensions/dynamic-resources/SKILL.md +8 -0
  771. package/examples/extensions/dynamic-resources/dynamic.json +79 -0
  772. package/examples/extensions/dynamic-resources/dynamic.md +5 -0
  773. package/examples/extensions/dynamic-resources/index.ts +15 -0
  774. package/examples/extensions/dynamic-tools.ts +74 -0
  775. package/examples/extensions/event-bus.ts +43 -0
  776. package/examples/extensions/file-trigger.ts +41 -0
  777. package/examples/extensions/git-checkpoint.ts +53 -0
  778. package/examples/extensions/github-issue-autocomplete.ts +185 -0
  779. package/examples/extensions/handoff.ts +191 -0
  780. package/examples/extensions/hello.ts +26 -0
  781. package/examples/extensions/hidden-thinking-label.ts +53 -0
  782. package/examples/extensions/inline-bash.ts +94 -0
  783. package/examples/extensions/input-transform.ts +43 -0
  784. package/examples/extensions/interactive-shell.ts +196 -0
  785. package/examples/extensions/mac-system-theme.ts +47 -0
  786. package/examples/extensions/message-renderer.ts +59 -0
  787. package/examples/extensions/minimal-mode.ts +426 -0
  788. package/examples/extensions/modal-editor.ts +85 -0
  789. package/examples/extensions/model-status.ts +31 -0
  790. package/examples/extensions/notify.ts +55 -0
  791. package/examples/extensions/overlay-qa-tests.ts +1348 -0
  792. package/examples/extensions/overlay-test.ts +150 -0
  793. package/examples/extensions/permission-gate.ts +34 -0
  794. package/examples/extensions/pirate.ts +47 -0
  795. package/examples/extensions/plan-mode/README.md +65 -0
  796. package/examples/extensions/plan-mode/index.ts +340 -0
  797. package/examples/extensions/plan-mode/utils.ts +168 -0
  798. package/examples/extensions/preset.ts +436 -0
  799. package/examples/extensions/prompt-customizer.ts +97 -0
  800. package/examples/extensions/protected-paths.ts +30 -0
  801. package/examples/extensions/provider-payload.ts +18 -0
  802. package/examples/extensions/qna.ts +122 -0
  803. package/examples/extensions/question.ts +264 -0
  804. package/examples/extensions/questionnaire.ts +427 -0
  805. package/examples/extensions/rainbow-editor.ts +88 -0
  806. package/examples/extensions/reload-runtime.ts +37 -0
  807. package/examples/extensions/rpc-demo.ts +118 -0
  808. package/examples/extensions/sandbox/index.ts +321 -0
  809. package/examples/extensions/sandbox/package-lock.json +92 -0
  810. package/examples/extensions/sandbox/package.json +19 -0
  811. package/examples/extensions/send-user-message.ts +97 -0
  812. package/examples/extensions/session-name.ts +27 -0
  813. package/examples/extensions/shutdown-command.ts +63 -0
  814. package/examples/extensions/snake.ts +343 -0
  815. package/examples/extensions/space-invaders.ts +560 -0
  816. package/examples/extensions/ssh.ts +220 -0
  817. package/examples/extensions/status-line.ts +32 -0
  818. package/examples/extensions/structured-output.ts +65 -0
  819. package/examples/extensions/subagent/README.md +172 -0
  820. package/examples/extensions/subagent/agents/planner.md +37 -0
  821. package/examples/extensions/subagent/agents/reviewer.md +35 -0
  822. package/examples/extensions/subagent/agents/scout.md +50 -0
  823. package/examples/extensions/subagent/agents/worker.md +24 -0
  824. package/examples/extensions/subagent/agents.ts +126 -0
  825. package/examples/extensions/subagent/index.ts +987 -0
  826. package/examples/extensions/subagent/prompts/implement-and-review.md +10 -0
  827. package/examples/extensions/subagent/prompts/implement.md +10 -0
  828. package/examples/extensions/subagent/prompts/scout-and-plan.md +9 -0
  829. package/examples/extensions/summarize.ts +206 -0
  830. package/examples/extensions/system-prompt-header.ts +17 -0
  831. package/examples/extensions/tic-tac-toe.ts +1008 -0
  832. package/examples/extensions/timed-confirm.ts +70 -0
  833. package/examples/extensions/titlebar-spinner.ts +58 -0
  834. package/examples/extensions/todo.ts +297 -0
  835. package/examples/extensions/tool-override.ts +144 -0
  836. package/examples/extensions/tools.ts +141 -0
  837. package/examples/extensions/trigger-compact.ts +50 -0
  838. package/examples/extensions/truncated-tool.ts +195 -0
  839. package/examples/extensions/widget-placement.ts +9 -0
  840. package/examples/extensions/with-deps/index.ts +32 -0
  841. package/examples/extensions/with-deps/package-lock.json +31 -0
  842. package/examples/extensions/with-deps/package.json +22 -0
  843. package/examples/extensions/working-indicator.ts +123 -0
  844. package/examples/extensions/working-message-test.ts +25 -0
  845. package/examples/rpc-extension-ui.ts +632 -0
  846. package/examples/sdk/01-minimal.ts +22 -0
  847. package/examples/sdk/02-custom-model.ts +49 -0
  848. package/examples/sdk/03-custom-prompt.ts +62 -0
  849. package/examples/sdk/04-skills.ts +55 -0
  850. package/examples/sdk/05-tools.ts +44 -0
  851. package/examples/sdk/06-extensions.ts +90 -0
  852. package/examples/sdk/07-context-files.ts +42 -0
  853. package/examples/sdk/08-prompt-templates.ts +51 -0
  854. package/examples/sdk/09-api-keys-and-oauth.ts +48 -0
  855. package/examples/sdk/10-settings.ts +53 -0
  856. package/examples/sdk/11-sessions.ts +48 -0
  857. package/examples/sdk/12-full-control.ts +73 -0
  858. package/examples/sdk/13-session-runtime.ts +67 -0
  859. package/examples/sdk/README.md +147 -0
  860. package/package.json +101 -33
  861. package/LICENSE +0 -21
@@ -0,0 +1 @@
1
+ {"version":3,"file":"shell.d.ts","sourceRoot":"","sources":["../../src/utils/shell.ts"],"names":[],"mappings":"AAKA,MAAM,WAAW,WAAW;IAC3B,KAAK,EAAE,MAAM,CAAC;IACd,IAAI,EAAE,MAAM,EAAE,CAAC;CACf;AAqCD;;;;;;GAMG;AACH,wBAAgB,cAAc,CAAC,eAAe,CAAC,EAAE,MAAM,GAAG,WAAW,CAqDpE;AAED,wBAAgB,WAAW,IAAI,MAAM,CAAC,UAAU,CAY/C;AAED;;;;;;;GAOG;AACH,wBAAgB,oBAAoB,CAAC,GAAG,EAAE,MAAM,GAAG,MAAM,CA8BxD;AAQD,wBAAgB,qBAAqB,CAAC,GAAG,EAAE,MAAM,GAAG,IAAI,CAEvD;AAED,wBAAgB,uBAAuB,CAAC,GAAG,EAAE,MAAM,GAAG,IAAI,CAEzD;AAED,wBAAgB,2BAA2B,IAAI,IAAI,CAKlD;AAED;;GAEG;AACH,wBAAgB,eAAe,CAAC,GAAG,EAAE,MAAM,GAAG,IAAI,CAwBjD","sourcesContent":["import { existsSync } from \"node:fs\";\nimport { delimiter } from \"node:path\";\nimport { spawn, spawnSync } from \"child_process\";\nimport { getBinDir } from \"../config.js\";\n\nexport interface ShellConfig {\n\tshell: string;\n\targs: string[];\n}\n\n/**\n * Find bash executable on PATH (cross-platform)\n */\nfunction findBashOnPath(): string | null {\n\tif (process.platform === \"win32\") {\n\t\t// Windows: Use 'where' and verify file exists (where can return non-existent paths)\n\t\ttry {\n\t\t\tconst result = spawnSync(\"where\", [\"bash.exe\"], { encoding: \"utf-8\", timeout: 5000 });\n\t\t\tif (result.status === 0 && result.stdout) {\n\t\t\t\tconst firstMatch = result.stdout.trim().split(/\\r?\\n/)[0];\n\t\t\t\tif (firstMatch && existsSync(firstMatch)) {\n\t\t\t\t\treturn firstMatch;\n\t\t\t\t}\n\t\t\t}\n\t\t} catch {\n\t\t\t// Ignore errors\n\t\t}\n\t\treturn null;\n\t}\n\n\t// Unix: Use 'which' and trust its output (handles Termux and special filesystems)\n\ttry {\n\t\tconst result = spawnSync(\"which\", [\"bash\"], { encoding: \"utf-8\", timeout: 5000 });\n\t\tif (result.status === 0 && result.stdout) {\n\t\t\tconst firstMatch = result.stdout.trim().split(/\\r?\\n/)[0];\n\t\t\tif (firstMatch) {\n\t\t\t\treturn firstMatch;\n\t\t\t}\n\t\t}\n\t} catch {\n\t\t// Ignore errors\n\t}\n\treturn null;\n}\n\n/**\n * Resolve shell configuration based on platform and an optional explicit shell path.\n * Resolution order:\n * 1. User-specified shellPath\n * 2. On Windows: Git Bash in known locations, then bash on PATH\n * 3. On Unix: /bin/bash, then bash on PATH, then fallback to sh\n */\nexport function getShellConfig(customShellPath?: string): ShellConfig {\n\t// 1. Check user-specified shell path\n\tif (customShellPath) {\n\t\tif (existsSync(customShellPath)) {\n\t\t\treturn { shell: customShellPath, args: [\"-c\"] };\n\t\t}\n\t\tthrow new Error(`Custom shell path not found: ${customShellPath}`);\n\t}\n\n\tif (process.platform === \"win32\") {\n\t\t// 2. Try Git Bash in known locations\n\t\tconst paths: string[] = [];\n\t\tconst programFiles = process.env.ProgramFiles;\n\t\tif (programFiles) {\n\t\t\tpaths.push(`${programFiles}\\\\Git\\\\bin\\\\bash.exe`);\n\t\t}\n\t\tconst programFilesX86 = process.env[\"ProgramFiles(x86)\"];\n\t\tif (programFilesX86) {\n\t\t\tpaths.push(`${programFilesX86}\\\\Git\\\\bin\\\\bash.exe`);\n\t\t}\n\n\t\tfor (const path of paths) {\n\t\t\tif (existsSync(path)) {\n\t\t\t\treturn { shell: path, args: [\"-c\"] };\n\t\t\t}\n\t\t}\n\n\t\t// 3. Fallback: search bash.exe on PATH (Cygwin, MSYS2, WSL, etc.)\n\t\tconst bashOnPath = findBashOnPath();\n\t\tif (bashOnPath) {\n\t\t\treturn { shell: bashOnPath, args: [\"-c\"] };\n\t\t}\n\n\t\tthrow new Error(\n\t\t\t`No bash shell found. Options:\\n` +\n\t\t\t\t` 1. Install Git for Windows: https://git-scm.com/download/win\\n` +\n\t\t\t\t` 2. Add your bash to PATH (Cygwin, MSYS2, etc.)\\n` +\n\t\t\t\t\" 3. Set shellPath in settings.json\\n\\n\" +\n\t\t\t\t`Searched Git Bash in:\\n${paths.map((p) => ` ${p}`).join(\"\\n\")}`,\n\t\t);\n\t}\n\n\t// Unix: try /bin/bash, then bash on PATH, then fallback to sh\n\tif (existsSync(\"/bin/bash\")) {\n\t\treturn { shell: \"/bin/bash\", args: [\"-c\"] };\n\t}\n\n\tconst bashOnPath = findBashOnPath();\n\tif (bashOnPath) {\n\t\treturn { shell: bashOnPath, args: [\"-c\"] };\n\t}\n\n\treturn { shell: \"sh\", args: [\"-c\"] };\n}\n\nexport function getShellEnv(): NodeJS.ProcessEnv {\n\tconst binDir = getBinDir();\n\tconst pathKey = Object.keys(process.env).find((key) => key.toLowerCase() === \"path\") ?? \"PATH\";\n\tconst currentPath = process.env[pathKey] ?? \"\";\n\tconst pathEntries = currentPath.split(delimiter).filter(Boolean);\n\tconst hasBinDir = pathEntries.includes(binDir);\n\tconst updatedPath = hasBinDir ? currentPath : [binDir, currentPath].filter(Boolean).join(delimiter);\n\n\treturn {\n\t\t...process.env,\n\t\t[pathKey]: updatedPath,\n\t};\n}\n\n/**\n * Sanitize binary output for display/storage.\n * Removes characters that crash string-width or cause display issues:\n * - Control characters (except tab, newline, carriage return)\n * - Lone surrogates\n * - Unicode Format characters (crash string-width due to a bug)\n * - Characters with undefined code points\n */\nexport function sanitizeBinaryOutput(str: string): string {\n\t// Use Array.from to properly iterate over code points (not code units)\n\t// This handles surrogate pairs correctly and catches edge cases where\n\t// codePointAt() might return undefined\n\treturn Array.from(str)\n\t\t.filter((char) => {\n\t\t\t// Filter out characters that cause string-width to crash\n\t\t\t// This includes:\n\t\t\t// - Unicode format characters\n\t\t\t// - Lone surrogates (already filtered by Array.from)\n\t\t\t// - Control chars except \\t \\n \\r\n\t\t\t// - Characters with undefined code points\n\n\t\t\tconst code = char.codePointAt(0);\n\n\t\t\t// Skip if code point is undefined (edge case with invalid strings)\n\t\t\tif (code === undefined) return false;\n\n\t\t\t// Allow tab, newline, carriage return\n\t\t\tif (code === 0x09 || code === 0x0a || code === 0x0d) return true;\n\n\t\t\t// Filter out control characters (0x00-0x1F, except 0x09, 0x0a, 0x0x0d)\n\t\t\tif (code <= 0x1f) return false;\n\n\t\t\t// Filter out Unicode format characters\n\t\t\tif (code >= 0xfff9 && code <= 0xfffb) return false;\n\n\t\t\treturn true;\n\t\t})\n\t\t.join(\"\");\n}\n\n/**\n * Detached child processes must be tracked so they can be killed on parent\n * shutdown signals (SIGHUP/SIGTERM).\n */\nconst trackedDetachedChildPids = new Set<number>();\n\nexport function trackDetachedChildPid(pid: number): void {\n\ttrackedDetachedChildPids.add(pid);\n}\n\nexport function untrackDetachedChildPid(pid: number): void {\n\ttrackedDetachedChildPids.delete(pid);\n}\n\nexport function killTrackedDetachedChildren(): void {\n\tfor (const pid of trackedDetachedChildPids) {\n\t\tkillProcessTree(pid);\n\t}\n\ttrackedDetachedChildPids.clear();\n}\n\n/**\n * Kill a process and all its children (cross-platform)\n */\nexport function killProcessTree(pid: number): void {\n\tif (process.platform === \"win32\") {\n\t\t// Use taskkill on Windows to kill process tree\n\t\ttry {\n\t\t\tspawn(\"taskkill\", [\"/F\", \"/T\", \"/PID\", String(pid)], {\n\t\t\t\tstdio: \"ignore\",\n\t\t\t\tdetached: true,\n\t\t\t});\n\t\t} catch {\n\t\t\t// Ignore errors if taskkill fails\n\t\t}\n\t} else {\n\t\t// Use SIGKILL on Unix/Linux/Mac\n\t\ttry {\n\t\t\tprocess.kill(-pid, \"SIGKILL\");\n\t\t} catch {\n\t\t\t// Fallback to killing just the child if process group kill fails\n\t\t\ttry {\n\t\t\t\tprocess.kill(pid, \"SIGKILL\");\n\t\t\t} catch {\n\t\t\t\t// Process already dead\n\t\t\t}\n\t\t}\n\t}\n}\n"]}
@@ -0,0 +1,190 @@
1
+ import { existsSync } from "node:fs";
2
+ import { delimiter } from "node:path";
3
+ import { spawn, spawnSync } from "child_process";
4
+ import { getBinDir } from "../config.js";
5
+ /**
6
+ * Find bash executable on PATH (cross-platform)
7
+ */
8
+ function findBashOnPath() {
9
+ if (process.platform === "win32") {
10
+ // Windows: Use 'where' and verify file exists (where can return non-existent paths)
11
+ try {
12
+ const result = spawnSync("where", ["bash.exe"], { encoding: "utf-8", timeout: 5000 });
13
+ if (result.status === 0 && result.stdout) {
14
+ const firstMatch = result.stdout.trim().split(/\r?\n/)[0];
15
+ if (firstMatch && existsSync(firstMatch)) {
16
+ return firstMatch;
17
+ }
18
+ }
19
+ }
20
+ catch {
21
+ // Ignore errors
22
+ }
23
+ return null;
24
+ }
25
+ // Unix: Use 'which' and trust its output (handles Termux and special filesystems)
26
+ try {
27
+ const result = spawnSync("which", ["bash"], { encoding: "utf-8", timeout: 5000 });
28
+ if (result.status === 0 && result.stdout) {
29
+ const firstMatch = result.stdout.trim().split(/\r?\n/)[0];
30
+ if (firstMatch) {
31
+ return firstMatch;
32
+ }
33
+ }
34
+ }
35
+ catch {
36
+ // Ignore errors
37
+ }
38
+ return null;
39
+ }
40
+ /**
41
+ * Resolve shell configuration based on platform and an optional explicit shell path.
42
+ * Resolution order:
43
+ * 1. User-specified shellPath
44
+ * 2. On Windows: Git Bash in known locations, then bash on PATH
45
+ * 3. On Unix: /bin/bash, then bash on PATH, then fallback to sh
46
+ */
47
+ export function getShellConfig(customShellPath) {
48
+ // 1. Check user-specified shell path
49
+ if (customShellPath) {
50
+ if (existsSync(customShellPath)) {
51
+ return { shell: customShellPath, args: ["-c"] };
52
+ }
53
+ throw new Error(`Custom shell path not found: ${customShellPath}`);
54
+ }
55
+ if (process.platform === "win32") {
56
+ // 2. Try Git Bash in known locations
57
+ const paths = [];
58
+ const programFiles = process.env.ProgramFiles;
59
+ if (programFiles) {
60
+ paths.push(`${programFiles}\\Git\\bin\\bash.exe`);
61
+ }
62
+ const programFilesX86 = process.env["ProgramFiles(x86)"];
63
+ if (programFilesX86) {
64
+ paths.push(`${programFilesX86}\\Git\\bin\\bash.exe`);
65
+ }
66
+ for (const path of paths) {
67
+ if (existsSync(path)) {
68
+ return { shell: path, args: ["-c"] };
69
+ }
70
+ }
71
+ // 3. Fallback: search bash.exe on PATH (Cygwin, MSYS2, WSL, etc.)
72
+ const bashOnPath = findBashOnPath();
73
+ if (bashOnPath) {
74
+ return { shell: bashOnPath, args: ["-c"] };
75
+ }
76
+ throw new Error(`No bash shell found. Options:\n` +
77
+ ` 1. Install Git for Windows: https://git-scm.com/download/win\n` +
78
+ ` 2. Add your bash to PATH (Cygwin, MSYS2, etc.)\n` +
79
+ " 3. Set shellPath in settings.json\n\n" +
80
+ `Searched Git Bash in:\n${paths.map((p) => ` ${p}`).join("\n")}`);
81
+ }
82
+ // Unix: try /bin/bash, then bash on PATH, then fallback to sh
83
+ if (existsSync("/bin/bash")) {
84
+ return { shell: "/bin/bash", args: ["-c"] };
85
+ }
86
+ const bashOnPath = findBashOnPath();
87
+ if (bashOnPath) {
88
+ return { shell: bashOnPath, args: ["-c"] };
89
+ }
90
+ return { shell: "sh", args: ["-c"] };
91
+ }
92
+ export function getShellEnv() {
93
+ const binDir = getBinDir();
94
+ const pathKey = Object.keys(process.env).find((key) => key.toLowerCase() === "path") ?? "PATH";
95
+ const currentPath = process.env[pathKey] ?? "";
96
+ const pathEntries = currentPath.split(delimiter).filter(Boolean);
97
+ const hasBinDir = pathEntries.includes(binDir);
98
+ const updatedPath = hasBinDir ? currentPath : [binDir, currentPath].filter(Boolean).join(delimiter);
99
+ return {
100
+ ...process.env,
101
+ [pathKey]: updatedPath,
102
+ };
103
+ }
104
+ /**
105
+ * Sanitize binary output for display/storage.
106
+ * Removes characters that crash string-width or cause display issues:
107
+ * - Control characters (except tab, newline, carriage return)
108
+ * - Lone surrogates
109
+ * - Unicode Format characters (crash string-width due to a bug)
110
+ * - Characters with undefined code points
111
+ */
112
+ export function sanitizeBinaryOutput(str) {
113
+ // Use Array.from to properly iterate over code points (not code units)
114
+ // This handles surrogate pairs correctly and catches edge cases where
115
+ // codePointAt() might return undefined
116
+ return Array.from(str)
117
+ .filter((char) => {
118
+ // Filter out characters that cause string-width to crash
119
+ // This includes:
120
+ // - Unicode format characters
121
+ // - Lone surrogates (already filtered by Array.from)
122
+ // - Control chars except \t \n \r
123
+ // - Characters with undefined code points
124
+ const code = char.codePointAt(0);
125
+ // Skip if code point is undefined (edge case with invalid strings)
126
+ if (code === undefined)
127
+ return false;
128
+ // Allow tab, newline, carriage return
129
+ if (code === 0x09 || code === 0x0a || code === 0x0d)
130
+ return true;
131
+ // Filter out control characters (0x00-0x1F, except 0x09, 0x0a, 0x0x0d)
132
+ if (code <= 0x1f)
133
+ return false;
134
+ // Filter out Unicode format characters
135
+ if (code >= 0xfff9 && code <= 0xfffb)
136
+ return false;
137
+ return true;
138
+ })
139
+ .join("");
140
+ }
141
+ /**
142
+ * Detached child processes must be tracked so they can be killed on parent
143
+ * shutdown signals (SIGHUP/SIGTERM).
144
+ */
145
+ const trackedDetachedChildPids = new Set();
146
+ export function trackDetachedChildPid(pid) {
147
+ trackedDetachedChildPids.add(pid);
148
+ }
149
+ export function untrackDetachedChildPid(pid) {
150
+ trackedDetachedChildPids.delete(pid);
151
+ }
152
+ export function killTrackedDetachedChildren() {
153
+ for (const pid of trackedDetachedChildPids) {
154
+ killProcessTree(pid);
155
+ }
156
+ trackedDetachedChildPids.clear();
157
+ }
158
+ /**
159
+ * Kill a process and all its children (cross-platform)
160
+ */
161
+ export function killProcessTree(pid) {
162
+ if (process.platform === "win32") {
163
+ // Use taskkill on Windows to kill process tree
164
+ try {
165
+ spawn("taskkill", ["/F", "/T", "/PID", String(pid)], {
166
+ stdio: "ignore",
167
+ detached: true,
168
+ });
169
+ }
170
+ catch {
171
+ // Ignore errors if taskkill fails
172
+ }
173
+ }
174
+ else {
175
+ // Use SIGKILL on Unix/Linux/Mac
176
+ try {
177
+ process.kill(-pid, "SIGKILL");
178
+ }
179
+ catch {
180
+ // Fallback to killing just the child if process group kill fails
181
+ try {
182
+ process.kill(pid, "SIGKILL");
183
+ }
184
+ catch {
185
+ // Process already dead
186
+ }
187
+ }
188
+ }
189
+ }
190
+ //# sourceMappingURL=shell.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"shell.js","sourceRoot":"","sources":["../../src/utils/shell.ts"],"names":[],"mappings":"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","sourcesContent":["import { existsSync } from \"node:fs\";\nimport { delimiter } from \"node:path\";\nimport { spawn, spawnSync } from \"child_process\";\nimport { getBinDir } from \"../config.js\";\n\nexport interface ShellConfig {\n\tshell: string;\n\targs: string[];\n}\n\n/**\n * Find bash executable on PATH (cross-platform)\n */\nfunction findBashOnPath(): string | null {\n\tif (process.platform === \"win32\") {\n\t\t// Windows: Use 'where' and verify file exists (where can return non-existent paths)\n\t\ttry {\n\t\t\tconst result = spawnSync(\"where\", [\"bash.exe\"], { encoding: \"utf-8\", timeout: 5000 });\n\t\t\tif (result.status === 0 && result.stdout) {\n\t\t\t\tconst firstMatch = result.stdout.trim().split(/\\r?\\n/)[0];\n\t\t\t\tif (firstMatch && existsSync(firstMatch)) {\n\t\t\t\t\treturn firstMatch;\n\t\t\t\t}\n\t\t\t}\n\t\t} catch {\n\t\t\t// Ignore errors\n\t\t}\n\t\treturn null;\n\t}\n\n\t// Unix: Use 'which' and trust its output (handles Termux and special filesystems)\n\ttry {\n\t\tconst result = spawnSync(\"which\", [\"bash\"], { encoding: \"utf-8\", timeout: 5000 });\n\t\tif (result.status === 0 && result.stdout) {\n\t\t\tconst firstMatch = result.stdout.trim().split(/\\r?\\n/)[0];\n\t\t\tif (firstMatch) {\n\t\t\t\treturn firstMatch;\n\t\t\t}\n\t\t}\n\t} catch {\n\t\t// Ignore errors\n\t}\n\treturn null;\n}\n\n/**\n * Resolve shell configuration based on platform and an optional explicit shell path.\n * Resolution order:\n * 1. User-specified shellPath\n * 2. On Windows: Git Bash in known locations, then bash on PATH\n * 3. On Unix: /bin/bash, then bash on PATH, then fallback to sh\n */\nexport function getShellConfig(customShellPath?: string): ShellConfig {\n\t// 1. Check user-specified shell path\n\tif (customShellPath) {\n\t\tif (existsSync(customShellPath)) {\n\t\t\treturn { shell: customShellPath, args: [\"-c\"] };\n\t\t}\n\t\tthrow new Error(`Custom shell path not found: ${customShellPath}`);\n\t}\n\n\tif (process.platform === \"win32\") {\n\t\t// 2. Try Git Bash in known locations\n\t\tconst paths: string[] = [];\n\t\tconst programFiles = process.env.ProgramFiles;\n\t\tif (programFiles) {\n\t\t\tpaths.push(`${programFiles}\\\\Git\\\\bin\\\\bash.exe`);\n\t\t}\n\t\tconst programFilesX86 = process.env[\"ProgramFiles(x86)\"];\n\t\tif (programFilesX86) {\n\t\t\tpaths.push(`${programFilesX86}\\\\Git\\\\bin\\\\bash.exe`);\n\t\t}\n\n\t\tfor (const path of paths) {\n\t\t\tif (existsSync(path)) {\n\t\t\t\treturn { shell: path, args: [\"-c\"] };\n\t\t\t}\n\t\t}\n\n\t\t// 3. Fallback: search bash.exe on PATH (Cygwin, MSYS2, WSL, etc.)\n\t\tconst bashOnPath = findBashOnPath();\n\t\tif (bashOnPath) {\n\t\t\treturn { shell: bashOnPath, args: [\"-c\"] };\n\t\t}\n\n\t\tthrow new Error(\n\t\t\t`No bash shell found. Options:\\n` +\n\t\t\t\t` 1. Install Git for Windows: https://git-scm.com/download/win\\n` +\n\t\t\t\t` 2. Add your bash to PATH (Cygwin, MSYS2, etc.)\\n` +\n\t\t\t\t\" 3. Set shellPath in settings.json\\n\\n\" +\n\t\t\t\t`Searched Git Bash in:\\n${paths.map((p) => ` ${p}`).join(\"\\n\")}`,\n\t\t);\n\t}\n\n\t// Unix: try /bin/bash, then bash on PATH, then fallback to sh\n\tif (existsSync(\"/bin/bash\")) {\n\t\treturn { shell: \"/bin/bash\", args: [\"-c\"] };\n\t}\n\n\tconst bashOnPath = findBashOnPath();\n\tif (bashOnPath) {\n\t\treturn { shell: bashOnPath, args: [\"-c\"] };\n\t}\n\n\treturn { shell: \"sh\", args: [\"-c\"] };\n}\n\nexport function getShellEnv(): NodeJS.ProcessEnv {\n\tconst binDir = getBinDir();\n\tconst pathKey = Object.keys(process.env).find((key) => key.toLowerCase() === \"path\") ?? \"PATH\";\n\tconst currentPath = process.env[pathKey] ?? \"\";\n\tconst pathEntries = currentPath.split(delimiter).filter(Boolean);\n\tconst hasBinDir = pathEntries.includes(binDir);\n\tconst updatedPath = hasBinDir ? currentPath : [binDir, currentPath].filter(Boolean).join(delimiter);\n\n\treturn {\n\t\t...process.env,\n\t\t[pathKey]: updatedPath,\n\t};\n}\n\n/**\n * Sanitize binary output for display/storage.\n * Removes characters that crash string-width or cause display issues:\n * - Control characters (except tab, newline, carriage return)\n * - Lone surrogates\n * - Unicode Format characters (crash string-width due to a bug)\n * - Characters with undefined code points\n */\nexport function sanitizeBinaryOutput(str: string): string {\n\t// Use Array.from to properly iterate over code points (not code units)\n\t// This handles surrogate pairs correctly and catches edge cases where\n\t// codePointAt() might return undefined\n\treturn Array.from(str)\n\t\t.filter((char) => {\n\t\t\t// Filter out characters that cause string-width to crash\n\t\t\t// This includes:\n\t\t\t// - Unicode format characters\n\t\t\t// - Lone surrogates (already filtered by Array.from)\n\t\t\t// - Control chars except \\t \\n \\r\n\t\t\t// - Characters with undefined code points\n\n\t\t\tconst code = char.codePointAt(0);\n\n\t\t\t// Skip if code point is undefined (edge case with invalid strings)\n\t\t\tif (code === undefined) return false;\n\n\t\t\t// Allow tab, newline, carriage return\n\t\t\tif (code === 0x09 || code === 0x0a || code === 0x0d) return true;\n\n\t\t\t// Filter out control characters (0x00-0x1F, except 0x09, 0x0a, 0x0x0d)\n\t\t\tif (code <= 0x1f) return false;\n\n\t\t\t// Filter out Unicode format characters\n\t\t\tif (code >= 0xfff9 && code <= 0xfffb) return false;\n\n\t\t\treturn true;\n\t\t})\n\t\t.join(\"\");\n}\n\n/**\n * Detached child processes must be tracked so they can be killed on parent\n * shutdown signals (SIGHUP/SIGTERM).\n */\nconst trackedDetachedChildPids = new Set<number>();\n\nexport function trackDetachedChildPid(pid: number): void {\n\ttrackedDetachedChildPids.add(pid);\n}\n\nexport function untrackDetachedChildPid(pid: number): void {\n\ttrackedDetachedChildPids.delete(pid);\n}\n\nexport function killTrackedDetachedChildren(): void {\n\tfor (const pid of trackedDetachedChildPids) {\n\t\tkillProcessTree(pid);\n\t}\n\ttrackedDetachedChildPids.clear();\n}\n\n/**\n * Kill a process and all its children (cross-platform)\n */\nexport function killProcessTree(pid: number): void {\n\tif (process.platform === \"win32\") {\n\t\t// Use taskkill on Windows to kill process tree\n\t\ttry {\n\t\t\tspawn(\"taskkill\", [\"/F\", \"/T\", \"/PID\", String(pid)], {\n\t\t\t\tstdio: \"ignore\",\n\t\t\t\tdetached: true,\n\t\t\t});\n\t\t} catch {\n\t\t\t// Ignore errors if taskkill fails\n\t\t}\n\t} else {\n\t\t// Use SIGKILL on Unix/Linux/Mac\n\t\ttry {\n\t\t\tprocess.kill(-pid, \"SIGKILL\");\n\t\t} catch {\n\t\t\t// Fallback to killing just the child if process group kill fails\n\t\t\ttry {\n\t\t\t\tprocess.kill(pid, \"SIGKILL\");\n\t\t\t} catch {\n\t\t\t\t// Process already dead\n\t\t\t}\n\t\t}\n\t}\n}\n"]}
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Sleep helper that respects abort signal.
3
+ */
4
+ export declare function sleep(ms: number, signal?: AbortSignal): Promise<void>;
5
+ //# sourceMappingURL=sleep.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sleep.d.ts","sourceRoot":"","sources":["../../src/utils/sleep.ts"],"names":[],"mappings":"AAAA;;GAEG;AACH,wBAAgB,KAAK,CAAC,EAAE,EAAE,MAAM,EAAE,MAAM,CAAC,EAAE,WAAW,GAAG,OAAO,CAAC,IAAI,CAAC,CAcrE","sourcesContent":["/**\n * Sleep helper that respects abort signal.\n */\nexport function sleep(ms: number, signal?: AbortSignal): Promise<void> {\n\treturn new Promise((resolve, reject) => {\n\t\tif (signal?.aborted) {\n\t\t\treject(new Error(\"Aborted\"));\n\t\t\treturn;\n\t\t}\n\n\t\tconst timeout = setTimeout(resolve, ms);\n\n\t\tsignal?.addEventListener(\"abort\", () => {\n\t\t\tclearTimeout(timeout);\n\t\t\treject(new Error(\"Aborted\"));\n\t\t});\n\t});\n}\n"]}
@@ -0,0 +1,17 @@
1
+ /**
2
+ * Sleep helper that respects abort signal.
3
+ */
4
+ export function sleep(ms, signal) {
5
+ return new Promise((resolve, reject) => {
6
+ if (signal?.aborted) {
7
+ reject(new Error("Aborted"));
8
+ return;
9
+ }
10
+ const timeout = setTimeout(resolve, ms);
11
+ signal?.addEventListener("abort", () => {
12
+ clearTimeout(timeout);
13
+ reject(new Error("Aborted"));
14
+ });
15
+ });
16
+ }
17
+ //# sourceMappingURL=sleep.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sleep.js","sourceRoot":"","sources":["../../src/utils/sleep.ts"],"names":[],"mappings":"AAAA;;GAEG;AACH,MAAM,UAAU,KAAK,CAAC,EAAU,EAAE,MAAoB,EAAiB;IACtE,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE,CAAC;QACvC,IAAI,MAAM,EAAE,OAAO,EAAE,CAAC;YACrB,MAAM,CAAC,IAAI,KAAK,CAAC,SAAS,CAAC,CAAC,CAAC;YAC7B,OAAO;QACR,CAAC;QAED,MAAM,OAAO,GAAG,UAAU,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC;QAExC,MAAM,EAAE,gBAAgB,CAAC,OAAO,EAAE,GAAG,EAAE,CAAC;YACvC,YAAY,CAAC,OAAO,CAAC,CAAC;YACtB,MAAM,CAAC,IAAI,KAAK,CAAC,SAAS,CAAC,CAAC,CAAC;QAAA,CAC7B,CAAC,CAAC;IAAA,CACH,CAAC,CAAC;AAAA,CACH","sourcesContent":["/**\n * Sleep helper that respects abort signal.\n */\nexport function sleep(ms: number, signal?: AbortSignal): Promise<void> {\n\treturn new Promise((resolve, reject) => {\n\t\tif (signal?.aborted) {\n\t\t\treject(new Error(\"Aborted\"));\n\t\t\treturn;\n\t\t}\n\n\t\tconst timeout = setTimeout(resolve, ms);\n\n\t\tsignal?.addEventListener(\"abort\", () => {\n\t\t\tclearTimeout(timeout);\n\t\t\treject(new Error(\"Aborted\"));\n\t\t});\n\t});\n}\n"]}
@@ -0,0 +1,2 @@
1
+ export declare function getThreadwellUserAgent(version: string): string;
2
+ //# sourceMappingURL=threadwell-user-agent.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"threadwell-user-agent.d.ts","sourceRoot":"","sources":["../../src/utils/threadwell-user-agent.ts"],"names":[],"mappings":"AAAA,wBAAgB,sBAAsB,CAAC,OAAO,EAAE,MAAM,GAAG,MAAM,CAG9D","sourcesContent":["export function getThreadwellUserAgent(version: string): string {\n\tconst runtime = process.versions.bun ? `bun/${process.versions.bun}` : `node/${process.version}`;\n\treturn `threadwell/${version} (${process.platform}; ${runtime}; ${process.arch})`;\n}\n"]}
@@ -0,0 +1,5 @@
1
+ export function getThreadwellUserAgent(version) {
2
+ const runtime = process.versions.bun ? `bun/${process.versions.bun}` : `node/${process.version}`;
3
+ return `threadwell/${version} (${process.platform}; ${runtime}; ${process.arch})`;
4
+ }
5
+ //# sourceMappingURL=threadwell-user-agent.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"threadwell-user-agent.js","sourceRoot":"","sources":["../../src/utils/threadwell-user-agent.ts"],"names":[],"mappings":"AAAA,MAAM,UAAU,sBAAsB,CAAC,OAAe,EAAU;IAC/D,MAAM,OAAO,GAAG,OAAO,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC,OAAO,OAAO,CAAC,QAAQ,CAAC,GAAG,EAAE,CAAC,CAAC,CAAC,QAAQ,OAAO,CAAC,OAAO,EAAE,CAAC;IACjG,OAAO,cAAc,OAAO,KAAK,OAAO,CAAC,QAAQ,KAAK,OAAO,KAAK,OAAO,CAAC,IAAI,GAAG,CAAC;AAAA,CAClF","sourcesContent":["export function getThreadwellUserAgent(version: string): string {\n\tconst runtime = process.versions.bun ? `bun/${process.versions.bun}` : `node/${process.version}`;\n\treturn `threadwell/${version} (${process.platform}; ${runtime}; ${process.arch})`;\n}\n"]}
@@ -0,0 +1,3 @@
1
+ export declare function getToolPath(tool: "fd" | "rg"): string | null;
2
+ export declare function ensureTool(tool: "fd" | "rg", silent?: boolean): Promise<string | undefined>;
3
+ //# sourceMappingURL=tools-manager.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tools-manager.d.ts","sourceRoot":"","sources":["../../src/utils/tools-manager.ts"],"names":[],"mappings":"AAqFA,wBAAgB,WAAW,CAAC,IAAI,EAAE,IAAI,GAAG,IAAI,GAAG,MAAM,GAAG,IAAI,CAmB5D;AAgJD,wBAAsB,UAAU,CAAC,IAAI,EAAE,IAAI,GAAG,IAAI,EAAE,MAAM,GAAE,OAAe,GAAG,OAAO,CAAC,MAAM,GAAG,SAAS,CAAC,CA2CxG","sourcesContent":["import chalk from \"chalk\";\nimport { spawnSync } from \"child_process\";\nimport extractZip from \"extract-zip\";\nimport { chmodSync, createWriteStream, existsSync, mkdirSync, readdirSync, renameSync, rmSync } from \"fs\";\nimport { arch, platform } from \"os\";\nimport { join } from \"path\";\nimport { Readable } from \"stream\";\nimport { pipeline } from \"stream/promises\";\nimport { APP_NAME, getBinDir } from \"../config.js\";\n\nconst TOOLS_DIR = getBinDir();\nconst NETWORK_TIMEOUT_MS = 10_000;\nconst DOWNLOAD_TIMEOUT_MS = 120_000;\n\nfunction isOfflineModeEnabled(): boolean {\n\tconst value = process.env.THREADWELL_OFFLINE;\n\tif (!value) return false;\n\treturn value === \"1\" || value.toLowerCase() === \"true\" || value.toLowerCase() === \"yes\";\n}\n\ninterface ToolConfig {\n\tname: string;\n\trepo: string; // GitHub repo (e.g., \"sharkdp/fd\")\n\tbinaryName: string; // Name of the binary inside the archive\n\tsystemBinaryNames?: string[]; // Alternative system command names to try before downloading\n\ttagPrefix: string; // Prefix for tags (e.g., \"v\" for v1.0.0, \"\" for 1.0.0)\n\tgetAssetName: (version: string, plat: string, architecture: string) => string | null;\n}\n\nconst TOOLS: Record<string, ToolConfig> = {\n\tfd: {\n\t\tname: \"fd\",\n\t\trepo: \"sharkdp/fd\",\n\t\tbinaryName: \"fd\",\n\t\tsystemBinaryNames: [\"fd\", \"fdfind\"],\n\t\ttagPrefix: \"v\",\n\t\tgetAssetName: (version, plat, architecture) => {\n\t\t\tif (plat === \"darwin\") {\n\t\t\t\tconst archStr = architecture === \"arm64\" ? \"aarch64\" : \"x86_64\";\n\t\t\t\treturn `fd-v${version}-${archStr}-apple-darwin.tar.gz`;\n\t\t\t} else if (plat === \"linux\") {\n\t\t\t\tconst archStr = architecture === \"arm64\" ? \"aarch64\" : \"x86_64\";\n\t\t\t\treturn `fd-v${version}-${archStr}-unknown-linux-gnu.tar.gz`;\n\t\t\t} else if (plat === \"win32\") {\n\t\t\t\tconst archStr = architecture === \"arm64\" ? \"aarch64\" : \"x86_64\";\n\t\t\t\treturn `fd-v${version}-${archStr}-pc-windows-msvc.zip`;\n\t\t\t}\n\t\t\treturn null;\n\t\t},\n\t},\n\trg: {\n\t\tname: \"ripgrep\",\n\t\trepo: \"BurntSushi/ripgrep\",\n\t\tbinaryName: \"rg\",\n\t\ttagPrefix: \"\",\n\t\tgetAssetName: (version, plat, architecture) => {\n\t\t\tif (plat === \"darwin\") {\n\t\t\t\tconst archStr = architecture === \"arm64\" ? \"aarch64\" : \"x86_64\";\n\t\t\t\treturn `ripgrep-${version}-${archStr}-apple-darwin.tar.gz`;\n\t\t\t} else if (plat === \"linux\") {\n\t\t\t\tif (architecture === \"arm64\") {\n\t\t\t\t\treturn `ripgrep-${version}-aarch64-unknown-linux-gnu.tar.gz`;\n\t\t\t\t}\n\t\t\t\treturn `ripgrep-${version}-x86_64-unknown-linux-musl.tar.gz`;\n\t\t\t} else if (plat === \"win32\") {\n\t\t\t\tconst archStr = architecture === \"arm64\" ? \"aarch64\" : \"x86_64\";\n\t\t\t\treturn `ripgrep-${version}-${archStr}-pc-windows-msvc.zip`;\n\t\t\t}\n\t\t\treturn null;\n\t\t},\n\t},\n};\n\n// Check if a command exists in PATH by trying to run it\nfunction commandExists(cmd: string): boolean {\n\ttry {\n\t\tconst result = spawnSync(cmd, [\"--version\"], { stdio: \"pipe\" });\n\t\t// Check for ENOENT error (command not found)\n\t\treturn result.error === undefined || result.error === null;\n\t} catch {\n\t\treturn false;\n\t}\n}\n\n// Get the path to a tool (system-wide or in our tools dir)\nexport function getToolPath(tool: \"fd\" | \"rg\"): string | null {\n\tconst config = TOOLS[tool];\n\tif (!config) return null;\n\n\t// Check our tools directory first\n\tconst localPath = join(TOOLS_DIR, config.binaryName + (platform() === \"win32\" ? \".exe\" : \"\"));\n\tif (existsSync(localPath)) {\n\t\treturn localPath;\n\t}\n\n\t// Check system PATH - if found, just return the command name (it's in PATH)\n\tconst systemBinaryNames = config.systemBinaryNames ?? [config.binaryName];\n\tfor (const systemBinaryName of systemBinaryNames) {\n\t\tif (commandExists(systemBinaryName)) {\n\t\t\treturn systemBinaryName;\n\t\t}\n\t}\n\n\treturn null;\n}\n\n// Fetch latest release version from GitHub\nasync function getLatestVersion(repo: string): Promise<string> {\n\tconst response = await fetch(`https://api.github.com/repos/${repo}/releases/latest`, {\n\t\theaders: { \"User-Agent\": `${APP_NAME}-coding-agent` },\n\t\tsignal: AbortSignal.timeout(NETWORK_TIMEOUT_MS),\n\t});\n\n\tif (!response.ok) {\n\t\tthrow new Error(`GitHub API error: ${response.status}`);\n\t}\n\n\tconst data = (await response.json()) as { tag_name: string };\n\treturn data.tag_name.replace(/^v/, \"\");\n}\n\n// Download a file from URL\nasync function downloadFile(url: string, dest: string): Promise<void> {\n\tconst response = await fetch(url, {\n\t\tsignal: AbortSignal.timeout(DOWNLOAD_TIMEOUT_MS),\n\t});\n\n\tif (!response.ok) {\n\t\tthrow new Error(`Failed to download: ${response.status}`);\n\t}\n\n\tif (!response.body) {\n\t\tthrow new Error(\"No response body\");\n\t}\n\n\tconst fileStream = createWriteStream(dest);\n\tawait pipeline(Readable.fromWeb(response.body as any), fileStream);\n}\n\nfunction findBinaryRecursively(rootDir: string, binaryFileName: string): string | null {\n\tconst stack: string[] = [rootDir];\n\n\twhile (stack.length > 0) {\n\t\tconst currentDir = stack.pop();\n\t\tif (!currentDir) continue;\n\n\t\tconst entries = readdirSync(currentDir, { withFileTypes: true });\n\t\tfor (const entry of entries) {\n\t\t\tconst fullPath = join(currentDir, entry.name);\n\t\t\tif (entry.isFile() && entry.name === binaryFileName) {\n\t\t\t\treturn fullPath;\n\t\t\t}\n\t\t\tif (entry.isDirectory()) {\n\t\t\t\tstack.push(fullPath);\n\t\t\t}\n\t\t}\n\t}\n\n\treturn null;\n}\n\n// Download and install a tool\nasync function downloadTool(tool: \"fd\" | \"rg\"): Promise<string> {\n\tconst config = TOOLS[tool];\n\tif (!config) throw new Error(`Unknown tool: ${tool}`);\n\n\tconst plat = platform();\n\tconst architecture = arch();\n\n\t// Get latest version\n\tconst version = await getLatestVersion(config.repo);\n\n\t// Get asset name for this platform\n\tconst assetName = config.getAssetName(version, plat, architecture);\n\tif (!assetName) {\n\t\tthrow new Error(`Unsupported platform: ${plat}/${architecture}`);\n\t}\n\n\t// Create tools directory\n\tmkdirSync(TOOLS_DIR, { recursive: true });\n\n\tconst downloadUrl = `https://github.com/${config.repo}/releases/download/${config.tagPrefix}${version}/${assetName}`;\n\tconst archivePath = join(TOOLS_DIR, assetName);\n\tconst binaryExt = plat === \"win32\" ? \".exe\" : \"\";\n\tconst binaryPath = join(TOOLS_DIR, config.binaryName + binaryExt);\n\n\t// Download\n\tawait downloadFile(downloadUrl, archivePath);\n\n\t// Extract into a unique temp directory. fd and rg downloads can run concurrently\n\t// during startup, so sharing a fixed directory causes races.\n\tconst extractDir = join(\n\t\tTOOLS_DIR,\n\t\t`extract_tmp_${config.binaryName}_${process.pid}_${Date.now()}_${Math.random().toString(36).slice(2, 10)}`,\n\t);\n\tmkdirSync(extractDir, { recursive: true });\n\n\ttry {\n\t\tif (assetName.endsWith(\".tar.gz\")) {\n\t\t\tconst extractResult = spawnSync(\"tar\", [\"xzf\", archivePath, \"-C\", extractDir], { stdio: \"pipe\" });\n\t\t\tif (extractResult.error || extractResult.status !== 0) {\n\t\t\t\tconst errMsg = extractResult.error?.message ?? extractResult.stderr?.toString().trim() ?? \"unknown error\";\n\t\t\t\tthrow new Error(`Failed to extract ${assetName}: ${errMsg}`);\n\t\t\t}\n\t\t} else if (assetName.endsWith(\".zip\")) {\n\t\t\tawait extractZip(archivePath, { dir: extractDir });\n\t\t} else {\n\t\t\tthrow new Error(`Unsupported archive format: ${assetName}`);\n\t\t}\n\n\t\t// Find the binary in extracted files. Some archives contain files directly\n\t\t// at root, others nest under a versioned subdirectory.\n\t\tconst binaryFileName = config.binaryName + binaryExt;\n\t\tconst extractedDir = join(extractDir, assetName.replace(/\\.(tar\\.gz|zip)$/, \"\"));\n\t\tconst extractedBinaryCandidates = [join(extractedDir, binaryFileName), join(extractDir, binaryFileName)];\n\t\tlet extractedBinary = extractedBinaryCandidates.find((candidate) => existsSync(candidate));\n\n\t\tif (!extractedBinary) {\n\t\t\textractedBinary = findBinaryRecursively(extractDir, binaryFileName) ?? undefined;\n\t\t}\n\n\t\tif (extractedBinary) {\n\t\t\trenameSync(extractedBinary, binaryPath);\n\t\t} else {\n\t\t\tthrow new Error(`Binary not found in archive: expected ${binaryFileName} under ${extractDir}`);\n\t\t}\n\n\t\t// Make executable (Unix only)\n\t\tif (plat !== \"win32\") {\n\t\t\tchmodSync(binaryPath, 0o755);\n\t\t}\n\t} finally {\n\t\t// Cleanup\n\t\trmSync(archivePath, { force: true });\n\t\trmSync(extractDir, { recursive: true, force: true });\n\t}\n\n\treturn binaryPath;\n}\n\n// Termux package names for tools\nconst TERMUX_PACKAGES: Record<string, string> = {\n\tfd: \"fd\",\n\trg: \"ripgrep\",\n};\n\n// Ensure a tool is available, downloading if necessary\n// Returns the path to the tool, or null if unavailable\nexport async function ensureTool(tool: \"fd\" | \"rg\", silent: boolean = false): Promise<string | undefined> {\n\tconst existingPath = getToolPath(tool);\n\tif (existingPath) {\n\t\treturn existingPath;\n\t}\n\n\tconst config = TOOLS[tool];\n\tif (!config) return undefined;\n\n\tif (isOfflineModeEnabled()) {\n\t\tif (!silent) {\n\t\t\tconsole.log(chalk.yellow(`${config.name} not found. Offline mode enabled, skipping download.`));\n\t\t}\n\t\treturn undefined;\n\t}\n\n\t// On Android/Termux, Linux binaries don't work due to Bionic libc incompatibility.\n\t// Users must install via pkg.\n\tif (platform() === \"android\") {\n\t\tconst pkgName = TERMUX_PACKAGES[tool] ?? tool;\n\t\tif (!silent) {\n\t\t\tconsole.log(chalk.yellow(`${config.name} not found. Install with: pkg install ${pkgName}`));\n\t\t}\n\t\treturn undefined;\n\t}\n\n\t// Tool not found - download it\n\tif (!silent) {\n\t\tconsole.log(chalk.dim(`${config.name} not found. Downloading...`));\n\t}\n\n\ttry {\n\t\tconst path = await downloadTool(tool);\n\t\tif (!silent) {\n\t\t\tconsole.log(chalk.dim(`${config.name} installed to ${path}`));\n\t\t}\n\t\treturn path;\n\t} catch (e) {\n\t\tif (!silent) {\n\t\t\tconsole.log(chalk.yellow(`Failed to download ${config.name}: ${e instanceof Error ? e.message : e}`));\n\t\t}\n\t\treturn undefined;\n\t}\n}\n"]}
@@ -0,0 +1,256 @@
1
+ import chalk from "chalk";
2
+ import { spawnSync } from "child_process";
3
+ import extractZip from "extract-zip";
4
+ import { chmodSync, createWriteStream, existsSync, mkdirSync, readdirSync, renameSync, rmSync } from "fs";
5
+ import { arch, platform } from "os";
6
+ import { join } from "path";
7
+ import { Readable } from "stream";
8
+ import { pipeline } from "stream/promises";
9
+ import { APP_NAME, getBinDir } from "../config.js";
10
+ const TOOLS_DIR = getBinDir();
11
+ const NETWORK_TIMEOUT_MS = 10_000;
12
+ const DOWNLOAD_TIMEOUT_MS = 120_000;
13
+ function isOfflineModeEnabled() {
14
+ const value = process.env.THREADWELL_OFFLINE;
15
+ if (!value)
16
+ return false;
17
+ return value === "1" || value.toLowerCase() === "true" || value.toLowerCase() === "yes";
18
+ }
19
+ const TOOLS = {
20
+ fd: {
21
+ name: "fd",
22
+ repo: "sharkdp/fd",
23
+ binaryName: "fd",
24
+ systemBinaryNames: ["fd", "fdfind"],
25
+ tagPrefix: "v",
26
+ getAssetName: (version, plat, architecture) => {
27
+ if (plat === "darwin") {
28
+ const archStr = architecture === "arm64" ? "aarch64" : "x86_64";
29
+ return `fd-v${version}-${archStr}-apple-darwin.tar.gz`;
30
+ }
31
+ else if (plat === "linux") {
32
+ const archStr = architecture === "arm64" ? "aarch64" : "x86_64";
33
+ return `fd-v${version}-${archStr}-unknown-linux-gnu.tar.gz`;
34
+ }
35
+ else if (plat === "win32") {
36
+ const archStr = architecture === "arm64" ? "aarch64" : "x86_64";
37
+ return `fd-v${version}-${archStr}-pc-windows-msvc.zip`;
38
+ }
39
+ return null;
40
+ },
41
+ },
42
+ rg: {
43
+ name: "ripgrep",
44
+ repo: "BurntSushi/ripgrep",
45
+ binaryName: "rg",
46
+ tagPrefix: "",
47
+ getAssetName: (version, plat, architecture) => {
48
+ if (plat === "darwin") {
49
+ const archStr = architecture === "arm64" ? "aarch64" : "x86_64";
50
+ return `ripgrep-${version}-${archStr}-apple-darwin.tar.gz`;
51
+ }
52
+ else if (plat === "linux") {
53
+ if (architecture === "arm64") {
54
+ return `ripgrep-${version}-aarch64-unknown-linux-gnu.tar.gz`;
55
+ }
56
+ return `ripgrep-${version}-x86_64-unknown-linux-musl.tar.gz`;
57
+ }
58
+ else if (plat === "win32") {
59
+ const archStr = architecture === "arm64" ? "aarch64" : "x86_64";
60
+ return `ripgrep-${version}-${archStr}-pc-windows-msvc.zip`;
61
+ }
62
+ return null;
63
+ },
64
+ },
65
+ };
66
+ // Check if a command exists in PATH by trying to run it
67
+ function commandExists(cmd) {
68
+ try {
69
+ const result = spawnSync(cmd, ["--version"], { stdio: "pipe" });
70
+ // Check for ENOENT error (command not found)
71
+ return result.error === undefined || result.error === null;
72
+ }
73
+ catch {
74
+ return false;
75
+ }
76
+ }
77
+ // Get the path to a tool (system-wide or in our tools dir)
78
+ export function getToolPath(tool) {
79
+ const config = TOOLS[tool];
80
+ if (!config)
81
+ return null;
82
+ // Check our tools directory first
83
+ const localPath = join(TOOLS_DIR, config.binaryName + (platform() === "win32" ? ".exe" : ""));
84
+ if (existsSync(localPath)) {
85
+ return localPath;
86
+ }
87
+ // Check system PATH - if found, just return the command name (it's in PATH)
88
+ const systemBinaryNames = config.systemBinaryNames ?? [config.binaryName];
89
+ for (const systemBinaryName of systemBinaryNames) {
90
+ if (commandExists(systemBinaryName)) {
91
+ return systemBinaryName;
92
+ }
93
+ }
94
+ return null;
95
+ }
96
+ // Fetch latest release version from GitHub
97
+ async function getLatestVersion(repo) {
98
+ const response = await fetch(`https://api.github.com/repos/${repo}/releases/latest`, {
99
+ headers: { "User-Agent": `${APP_NAME}-coding-agent` },
100
+ signal: AbortSignal.timeout(NETWORK_TIMEOUT_MS),
101
+ });
102
+ if (!response.ok) {
103
+ throw new Error(`GitHub API error: ${response.status}`);
104
+ }
105
+ const data = (await response.json());
106
+ return data.tag_name.replace(/^v/, "");
107
+ }
108
+ // Download a file from URL
109
+ async function downloadFile(url, dest) {
110
+ const response = await fetch(url, {
111
+ signal: AbortSignal.timeout(DOWNLOAD_TIMEOUT_MS),
112
+ });
113
+ if (!response.ok) {
114
+ throw new Error(`Failed to download: ${response.status}`);
115
+ }
116
+ if (!response.body) {
117
+ throw new Error("No response body");
118
+ }
119
+ const fileStream = createWriteStream(dest);
120
+ await pipeline(Readable.fromWeb(response.body), fileStream);
121
+ }
122
+ function findBinaryRecursively(rootDir, binaryFileName) {
123
+ const stack = [rootDir];
124
+ while (stack.length > 0) {
125
+ const currentDir = stack.pop();
126
+ if (!currentDir)
127
+ continue;
128
+ const entries = readdirSync(currentDir, { withFileTypes: true });
129
+ for (const entry of entries) {
130
+ const fullPath = join(currentDir, entry.name);
131
+ if (entry.isFile() && entry.name === binaryFileName) {
132
+ return fullPath;
133
+ }
134
+ if (entry.isDirectory()) {
135
+ stack.push(fullPath);
136
+ }
137
+ }
138
+ }
139
+ return null;
140
+ }
141
+ // Download and install a tool
142
+ async function downloadTool(tool) {
143
+ const config = TOOLS[tool];
144
+ if (!config)
145
+ throw new Error(`Unknown tool: ${tool}`);
146
+ const plat = platform();
147
+ const architecture = arch();
148
+ // Get latest version
149
+ const version = await getLatestVersion(config.repo);
150
+ // Get asset name for this platform
151
+ const assetName = config.getAssetName(version, plat, architecture);
152
+ if (!assetName) {
153
+ throw new Error(`Unsupported platform: ${plat}/${architecture}`);
154
+ }
155
+ // Create tools directory
156
+ mkdirSync(TOOLS_DIR, { recursive: true });
157
+ const downloadUrl = `https://github.com/${config.repo}/releases/download/${config.tagPrefix}${version}/${assetName}`;
158
+ const archivePath = join(TOOLS_DIR, assetName);
159
+ const binaryExt = plat === "win32" ? ".exe" : "";
160
+ const binaryPath = join(TOOLS_DIR, config.binaryName + binaryExt);
161
+ // Download
162
+ await downloadFile(downloadUrl, archivePath);
163
+ // Extract into a unique temp directory. fd and rg downloads can run concurrently
164
+ // during startup, so sharing a fixed directory causes races.
165
+ const extractDir = join(TOOLS_DIR, `extract_tmp_${config.binaryName}_${process.pid}_${Date.now()}_${Math.random().toString(36).slice(2, 10)}`);
166
+ mkdirSync(extractDir, { recursive: true });
167
+ try {
168
+ if (assetName.endsWith(".tar.gz")) {
169
+ const extractResult = spawnSync("tar", ["xzf", archivePath, "-C", extractDir], { stdio: "pipe" });
170
+ if (extractResult.error || extractResult.status !== 0) {
171
+ const errMsg = extractResult.error?.message ?? extractResult.stderr?.toString().trim() ?? "unknown error";
172
+ throw new Error(`Failed to extract ${assetName}: ${errMsg}`);
173
+ }
174
+ }
175
+ else if (assetName.endsWith(".zip")) {
176
+ await extractZip(archivePath, { dir: extractDir });
177
+ }
178
+ else {
179
+ throw new Error(`Unsupported archive format: ${assetName}`);
180
+ }
181
+ // Find the binary in extracted files. Some archives contain files directly
182
+ // at root, others nest under a versioned subdirectory.
183
+ const binaryFileName = config.binaryName + binaryExt;
184
+ const extractedDir = join(extractDir, assetName.replace(/\.(tar\.gz|zip)$/, ""));
185
+ const extractedBinaryCandidates = [join(extractedDir, binaryFileName), join(extractDir, binaryFileName)];
186
+ let extractedBinary = extractedBinaryCandidates.find((candidate) => existsSync(candidate));
187
+ if (!extractedBinary) {
188
+ extractedBinary = findBinaryRecursively(extractDir, binaryFileName) ?? undefined;
189
+ }
190
+ if (extractedBinary) {
191
+ renameSync(extractedBinary, binaryPath);
192
+ }
193
+ else {
194
+ throw new Error(`Binary not found in archive: expected ${binaryFileName} under ${extractDir}`);
195
+ }
196
+ // Make executable (Unix only)
197
+ if (plat !== "win32") {
198
+ chmodSync(binaryPath, 0o755);
199
+ }
200
+ }
201
+ finally {
202
+ // Cleanup
203
+ rmSync(archivePath, { force: true });
204
+ rmSync(extractDir, { recursive: true, force: true });
205
+ }
206
+ return binaryPath;
207
+ }
208
+ // Termux package names for tools
209
+ const TERMUX_PACKAGES = {
210
+ fd: "fd",
211
+ rg: "ripgrep",
212
+ };
213
+ // Ensure a tool is available, downloading if necessary
214
+ // Returns the path to the tool, or null if unavailable
215
+ export async function ensureTool(tool, silent = false) {
216
+ const existingPath = getToolPath(tool);
217
+ if (existingPath) {
218
+ return existingPath;
219
+ }
220
+ const config = TOOLS[tool];
221
+ if (!config)
222
+ return undefined;
223
+ if (isOfflineModeEnabled()) {
224
+ if (!silent) {
225
+ console.log(chalk.yellow(`${config.name} not found. Offline mode enabled, skipping download.`));
226
+ }
227
+ return undefined;
228
+ }
229
+ // On Android/Termux, Linux binaries don't work due to Bionic libc incompatibility.
230
+ // Users must install via pkg.
231
+ if (platform() === "android") {
232
+ const pkgName = TERMUX_PACKAGES[tool] ?? tool;
233
+ if (!silent) {
234
+ console.log(chalk.yellow(`${config.name} not found. Install with: pkg install ${pkgName}`));
235
+ }
236
+ return undefined;
237
+ }
238
+ // Tool not found - download it
239
+ if (!silent) {
240
+ console.log(chalk.dim(`${config.name} not found. Downloading...`));
241
+ }
242
+ try {
243
+ const path = await downloadTool(tool);
244
+ if (!silent) {
245
+ console.log(chalk.dim(`${config.name} installed to ${path}`));
246
+ }
247
+ return path;
248
+ }
249
+ catch (e) {
250
+ if (!silent) {
251
+ console.log(chalk.yellow(`Failed to download ${config.name}: ${e instanceof Error ? e.message : e}`));
252
+ }
253
+ return undefined;
254
+ }
255
+ }
256
+ //# sourceMappingURL=tools-manager.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tools-manager.js","sourceRoot":"","sources":["../../src/utils/tools-manager.ts"],"names":[],"mappings":"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","sourcesContent":["import chalk from \"chalk\";\nimport { spawnSync } from \"child_process\";\nimport extractZip from \"extract-zip\";\nimport { chmodSync, createWriteStream, existsSync, mkdirSync, readdirSync, renameSync, rmSync } from \"fs\";\nimport { arch, platform } from \"os\";\nimport { join } from \"path\";\nimport { Readable } from \"stream\";\nimport { pipeline } from \"stream/promises\";\nimport { APP_NAME, getBinDir } from \"../config.js\";\n\nconst TOOLS_DIR = getBinDir();\nconst NETWORK_TIMEOUT_MS = 10_000;\nconst DOWNLOAD_TIMEOUT_MS = 120_000;\n\nfunction isOfflineModeEnabled(): boolean {\n\tconst value = process.env.THREADWELL_OFFLINE;\n\tif (!value) return false;\n\treturn value === \"1\" || value.toLowerCase() === \"true\" || value.toLowerCase() === \"yes\";\n}\n\ninterface ToolConfig {\n\tname: string;\n\trepo: string; // GitHub repo (e.g., \"sharkdp/fd\")\n\tbinaryName: string; // Name of the binary inside the archive\n\tsystemBinaryNames?: string[]; // Alternative system command names to try before downloading\n\ttagPrefix: string; // Prefix for tags (e.g., \"v\" for v1.0.0, \"\" for 1.0.0)\n\tgetAssetName: (version: string, plat: string, architecture: string) => string | null;\n}\n\nconst TOOLS: Record<string, ToolConfig> = {\n\tfd: {\n\t\tname: \"fd\",\n\t\trepo: \"sharkdp/fd\",\n\t\tbinaryName: \"fd\",\n\t\tsystemBinaryNames: [\"fd\", \"fdfind\"],\n\t\ttagPrefix: \"v\",\n\t\tgetAssetName: (version, plat, architecture) => {\n\t\t\tif (plat === \"darwin\") {\n\t\t\t\tconst archStr = architecture === \"arm64\" ? \"aarch64\" : \"x86_64\";\n\t\t\t\treturn `fd-v${version}-${archStr}-apple-darwin.tar.gz`;\n\t\t\t} else if (plat === \"linux\") {\n\t\t\t\tconst archStr = architecture === \"arm64\" ? \"aarch64\" : \"x86_64\";\n\t\t\t\treturn `fd-v${version}-${archStr}-unknown-linux-gnu.tar.gz`;\n\t\t\t} else if (plat === \"win32\") {\n\t\t\t\tconst archStr = architecture === \"arm64\" ? \"aarch64\" : \"x86_64\";\n\t\t\t\treturn `fd-v${version}-${archStr}-pc-windows-msvc.zip`;\n\t\t\t}\n\t\t\treturn null;\n\t\t},\n\t},\n\trg: {\n\t\tname: \"ripgrep\",\n\t\trepo: \"BurntSushi/ripgrep\",\n\t\tbinaryName: \"rg\",\n\t\ttagPrefix: \"\",\n\t\tgetAssetName: (version, plat, architecture) => {\n\t\t\tif (plat === \"darwin\") {\n\t\t\t\tconst archStr = architecture === \"arm64\" ? \"aarch64\" : \"x86_64\";\n\t\t\t\treturn `ripgrep-${version}-${archStr}-apple-darwin.tar.gz`;\n\t\t\t} else if (plat === \"linux\") {\n\t\t\t\tif (architecture === \"arm64\") {\n\t\t\t\t\treturn `ripgrep-${version}-aarch64-unknown-linux-gnu.tar.gz`;\n\t\t\t\t}\n\t\t\t\treturn `ripgrep-${version}-x86_64-unknown-linux-musl.tar.gz`;\n\t\t\t} else if (plat === \"win32\") {\n\t\t\t\tconst archStr = architecture === \"arm64\" ? \"aarch64\" : \"x86_64\";\n\t\t\t\treturn `ripgrep-${version}-${archStr}-pc-windows-msvc.zip`;\n\t\t\t}\n\t\t\treturn null;\n\t\t},\n\t},\n};\n\n// Check if a command exists in PATH by trying to run it\nfunction commandExists(cmd: string): boolean {\n\ttry {\n\t\tconst result = spawnSync(cmd, [\"--version\"], { stdio: \"pipe\" });\n\t\t// Check for ENOENT error (command not found)\n\t\treturn result.error === undefined || result.error === null;\n\t} catch {\n\t\treturn false;\n\t}\n}\n\n// Get the path to a tool (system-wide or in our tools dir)\nexport function getToolPath(tool: \"fd\" | \"rg\"): string | null {\n\tconst config = TOOLS[tool];\n\tif (!config) return null;\n\n\t// Check our tools directory first\n\tconst localPath = join(TOOLS_DIR, config.binaryName + (platform() === \"win32\" ? \".exe\" : \"\"));\n\tif (existsSync(localPath)) {\n\t\treturn localPath;\n\t}\n\n\t// Check system PATH - if found, just return the command name (it's in PATH)\n\tconst systemBinaryNames = config.systemBinaryNames ?? [config.binaryName];\n\tfor (const systemBinaryName of systemBinaryNames) {\n\t\tif (commandExists(systemBinaryName)) {\n\t\t\treturn systemBinaryName;\n\t\t}\n\t}\n\n\treturn null;\n}\n\n// Fetch latest release version from GitHub\nasync function getLatestVersion(repo: string): Promise<string> {\n\tconst response = await fetch(`https://api.github.com/repos/${repo}/releases/latest`, {\n\t\theaders: { \"User-Agent\": `${APP_NAME}-coding-agent` },\n\t\tsignal: AbortSignal.timeout(NETWORK_TIMEOUT_MS),\n\t});\n\n\tif (!response.ok) {\n\t\tthrow new Error(`GitHub API error: ${response.status}`);\n\t}\n\n\tconst data = (await response.json()) as { tag_name: string };\n\treturn data.tag_name.replace(/^v/, \"\");\n}\n\n// Download a file from URL\nasync function downloadFile(url: string, dest: string): Promise<void> {\n\tconst response = await fetch(url, {\n\t\tsignal: AbortSignal.timeout(DOWNLOAD_TIMEOUT_MS),\n\t});\n\n\tif (!response.ok) {\n\t\tthrow new Error(`Failed to download: ${response.status}`);\n\t}\n\n\tif (!response.body) {\n\t\tthrow new Error(\"No response body\");\n\t}\n\n\tconst fileStream = createWriteStream(dest);\n\tawait pipeline(Readable.fromWeb(response.body as any), fileStream);\n}\n\nfunction findBinaryRecursively(rootDir: string, binaryFileName: string): string | null {\n\tconst stack: string[] = [rootDir];\n\n\twhile (stack.length > 0) {\n\t\tconst currentDir = stack.pop();\n\t\tif (!currentDir) continue;\n\n\t\tconst entries = readdirSync(currentDir, { withFileTypes: true });\n\t\tfor (const entry of entries) {\n\t\t\tconst fullPath = join(currentDir, entry.name);\n\t\t\tif (entry.isFile() && entry.name === binaryFileName) {\n\t\t\t\treturn fullPath;\n\t\t\t}\n\t\t\tif (entry.isDirectory()) {\n\t\t\t\tstack.push(fullPath);\n\t\t\t}\n\t\t}\n\t}\n\n\treturn null;\n}\n\n// Download and install a tool\nasync function downloadTool(tool: \"fd\" | \"rg\"): Promise<string> {\n\tconst config = TOOLS[tool];\n\tif (!config) throw new Error(`Unknown tool: ${tool}`);\n\n\tconst plat = platform();\n\tconst architecture = arch();\n\n\t// Get latest version\n\tconst version = await getLatestVersion(config.repo);\n\n\t// Get asset name for this platform\n\tconst assetName = config.getAssetName(version, plat, architecture);\n\tif (!assetName) {\n\t\tthrow new Error(`Unsupported platform: ${plat}/${architecture}`);\n\t}\n\n\t// Create tools directory\n\tmkdirSync(TOOLS_DIR, { recursive: true });\n\n\tconst downloadUrl = `https://github.com/${config.repo}/releases/download/${config.tagPrefix}${version}/${assetName}`;\n\tconst archivePath = join(TOOLS_DIR, assetName);\n\tconst binaryExt = plat === \"win32\" ? \".exe\" : \"\";\n\tconst binaryPath = join(TOOLS_DIR, config.binaryName + binaryExt);\n\n\t// Download\n\tawait downloadFile(downloadUrl, archivePath);\n\n\t// Extract into a unique temp directory. fd and rg downloads can run concurrently\n\t// during startup, so sharing a fixed directory causes races.\n\tconst extractDir = join(\n\t\tTOOLS_DIR,\n\t\t`extract_tmp_${config.binaryName}_${process.pid}_${Date.now()}_${Math.random().toString(36).slice(2, 10)}`,\n\t);\n\tmkdirSync(extractDir, { recursive: true });\n\n\ttry {\n\t\tif (assetName.endsWith(\".tar.gz\")) {\n\t\t\tconst extractResult = spawnSync(\"tar\", [\"xzf\", archivePath, \"-C\", extractDir], { stdio: \"pipe\" });\n\t\t\tif (extractResult.error || extractResult.status !== 0) {\n\t\t\t\tconst errMsg = extractResult.error?.message ?? extractResult.stderr?.toString().trim() ?? \"unknown error\";\n\t\t\t\tthrow new Error(`Failed to extract ${assetName}: ${errMsg}`);\n\t\t\t}\n\t\t} else if (assetName.endsWith(\".zip\")) {\n\t\t\tawait extractZip(archivePath, { dir: extractDir });\n\t\t} else {\n\t\t\tthrow new Error(`Unsupported archive format: ${assetName}`);\n\t\t}\n\n\t\t// Find the binary in extracted files. Some archives contain files directly\n\t\t// at root, others nest under a versioned subdirectory.\n\t\tconst binaryFileName = config.binaryName + binaryExt;\n\t\tconst extractedDir = join(extractDir, assetName.replace(/\\.(tar\\.gz|zip)$/, \"\"));\n\t\tconst extractedBinaryCandidates = [join(extractedDir, binaryFileName), join(extractDir, binaryFileName)];\n\t\tlet extractedBinary = extractedBinaryCandidates.find((candidate) => existsSync(candidate));\n\n\t\tif (!extractedBinary) {\n\t\t\textractedBinary = findBinaryRecursively(extractDir, binaryFileName) ?? undefined;\n\t\t}\n\n\t\tif (extractedBinary) {\n\t\t\trenameSync(extractedBinary, binaryPath);\n\t\t} else {\n\t\t\tthrow new Error(`Binary not found in archive: expected ${binaryFileName} under ${extractDir}`);\n\t\t}\n\n\t\t// Make executable (Unix only)\n\t\tif (plat !== \"win32\") {\n\t\t\tchmodSync(binaryPath, 0o755);\n\t\t}\n\t} finally {\n\t\t// Cleanup\n\t\trmSync(archivePath, { force: true });\n\t\trmSync(extractDir, { recursive: true, force: true });\n\t}\n\n\treturn binaryPath;\n}\n\n// Termux package names for tools\nconst TERMUX_PACKAGES: Record<string, string> = {\n\tfd: \"fd\",\n\trg: \"ripgrep\",\n};\n\n// Ensure a tool is available, downloading if necessary\n// Returns the path to the tool, or null if unavailable\nexport async function ensureTool(tool: \"fd\" | \"rg\", silent: boolean = false): Promise<string | undefined> {\n\tconst existingPath = getToolPath(tool);\n\tif (existingPath) {\n\t\treturn existingPath;\n\t}\n\n\tconst config = TOOLS[tool];\n\tif (!config) return undefined;\n\n\tif (isOfflineModeEnabled()) {\n\t\tif (!silent) {\n\t\t\tconsole.log(chalk.yellow(`${config.name} not found. Offline mode enabled, skipping download.`));\n\t\t}\n\t\treturn undefined;\n\t}\n\n\t// On Android/Termux, Linux binaries don't work due to Bionic libc incompatibility.\n\t// Users must install via pkg.\n\tif (platform() === \"android\") {\n\t\tconst pkgName = TERMUX_PACKAGES[tool] ?? tool;\n\t\tif (!silent) {\n\t\t\tconsole.log(chalk.yellow(`${config.name} not found. Install with: pkg install ${pkgName}`));\n\t\t}\n\t\treturn undefined;\n\t}\n\n\t// Tool not found - download it\n\tif (!silent) {\n\t\tconsole.log(chalk.dim(`${config.name} not found. Downloading...`));\n\t}\n\n\ttry {\n\t\tconst path = await downloadTool(tool);\n\t\tif (!silent) {\n\t\t\tconsole.log(chalk.dim(`${config.name} installed to ${path}`));\n\t\t}\n\t\treturn path;\n\t} catch (e) {\n\t\tif (!silent) {\n\t\t\tconsole.log(chalk.yellow(`Failed to download ${config.name}: ${e instanceof Error ? e.message : e}`));\n\t\t}\n\t\treturn undefined;\n\t}\n}\n"]}
@@ -0,0 +1,14 @@
1
+ export interface LatestThreadwellRelease {
2
+ version: string;
3
+ packageName?: string;
4
+ }
5
+ export declare function comparePackageVersions(leftVersion: string, rightVersion: string): number | undefined;
6
+ export declare function isNewerPackageVersion(candidateVersion: string, currentVersion: string): boolean;
7
+ export declare function getLatestThreadwellRelease(currentVersion: string, options?: {
8
+ timeoutMs?: number;
9
+ }): Promise<LatestThreadwellRelease | undefined>;
10
+ export declare function getLatestThreadwellVersion(currentVersion: string, options?: {
11
+ timeoutMs?: number;
12
+ }): Promise<string | undefined>;
13
+ export declare function checkForNewThreadwellVersion(currentVersion: string): Promise<string | undefined>;
14
+ //# sourceMappingURL=version-check.d.ts.map