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":"git.js","sourceRoot":"","sources":["../../src/utils/git.ts"],"names":[],"mappings":"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","sourcesContent":["import hostedGitInfo from \"hosted-git-info\";\n\n/**\n * Parsed git URL information.\n */\nexport type GitSource = {\n\t/** Always \"git\" for git sources */\n\ttype: \"git\";\n\t/** Clone URL (always valid for git clone, without ref suffix) */\n\trepo: string;\n\t/** Git host domain (e.g., \"github.com\") */\n\thost: string;\n\t/** Repository path (e.g., \"user/repo\") */\n\tpath: string;\n\t/** Git ref (branch, tag, commit) if specified */\n\tref?: string;\n\t/** True if ref was specified (package won't be auto-updated) */\n\tpinned: boolean;\n};\n\nfunction splitRef(url: string): { repo: string; ref?: string } {\n\tconst scpLikeMatch = url.match(/^git@([^:]+):(.+)$/);\n\tif (scpLikeMatch) {\n\t\tconst pathWithMaybeRef = scpLikeMatch[2] ?? \"\";\n\t\tconst refSeparator = pathWithMaybeRef.indexOf(\"@\");\n\t\tif (refSeparator < 0) return { repo: url };\n\t\tconst repoPath = pathWithMaybeRef.slice(0, refSeparator);\n\t\tconst ref = pathWithMaybeRef.slice(refSeparator + 1);\n\t\tif (!repoPath || !ref) return { repo: url };\n\t\treturn {\n\t\t\trepo: `git@${scpLikeMatch[1] ?? \"\"}:${repoPath}`,\n\t\t\tref,\n\t\t};\n\t}\n\n\tif (url.includes(\"://\")) {\n\t\ttry {\n\t\t\tconst parsed = new URL(url);\n\t\t\tconst pathWithMaybeRef = parsed.pathname.replace(/^\\/+/, \"\");\n\t\t\tconst refSeparator = pathWithMaybeRef.indexOf(\"@\");\n\t\t\tif (refSeparator < 0) return { repo: url };\n\t\t\tconst repoPath = pathWithMaybeRef.slice(0, refSeparator);\n\t\t\tconst ref = pathWithMaybeRef.slice(refSeparator + 1);\n\t\t\tif (!repoPath || !ref) return { repo: url };\n\t\t\tparsed.pathname = `/${repoPath}`;\n\t\t\treturn {\n\t\t\t\trepo: parsed.toString().replace(/\\/$/, \"\"),\n\t\t\t\tref,\n\t\t\t};\n\t\t} catch {\n\t\t\treturn { repo: url };\n\t\t}\n\t}\n\n\tconst slashIndex = url.indexOf(\"/\");\n\tif (slashIndex < 0) {\n\t\treturn { repo: url };\n\t}\n\tconst host = url.slice(0, slashIndex);\n\tconst pathWithMaybeRef = url.slice(slashIndex + 1);\n\tconst refSeparator = pathWithMaybeRef.indexOf(\"@\");\n\tif (refSeparator < 0) {\n\t\treturn { repo: url };\n\t}\n\tconst repoPath = pathWithMaybeRef.slice(0, refSeparator);\n\tconst ref = pathWithMaybeRef.slice(refSeparator + 1);\n\tif (!repoPath || !ref) {\n\t\treturn { repo: url };\n\t}\n\treturn {\n\t\trepo: `${host}/${repoPath}`,\n\t\tref,\n\t};\n}\n\nfunction parseGenericGitUrl(url: string): GitSource | null {\n\tconst { repo: repoWithoutRef, ref } = splitRef(url);\n\tlet repo = repoWithoutRef;\n\tlet host = \"\";\n\tlet path = \"\";\n\n\tconst scpLikeMatch = repoWithoutRef.match(/^git@([^:]+):(.+)$/);\n\tif (scpLikeMatch) {\n\t\thost = scpLikeMatch[1] ?? \"\";\n\t\tpath = scpLikeMatch[2] ?? \"\";\n\t} else if (\n\t\trepoWithoutRef.startsWith(\"https://\") ||\n\t\trepoWithoutRef.startsWith(\"http://\") ||\n\t\trepoWithoutRef.startsWith(\"ssh://\") ||\n\t\trepoWithoutRef.startsWith(\"git://\")\n\t) {\n\t\ttry {\n\t\t\tconst parsed = new URL(repoWithoutRef);\n\t\t\thost = parsed.hostname;\n\t\t\tpath = parsed.pathname.replace(/^\\/+/, \"\");\n\t\t} catch {\n\t\t\treturn null;\n\t\t}\n\t} else {\n\t\tconst slashIndex = repoWithoutRef.indexOf(\"/\");\n\t\tif (slashIndex < 0) {\n\t\t\treturn null;\n\t\t}\n\t\thost = repoWithoutRef.slice(0, slashIndex);\n\t\tpath = repoWithoutRef.slice(slashIndex + 1);\n\t\tif (!host.includes(\".\") && host !== \"localhost\") {\n\t\t\treturn null;\n\t\t}\n\t\trepo = `https://${repoWithoutRef}`;\n\t}\n\n\tconst normalizedPath = path.replace(/\\.git$/, \"\").replace(/^\\/+/, \"\");\n\tif (!host || !normalizedPath || normalizedPath.split(\"/\").length < 2) {\n\t\treturn null;\n\t}\n\n\treturn {\n\t\ttype: \"git\",\n\t\trepo,\n\t\thost,\n\t\tpath: normalizedPath,\n\t\tref,\n\t\tpinned: Boolean(ref),\n\t};\n}\n\n/**\n * Parse git source into a GitSource.\n *\n * Rules:\n * - With git: prefix, accept all historical shorthand forms.\n * - Without git: prefix, only accept explicit protocol URLs.\n */\nexport function parseGitUrl(source: string): GitSource | null {\n\tconst trimmed = source.trim();\n\tconst hasGitPrefix = trimmed.startsWith(\"git:\");\n\tconst url = hasGitPrefix ? trimmed.slice(4).trim() : trimmed;\n\n\tif (!hasGitPrefix && !/^(https?|ssh|git):\\/\\//i.test(url)) {\n\t\treturn null;\n\t}\n\n\tconst split = splitRef(url);\n\n\tconst hostedCandidates = [split.ref ? `${split.repo}#${split.ref}` : undefined, url].filter(\n\t\t(value): value is string => Boolean(value),\n\t);\n\tfor (const candidate of hostedCandidates) {\n\t\tconst info = hostedGitInfo.fromUrl(candidate);\n\t\tif (info) {\n\t\t\tif (split.ref && info.project?.includes(\"@\")) {\n\t\t\t\tcontinue;\n\t\t\t}\n\t\t\tconst useHttpsPrefix =\n\t\t\t\t!split.repo.startsWith(\"http://\") &&\n\t\t\t\t!split.repo.startsWith(\"https://\") &&\n\t\t\t\t!split.repo.startsWith(\"ssh://\") &&\n\t\t\t\t!split.repo.startsWith(\"git://\") &&\n\t\t\t\t!split.repo.startsWith(\"git@\");\n\t\t\treturn {\n\t\t\t\ttype: \"git\",\n\t\t\t\trepo: useHttpsPrefix ? `https://${split.repo}` : split.repo,\n\t\t\t\thost: info.domain || \"\",\n\t\t\t\tpath: `${info.user}/${info.project}`.replace(/\\.git$/, \"\"),\n\t\t\t\tref: info.committish || split.ref || undefined,\n\t\t\t\tpinned: Boolean(info.committish || split.ref),\n\t\t\t};\n\t\t}\n\t}\n\n\tconst httpsCandidates = [split.ref ? `https://${split.repo}#${split.ref}` : undefined, `https://${url}`].filter(\n\t\t(value): value is string => Boolean(value),\n\t);\n\tfor (const candidate of httpsCandidates) {\n\t\tconst info = hostedGitInfo.fromUrl(candidate);\n\t\tif (info) {\n\t\t\tif (split.ref && info.project?.includes(\"@\")) {\n\t\t\t\tcontinue;\n\t\t\t}\n\t\t\treturn {\n\t\t\t\ttype: \"git\",\n\t\t\t\trepo: `https://${split.repo}`,\n\t\t\t\thost: info.domain || \"\",\n\t\t\t\tpath: `${info.user}/${info.project}`.replace(/\\.git$/, \"\"),\n\t\t\t\tref: info.committish || split.ref || undefined,\n\t\t\t\tpinned: Boolean(info.committish || split.ref),\n\t\t\t};\n\t\t}\n\t}\n\n\treturn parseGenericGitUrl(url);\n}\n"]}
@@ -0,0 +1,9 @@
1
+ /**
2
+ * Convert image to PNG format for terminal display.
3
+ * Kitty graphics protocol requires PNG format (f=100).
4
+ */
5
+ export declare function convertToPng(base64Data: string, mimeType: string): Promise<{
6
+ data: string;
7
+ mimeType: string;
8
+ } | null>;
9
+ //# sourceMappingURL=image-convert.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"image-convert.d.ts","sourceRoot":"","sources":["../../src/utils/image-convert.ts"],"names":[],"mappings":"AAGA;;;GAGG;AACH,wBAAsB,YAAY,CACjC,UAAU,EAAE,MAAM,EAClB,QAAQ,EAAE,MAAM,GACd,OAAO,CAAC;IAAE,IAAI,EAAE,MAAM,CAAC;IAAC,QAAQ,EAAE,MAAM,CAAA;CAAE,GAAG,IAAI,CAAC,CA8BpD","sourcesContent":["import { applyExifOrientation } from \"./exif-orientation.js\";\nimport { loadPhoton } from \"./photon.js\";\n\n/**\n * Convert image to PNG format for terminal display.\n * Kitty graphics protocol requires PNG format (f=100).\n */\nexport async function convertToPng(\n\tbase64Data: string,\n\tmimeType: string,\n): Promise<{ data: string; mimeType: string } | null> {\n\t// Already PNG, no conversion needed\n\tif (mimeType === \"image/png\") {\n\t\treturn { data: base64Data, mimeType };\n\t}\n\n\tconst photon = await loadPhoton();\n\tif (!photon) {\n\t\t// Photon not available, can't convert\n\t\treturn null;\n\t}\n\n\ttry {\n\t\tconst bytes = new Uint8Array(Buffer.from(base64Data, \"base64\"));\n\t\tconst rawImage = photon.PhotonImage.new_from_byteslice(bytes);\n\t\tconst image = applyExifOrientation(photon, rawImage, bytes);\n\t\tif (image !== rawImage) rawImage.free();\n\t\ttry {\n\t\t\tconst pngBuffer = image.get_bytes();\n\t\t\treturn {\n\t\t\t\tdata: Buffer.from(pngBuffer).toString(\"base64\"),\n\t\t\t\tmimeType: \"image/png\",\n\t\t\t};\n\t\t} finally {\n\t\t\timage.free();\n\t\t}\n\t} catch {\n\t\t// Conversion failed\n\t\treturn null;\n\t}\n}\n"]}
@@ -0,0 +1,39 @@
1
+ import { applyExifOrientation } from "./exif-orientation.js";
2
+ import { loadPhoton } from "./photon.js";
3
+ /**
4
+ * Convert image to PNG format for terminal display.
5
+ * Kitty graphics protocol requires PNG format (f=100).
6
+ */
7
+ export async function convertToPng(base64Data, mimeType) {
8
+ // Already PNG, no conversion needed
9
+ if (mimeType === "image/png") {
10
+ return { data: base64Data, mimeType };
11
+ }
12
+ const photon = await loadPhoton();
13
+ if (!photon) {
14
+ // Photon not available, can't convert
15
+ return null;
16
+ }
17
+ try {
18
+ const bytes = new Uint8Array(Buffer.from(base64Data, "base64"));
19
+ const rawImage = photon.PhotonImage.new_from_byteslice(bytes);
20
+ const image = applyExifOrientation(photon, rawImage, bytes);
21
+ if (image !== rawImage)
22
+ rawImage.free();
23
+ try {
24
+ const pngBuffer = image.get_bytes();
25
+ return {
26
+ data: Buffer.from(pngBuffer).toString("base64"),
27
+ mimeType: "image/png",
28
+ };
29
+ }
30
+ finally {
31
+ image.free();
32
+ }
33
+ }
34
+ catch {
35
+ // Conversion failed
36
+ return null;
37
+ }
38
+ }
39
+ //# sourceMappingURL=image-convert.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"image-convert.js","sourceRoot":"","sources":["../../src/utils/image-convert.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,oBAAoB,EAAE,MAAM,uBAAuB,CAAC;AAC7D,OAAO,EAAE,UAAU,EAAE,MAAM,aAAa,CAAC;AAEzC;;;GAGG;AACH,MAAM,CAAC,KAAK,UAAU,YAAY,CACjC,UAAkB,EAClB,QAAgB,EACqC;IACrD,oCAAoC;IACpC,IAAI,QAAQ,KAAK,WAAW,EAAE,CAAC;QAC9B,OAAO,EAAE,IAAI,EAAE,UAAU,EAAE,QAAQ,EAAE,CAAC;IACvC,CAAC;IAED,MAAM,MAAM,GAAG,MAAM,UAAU,EAAE,CAAC;IAClC,IAAI,CAAC,MAAM,EAAE,CAAC;QACb,sCAAsC;QACtC,OAAO,IAAI,CAAC;IACb,CAAC;IAED,IAAI,CAAC;QACJ,MAAM,KAAK,GAAG,IAAI,UAAU,CAAC,MAAM,CAAC,IAAI,CAAC,UAAU,EAAE,QAAQ,CAAC,CAAC,CAAC;QAChE,MAAM,QAAQ,GAAG,MAAM,CAAC,WAAW,CAAC,kBAAkB,CAAC,KAAK,CAAC,CAAC;QAC9D,MAAM,KAAK,GAAG,oBAAoB,CAAC,MAAM,EAAE,QAAQ,EAAE,KAAK,CAAC,CAAC;QAC5D,IAAI,KAAK,KAAK,QAAQ;YAAE,QAAQ,CAAC,IAAI,EAAE,CAAC;QACxC,IAAI,CAAC;YACJ,MAAM,SAAS,GAAG,KAAK,CAAC,SAAS,EAAE,CAAC;YACpC,OAAO;gBACN,IAAI,EAAE,MAAM,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,QAAQ,CAAC,QAAQ,CAAC;gBAC/C,QAAQ,EAAE,WAAW;aACrB,CAAC;QACH,CAAC;gBAAS,CAAC;YACV,KAAK,CAAC,IAAI,EAAE,CAAC;QACd,CAAC;IACF,CAAC;IAAC,MAAM,CAAC;QACR,oBAAoB;QACpB,OAAO,IAAI,CAAC;IACb,CAAC;AAAA,CACD","sourcesContent":["import { applyExifOrientation } from \"./exif-orientation.js\";\nimport { loadPhoton } from \"./photon.js\";\n\n/**\n * Convert image to PNG format for terminal display.\n * Kitty graphics protocol requires PNG format (f=100).\n */\nexport async function convertToPng(\n\tbase64Data: string,\n\tmimeType: string,\n): Promise<{ data: string; mimeType: string } | null> {\n\t// Already PNG, no conversion needed\n\tif (mimeType === \"image/png\") {\n\t\treturn { data: base64Data, mimeType };\n\t}\n\n\tconst photon = await loadPhoton();\n\tif (!photon) {\n\t\t// Photon not available, can't convert\n\t\treturn null;\n\t}\n\n\ttry {\n\t\tconst bytes = new Uint8Array(Buffer.from(base64Data, \"base64\"));\n\t\tconst rawImage = photon.PhotonImage.new_from_byteslice(bytes);\n\t\tconst image = applyExifOrientation(photon, rawImage, bytes);\n\t\tif (image !== rawImage) rawImage.free();\n\t\ttry {\n\t\t\tconst pngBuffer = image.get_bytes();\n\t\t\treturn {\n\t\t\t\tdata: Buffer.from(pngBuffer).toString(\"base64\"),\n\t\t\t\tmimeType: \"image/png\",\n\t\t\t};\n\t\t} finally {\n\t\t\timage.free();\n\t\t}\n\t} catch {\n\t\t// Conversion failed\n\t\treturn null;\n\t}\n}\n"]}
@@ -0,0 +1,36 @@
1
+ import type { ImageContent } from "@threadwell/ai";
2
+ export interface ImageResizeOptions {
3
+ maxWidth?: number;
4
+ maxHeight?: number;
5
+ maxBytes?: number;
6
+ jpegQuality?: number;
7
+ }
8
+ export interface ResizedImage {
9
+ data: string;
10
+ mimeType: string;
11
+ originalWidth: number;
12
+ originalHeight: number;
13
+ width: number;
14
+ height: number;
15
+ wasResized: boolean;
16
+ }
17
+ /**
18
+ * Resize an image to fit within the specified max dimensions and encoded file size.
19
+ * Returns null if the image cannot be resized below maxBytes.
20
+ *
21
+ * Uses Photon (Rust/WASM) for image processing. If Photon is not available,
22
+ * returns null.
23
+ *
24
+ * Strategy for staying under maxBytes:
25
+ * 1. First resize to maxWidth/maxHeight
26
+ * 2. Try both PNG and JPEG formats, pick the smaller one
27
+ * 3. If still too large, try JPEG with decreasing quality
28
+ * 4. If still too large, progressively reduce dimensions until 1x1
29
+ */
30
+ export declare function resizeImage(img: ImageContent, options?: ImageResizeOptions): Promise<ResizedImage | null>;
31
+ /**
32
+ * Format a dimension note for resized images.
33
+ * This helps the model understand the coordinate mapping.
34
+ */
35
+ export declare function formatDimensionNote(result: ResizedImage): string | undefined;
36
+ //# sourceMappingURL=image-resize.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"image-resize.d.ts","sourceRoot":"","sources":["../../src/utils/image-resize.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,gBAAgB,CAAC;AAInD,MAAM,WAAW,kBAAkB;IAClC,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,WAAW,CAAC,EAAE,MAAM,CAAC;CACrB;AAED,MAAM,WAAW,YAAY;IAC5B,IAAI,EAAE,MAAM,CAAC;IACb,QAAQ,EAAE,MAAM,CAAC;IACjB,aAAa,EAAE,MAAM,CAAC;IACtB,cAAc,EAAE,MAAM,CAAC;IACvB,KAAK,EAAE,MAAM,CAAC;IACd,MAAM,EAAE,MAAM,CAAC;IACf,UAAU,EAAE,OAAO,CAAC;CACpB;AA2BD;;;;;;;;;;;;GAYG;AACH,wBAAsB,WAAW,CAAC,GAAG,EAAE,YAAY,EAAE,OAAO,CAAC,EAAE,kBAAkB,GAAG,OAAO,CAAC,YAAY,GAAG,IAAI,CAAC,CAuG/G;AAED;;;GAGG;AACH,wBAAgB,mBAAmB,CAAC,MAAM,EAAE,YAAY,GAAG,MAAM,GAAG,SAAS,CAO5E","sourcesContent":["import type { ImageContent } from \"@threadwell/ai\";\nimport { applyExifOrientation } from \"./exif-orientation.js\";\nimport { loadPhoton } from \"./photon.js\";\n\nexport interface ImageResizeOptions {\n\tmaxWidth?: number; // Default: 2000\n\tmaxHeight?: number; // Default: 2000\n\tmaxBytes?: number; // Default: 4.5MB of base64 payload (below Anthropic's 5MB limit)\n\tjpegQuality?: number; // Default: 80\n}\n\nexport interface ResizedImage {\n\tdata: string; // base64\n\tmimeType: string;\n\toriginalWidth: number;\n\toriginalHeight: number;\n\twidth: number;\n\theight: number;\n\twasResized: boolean;\n}\n\n// 4.5MB of base64 payload. Provides headroom below Anthropic's 5MB limit.\nconst DEFAULT_MAX_BYTES = 4.5 * 1024 * 1024;\n\nconst DEFAULT_OPTIONS: Required<ImageResizeOptions> = {\n\tmaxWidth: 2000,\n\tmaxHeight: 2000,\n\tmaxBytes: DEFAULT_MAX_BYTES,\n\tjpegQuality: 80,\n};\n\ninterface EncodedCandidate {\n\tdata: string;\n\tencodedSize: number;\n\tmimeType: string;\n}\n\nfunction encodeCandidate(buffer: Uint8Array, mimeType: string): EncodedCandidate {\n\tconst data = Buffer.from(buffer).toString(\"base64\");\n\treturn {\n\t\tdata,\n\t\tencodedSize: Buffer.byteLength(data, \"utf-8\"),\n\t\tmimeType,\n\t};\n}\n\n/**\n * Resize an image to fit within the specified max dimensions and encoded file size.\n * Returns null if the image cannot be resized below maxBytes.\n *\n * Uses Photon (Rust/WASM) for image processing. If Photon is not available,\n * returns null.\n *\n * Strategy for staying under maxBytes:\n * 1. First resize to maxWidth/maxHeight\n * 2. Try both PNG and JPEG formats, pick the smaller one\n * 3. If still too large, try JPEG with decreasing quality\n * 4. If still too large, progressively reduce dimensions until 1x1\n */\nexport async function resizeImage(img: ImageContent, options?: ImageResizeOptions): Promise<ResizedImage | null> {\n\tconst opts = { ...DEFAULT_OPTIONS, ...options };\n\tconst inputBuffer = Buffer.from(img.data, \"base64\");\n\tconst inputBase64Size = Buffer.byteLength(img.data, \"utf-8\");\n\n\tconst photon = await loadPhoton();\n\tif (!photon) {\n\t\treturn null;\n\t}\n\n\tlet image: ReturnType<typeof photon.PhotonImage.new_from_byteslice> | undefined;\n\ttry {\n\t\tconst inputBytes = new Uint8Array(inputBuffer);\n\t\tconst rawImage = photon.PhotonImage.new_from_byteslice(inputBytes);\n\t\timage = applyExifOrientation(photon, rawImage, inputBytes);\n\t\tif (image !== rawImage) rawImage.free();\n\n\t\tconst originalWidth = image.get_width();\n\t\tconst originalHeight = image.get_height();\n\t\tconst format = img.mimeType?.split(\"/\")[1] ?? \"png\";\n\n\t\t// Check if already within all limits (dimensions AND encoded size)\n\t\tif (originalWidth <= opts.maxWidth && originalHeight <= opts.maxHeight && inputBase64Size < opts.maxBytes) {\n\t\t\treturn {\n\t\t\t\tdata: img.data,\n\t\t\t\tmimeType: img.mimeType ?? `image/${format}`,\n\t\t\t\toriginalWidth,\n\t\t\t\toriginalHeight,\n\t\t\t\twidth: originalWidth,\n\t\t\t\theight: originalHeight,\n\t\t\t\twasResized: false,\n\t\t\t};\n\t\t}\n\n\t\t// Calculate initial dimensions respecting max limits\n\t\tlet targetWidth = originalWidth;\n\t\tlet targetHeight = originalHeight;\n\n\t\tif (targetWidth > opts.maxWidth) {\n\t\t\ttargetHeight = Math.round((targetHeight * opts.maxWidth) / targetWidth);\n\t\t\ttargetWidth = opts.maxWidth;\n\t\t}\n\t\tif (targetHeight > opts.maxHeight) {\n\t\t\ttargetWidth = Math.round((targetWidth * opts.maxHeight) / targetHeight);\n\t\t\ttargetHeight = opts.maxHeight;\n\t\t}\n\n\t\tfunction tryEncodings(width: number, height: number, jpegQualities: number[]): EncodedCandidate[] {\n\t\t\tconst resized = photon!.resize(image!, width, height, photon!.SamplingFilter.Lanczos3);\n\n\t\t\ttry {\n\t\t\t\tconst candidates: EncodedCandidate[] = [encodeCandidate(resized.get_bytes(), \"image/png\")];\n\t\t\t\tfor (const quality of jpegQualities) {\n\t\t\t\t\tcandidates.push(encodeCandidate(resized.get_bytes_jpeg(quality), \"image/jpeg\"));\n\t\t\t\t}\n\t\t\t\treturn candidates;\n\t\t\t} finally {\n\t\t\t\tresized.free();\n\t\t\t}\n\t\t}\n\n\t\tconst qualitySteps = Array.from(new Set([opts.jpegQuality, 85, 70, 55, 40]));\n\t\tlet currentWidth = targetWidth;\n\t\tlet currentHeight = targetHeight;\n\n\t\twhile (true) {\n\t\t\tconst candidates = tryEncodings(currentWidth, currentHeight, qualitySteps);\n\t\t\tfor (const candidate of candidates) {\n\t\t\t\tif (candidate.encodedSize < opts.maxBytes) {\n\t\t\t\t\treturn {\n\t\t\t\t\t\tdata: candidate.data,\n\t\t\t\t\t\tmimeType: candidate.mimeType,\n\t\t\t\t\t\toriginalWidth,\n\t\t\t\t\t\toriginalHeight,\n\t\t\t\t\t\twidth: currentWidth,\n\t\t\t\t\t\theight: currentHeight,\n\t\t\t\t\t\twasResized: true,\n\t\t\t\t\t};\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tif (currentWidth === 1 && currentHeight === 1) {\n\t\t\t\tbreak;\n\t\t\t}\n\n\t\t\tconst nextWidth = currentWidth === 1 ? 1 : Math.max(1, Math.floor(currentWidth * 0.75));\n\t\t\tconst nextHeight = currentHeight === 1 ? 1 : Math.max(1, Math.floor(currentHeight * 0.75));\n\t\t\tif (nextWidth === currentWidth && nextHeight === currentHeight) {\n\t\t\t\tbreak;\n\t\t\t}\n\n\t\t\tcurrentWidth = nextWidth;\n\t\t\tcurrentHeight = nextHeight;\n\t\t}\n\n\t\treturn null;\n\t} catch {\n\t\treturn null;\n\t} finally {\n\t\tif (image) {\n\t\t\timage.free();\n\t\t}\n\t}\n}\n\n/**\n * Format a dimension note for resized images.\n * This helps the model understand the coordinate mapping.\n */\nexport function formatDimensionNote(result: ResizedImage): string | undefined {\n\tif (!result.wasResized) {\n\t\treturn undefined;\n\t}\n\n\tconst scale = result.originalWidth / result.width;\n\treturn `[Image: original ${result.originalWidth}x${result.originalHeight}, displayed at ${result.width}x${result.height}. Multiply coordinates by ${scale.toFixed(2)} to map to original image.]`;\n}\n"]}
@@ -0,0 +1,137 @@
1
+ import { applyExifOrientation } from "./exif-orientation.js";
2
+ import { loadPhoton } from "./photon.js";
3
+ // 4.5MB of base64 payload. Provides headroom below Anthropic's 5MB limit.
4
+ const DEFAULT_MAX_BYTES = 4.5 * 1024 * 1024;
5
+ const DEFAULT_OPTIONS = {
6
+ maxWidth: 2000,
7
+ maxHeight: 2000,
8
+ maxBytes: DEFAULT_MAX_BYTES,
9
+ jpegQuality: 80,
10
+ };
11
+ function encodeCandidate(buffer, mimeType) {
12
+ const data = Buffer.from(buffer).toString("base64");
13
+ return {
14
+ data,
15
+ encodedSize: Buffer.byteLength(data, "utf-8"),
16
+ mimeType,
17
+ };
18
+ }
19
+ /**
20
+ * Resize an image to fit within the specified max dimensions and encoded file size.
21
+ * Returns null if the image cannot be resized below maxBytes.
22
+ *
23
+ * Uses Photon (Rust/WASM) for image processing. If Photon is not available,
24
+ * returns null.
25
+ *
26
+ * Strategy for staying under maxBytes:
27
+ * 1. First resize to maxWidth/maxHeight
28
+ * 2. Try both PNG and JPEG formats, pick the smaller one
29
+ * 3. If still too large, try JPEG with decreasing quality
30
+ * 4. If still too large, progressively reduce dimensions until 1x1
31
+ */
32
+ export async function resizeImage(img, options) {
33
+ const opts = { ...DEFAULT_OPTIONS, ...options };
34
+ const inputBuffer = Buffer.from(img.data, "base64");
35
+ const inputBase64Size = Buffer.byteLength(img.data, "utf-8");
36
+ const photon = await loadPhoton();
37
+ if (!photon) {
38
+ return null;
39
+ }
40
+ let image;
41
+ try {
42
+ const inputBytes = new Uint8Array(inputBuffer);
43
+ const rawImage = photon.PhotonImage.new_from_byteslice(inputBytes);
44
+ image = applyExifOrientation(photon, rawImage, inputBytes);
45
+ if (image !== rawImage)
46
+ rawImage.free();
47
+ const originalWidth = image.get_width();
48
+ const originalHeight = image.get_height();
49
+ const format = img.mimeType?.split("/")[1] ?? "png";
50
+ // Check if already within all limits (dimensions AND encoded size)
51
+ if (originalWidth <= opts.maxWidth && originalHeight <= opts.maxHeight && inputBase64Size < opts.maxBytes) {
52
+ return {
53
+ data: img.data,
54
+ mimeType: img.mimeType ?? `image/${format}`,
55
+ originalWidth,
56
+ originalHeight,
57
+ width: originalWidth,
58
+ height: originalHeight,
59
+ wasResized: false,
60
+ };
61
+ }
62
+ // Calculate initial dimensions respecting max limits
63
+ let targetWidth = originalWidth;
64
+ let targetHeight = originalHeight;
65
+ if (targetWidth > opts.maxWidth) {
66
+ targetHeight = Math.round((targetHeight * opts.maxWidth) / targetWidth);
67
+ targetWidth = opts.maxWidth;
68
+ }
69
+ if (targetHeight > opts.maxHeight) {
70
+ targetWidth = Math.round((targetWidth * opts.maxHeight) / targetHeight);
71
+ targetHeight = opts.maxHeight;
72
+ }
73
+ function tryEncodings(width, height, jpegQualities) {
74
+ const resized = photon.resize(image, width, height, photon.SamplingFilter.Lanczos3);
75
+ try {
76
+ const candidates = [encodeCandidate(resized.get_bytes(), "image/png")];
77
+ for (const quality of jpegQualities) {
78
+ candidates.push(encodeCandidate(resized.get_bytes_jpeg(quality), "image/jpeg"));
79
+ }
80
+ return candidates;
81
+ }
82
+ finally {
83
+ resized.free();
84
+ }
85
+ }
86
+ const qualitySteps = Array.from(new Set([opts.jpegQuality, 85, 70, 55, 40]));
87
+ let currentWidth = targetWidth;
88
+ let currentHeight = targetHeight;
89
+ while (true) {
90
+ const candidates = tryEncodings(currentWidth, currentHeight, qualitySteps);
91
+ for (const candidate of candidates) {
92
+ if (candidate.encodedSize < opts.maxBytes) {
93
+ return {
94
+ data: candidate.data,
95
+ mimeType: candidate.mimeType,
96
+ originalWidth,
97
+ originalHeight,
98
+ width: currentWidth,
99
+ height: currentHeight,
100
+ wasResized: true,
101
+ };
102
+ }
103
+ }
104
+ if (currentWidth === 1 && currentHeight === 1) {
105
+ break;
106
+ }
107
+ const nextWidth = currentWidth === 1 ? 1 : Math.max(1, Math.floor(currentWidth * 0.75));
108
+ const nextHeight = currentHeight === 1 ? 1 : Math.max(1, Math.floor(currentHeight * 0.75));
109
+ if (nextWidth === currentWidth && nextHeight === currentHeight) {
110
+ break;
111
+ }
112
+ currentWidth = nextWidth;
113
+ currentHeight = nextHeight;
114
+ }
115
+ return null;
116
+ }
117
+ catch {
118
+ return null;
119
+ }
120
+ finally {
121
+ if (image) {
122
+ image.free();
123
+ }
124
+ }
125
+ }
126
+ /**
127
+ * Format a dimension note for resized images.
128
+ * This helps the model understand the coordinate mapping.
129
+ */
130
+ export function formatDimensionNote(result) {
131
+ if (!result.wasResized) {
132
+ return undefined;
133
+ }
134
+ const scale = result.originalWidth / result.width;
135
+ return `[Image: original ${result.originalWidth}x${result.originalHeight}, displayed at ${result.width}x${result.height}. Multiply coordinates by ${scale.toFixed(2)} to map to original image.]`;
136
+ }
137
+ //# sourceMappingURL=image-resize.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"image-resize.js","sourceRoot":"","sources":["../../src/utils/image-resize.ts"],"names":[],"mappings":"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","sourcesContent":["import type { ImageContent } from \"@threadwell/ai\";\nimport { applyExifOrientation } from \"./exif-orientation.js\";\nimport { loadPhoton } from \"./photon.js\";\n\nexport interface ImageResizeOptions {\n\tmaxWidth?: number; // Default: 2000\n\tmaxHeight?: number; // Default: 2000\n\tmaxBytes?: number; // Default: 4.5MB of base64 payload (below Anthropic's 5MB limit)\n\tjpegQuality?: number; // Default: 80\n}\n\nexport interface ResizedImage {\n\tdata: string; // base64\n\tmimeType: string;\n\toriginalWidth: number;\n\toriginalHeight: number;\n\twidth: number;\n\theight: number;\n\twasResized: boolean;\n}\n\n// 4.5MB of base64 payload. Provides headroom below Anthropic's 5MB limit.\nconst DEFAULT_MAX_BYTES = 4.5 * 1024 * 1024;\n\nconst DEFAULT_OPTIONS: Required<ImageResizeOptions> = {\n\tmaxWidth: 2000,\n\tmaxHeight: 2000,\n\tmaxBytes: DEFAULT_MAX_BYTES,\n\tjpegQuality: 80,\n};\n\ninterface EncodedCandidate {\n\tdata: string;\n\tencodedSize: number;\n\tmimeType: string;\n}\n\nfunction encodeCandidate(buffer: Uint8Array, mimeType: string): EncodedCandidate {\n\tconst data = Buffer.from(buffer).toString(\"base64\");\n\treturn {\n\t\tdata,\n\t\tencodedSize: Buffer.byteLength(data, \"utf-8\"),\n\t\tmimeType,\n\t};\n}\n\n/**\n * Resize an image to fit within the specified max dimensions and encoded file size.\n * Returns null if the image cannot be resized below maxBytes.\n *\n * Uses Photon (Rust/WASM) for image processing. If Photon is not available,\n * returns null.\n *\n * Strategy for staying under maxBytes:\n * 1. First resize to maxWidth/maxHeight\n * 2. Try both PNG and JPEG formats, pick the smaller one\n * 3. If still too large, try JPEG with decreasing quality\n * 4. If still too large, progressively reduce dimensions until 1x1\n */\nexport async function resizeImage(img: ImageContent, options?: ImageResizeOptions): Promise<ResizedImage | null> {\n\tconst opts = { ...DEFAULT_OPTIONS, ...options };\n\tconst inputBuffer = Buffer.from(img.data, \"base64\");\n\tconst inputBase64Size = Buffer.byteLength(img.data, \"utf-8\");\n\n\tconst photon = await loadPhoton();\n\tif (!photon) {\n\t\treturn null;\n\t}\n\n\tlet image: ReturnType<typeof photon.PhotonImage.new_from_byteslice> | undefined;\n\ttry {\n\t\tconst inputBytes = new Uint8Array(inputBuffer);\n\t\tconst rawImage = photon.PhotonImage.new_from_byteslice(inputBytes);\n\t\timage = applyExifOrientation(photon, rawImage, inputBytes);\n\t\tif (image !== rawImage) rawImage.free();\n\n\t\tconst originalWidth = image.get_width();\n\t\tconst originalHeight = image.get_height();\n\t\tconst format = img.mimeType?.split(\"/\")[1] ?? \"png\";\n\n\t\t// Check if already within all limits (dimensions AND encoded size)\n\t\tif (originalWidth <= opts.maxWidth && originalHeight <= opts.maxHeight && inputBase64Size < opts.maxBytes) {\n\t\t\treturn {\n\t\t\t\tdata: img.data,\n\t\t\t\tmimeType: img.mimeType ?? `image/${format}`,\n\t\t\t\toriginalWidth,\n\t\t\t\toriginalHeight,\n\t\t\t\twidth: originalWidth,\n\t\t\t\theight: originalHeight,\n\t\t\t\twasResized: false,\n\t\t\t};\n\t\t}\n\n\t\t// Calculate initial dimensions respecting max limits\n\t\tlet targetWidth = originalWidth;\n\t\tlet targetHeight = originalHeight;\n\n\t\tif (targetWidth > opts.maxWidth) {\n\t\t\ttargetHeight = Math.round((targetHeight * opts.maxWidth) / targetWidth);\n\t\t\ttargetWidth = opts.maxWidth;\n\t\t}\n\t\tif (targetHeight > opts.maxHeight) {\n\t\t\ttargetWidth = Math.round((targetWidth * opts.maxHeight) / targetHeight);\n\t\t\ttargetHeight = opts.maxHeight;\n\t\t}\n\n\t\tfunction tryEncodings(width: number, height: number, jpegQualities: number[]): EncodedCandidate[] {\n\t\t\tconst resized = photon!.resize(image!, width, height, photon!.SamplingFilter.Lanczos3);\n\n\t\t\ttry {\n\t\t\t\tconst candidates: EncodedCandidate[] = [encodeCandidate(resized.get_bytes(), \"image/png\")];\n\t\t\t\tfor (const quality of jpegQualities) {\n\t\t\t\t\tcandidates.push(encodeCandidate(resized.get_bytes_jpeg(quality), \"image/jpeg\"));\n\t\t\t\t}\n\t\t\t\treturn candidates;\n\t\t\t} finally {\n\t\t\t\tresized.free();\n\t\t\t}\n\t\t}\n\n\t\tconst qualitySteps = Array.from(new Set([opts.jpegQuality, 85, 70, 55, 40]));\n\t\tlet currentWidth = targetWidth;\n\t\tlet currentHeight = targetHeight;\n\n\t\twhile (true) {\n\t\t\tconst candidates = tryEncodings(currentWidth, currentHeight, qualitySteps);\n\t\t\tfor (const candidate of candidates) {\n\t\t\t\tif (candidate.encodedSize < opts.maxBytes) {\n\t\t\t\t\treturn {\n\t\t\t\t\t\tdata: candidate.data,\n\t\t\t\t\t\tmimeType: candidate.mimeType,\n\t\t\t\t\t\toriginalWidth,\n\t\t\t\t\t\toriginalHeight,\n\t\t\t\t\t\twidth: currentWidth,\n\t\t\t\t\t\theight: currentHeight,\n\t\t\t\t\t\twasResized: true,\n\t\t\t\t\t};\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tif (currentWidth === 1 && currentHeight === 1) {\n\t\t\t\tbreak;\n\t\t\t}\n\n\t\t\tconst nextWidth = currentWidth === 1 ? 1 : Math.max(1, Math.floor(currentWidth * 0.75));\n\t\t\tconst nextHeight = currentHeight === 1 ? 1 : Math.max(1, Math.floor(currentHeight * 0.75));\n\t\t\tif (nextWidth === currentWidth && nextHeight === currentHeight) {\n\t\t\t\tbreak;\n\t\t\t}\n\n\t\t\tcurrentWidth = nextWidth;\n\t\t\tcurrentHeight = nextHeight;\n\t\t}\n\n\t\treturn null;\n\t} catch {\n\t\treturn null;\n\t} finally {\n\t\tif (image) {\n\t\t\timage.free();\n\t\t}\n\t}\n}\n\n/**\n * Format a dimension note for resized images.\n * This helps the model understand the coordinate mapping.\n */\nexport function formatDimensionNote(result: ResizedImage): string | undefined {\n\tif (!result.wasResized) {\n\t\treturn undefined;\n\t}\n\n\tconst scale = result.originalWidth / result.width;\n\treturn `[Image: original ${result.originalWidth}x${result.originalHeight}, displayed at ${result.width}x${result.height}. Multiply coordinates by ${scale.toFixed(2)} to map to original image.]`;\n}\n"]}
@@ -0,0 +1,2 @@
1
+ export declare function detectSupportedImageMimeTypeFromFile(filePath: string): Promise<string | null>;
2
+ //# sourceMappingURL=mime.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"mime.d.ts","sourceRoot":"","sources":["../../src/utils/mime.ts"],"names":[],"mappings":"AAOA,wBAAsB,oCAAoC,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,GAAG,IAAI,CAAC,CAsBnG","sourcesContent":["import { open } from \"node:fs/promises\";\nimport { fileTypeFromBuffer } from \"file-type\";\n\nconst IMAGE_MIME_TYPES = new Set([\"image/jpeg\", \"image/png\", \"image/gif\", \"image/webp\"]);\n\nconst FILE_TYPE_SNIFF_BYTES = 4100;\n\nexport async function detectSupportedImageMimeTypeFromFile(filePath: string): Promise<string | null> {\n\tconst fileHandle = await open(filePath, \"r\");\n\ttry {\n\t\tconst buffer = Buffer.alloc(FILE_TYPE_SNIFF_BYTES);\n\t\tconst { bytesRead } = await fileHandle.read(buffer, 0, FILE_TYPE_SNIFF_BYTES, 0);\n\t\tif (bytesRead === 0) {\n\t\t\treturn null;\n\t\t}\n\n\t\tconst fileType = await fileTypeFromBuffer(buffer.subarray(0, bytesRead));\n\t\tif (!fileType) {\n\t\t\treturn null;\n\t\t}\n\n\t\tif (!IMAGE_MIME_TYPES.has(fileType.mime)) {\n\t\t\treturn null;\n\t\t}\n\n\t\treturn fileType.mime;\n\t} finally {\n\t\tawait fileHandle.close();\n\t}\n}\n"]}
@@ -0,0 +1,26 @@
1
+ import { open } from "node:fs/promises";
2
+ import { fileTypeFromBuffer } from "file-type";
3
+ const IMAGE_MIME_TYPES = new Set(["image/jpeg", "image/png", "image/gif", "image/webp"]);
4
+ const FILE_TYPE_SNIFF_BYTES = 4100;
5
+ export async function detectSupportedImageMimeTypeFromFile(filePath) {
6
+ const fileHandle = await open(filePath, "r");
7
+ try {
8
+ const buffer = Buffer.alloc(FILE_TYPE_SNIFF_BYTES);
9
+ const { bytesRead } = await fileHandle.read(buffer, 0, FILE_TYPE_SNIFF_BYTES, 0);
10
+ if (bytesRead === 0) {
11
+ return null;
12
+ }
13
+ const fileType = await fileTypeFromBuffer(buffer.subarray(0, bytesRead));
14
+ if (!fileType) {
15
+ return null;
16
+ }
17
+ if (!IMAGE_MIME_TYPES.has(fileType.mime)) {
18
+ return null;
19
+ }
20
+ return fileType.mime;
21
+ }
22
+ finally {
23
+ await fileHandle.close();
24
+ }
25
+ }
26
+ //# sourceMappingURL=mime.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"mime.js","sourceRoot":"","sources":["../../src/utils/mime.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAE,MAAM,kBAAkB,CAAC;AACxC,OAAO,EAAE,kBAAkB,EAAE,MAAM,WAAW,CAAC;AAE/C,MAAM,gBAAgB,GAAG,IAAI,GAAG,CAAC,CAAC,YAAY,EAAE,WAAW,EAAE,WAAW,EAAE,YAAY,CAAC,CAAC,CAAC;AAEzF,MAAM,qBAAqB,GAAG,IAAI,CAAC;AAEnC,MAAM,CAAC,KAAK,UAAU,oCAAoC,CAAC,QAAgB,EAA0B;IACpG,MAAM,UAAU,GAAG,MAAM,IAAI,CAAC,QAAQ,EAAE,GAAG,CAAC,CAAC;IAC7C,IAAI,CAAC;QACJ,MAAM,MAAM,GAAG,MAAM,CAAC,KAAK,CAAC,qBAAqB,CAAC,CAAC;QACnD,MAAM,EAAE,SAAS,EAAE,GAAG,MAAM,UAAU,CAAC,IAAI,CAAC,MAAM,EAAE,CAAC,EAAE,qBAAqB,EAAE,CAAC,CAAC,CAAC;QACjF,IAAI,SAAS,KAAK,CAAC,EAAE,CAAC;YACrB,OAAO,IAAI,CAAC;QACb,CAAC;QAED,MAAM,QAAQ,GAAG,MAAM,kBAAkB,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,EAAE,SAAS,CAAC,CAAC,CAAC;QACzE,IAAI,CAAC,QAAQ,EAAE,CAAC;YACf,OAAO,IAAI,CAAC;QACb,CAAC;QAED,IAAI,CAAC,gBAAgB,CAAC,GAAG,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC;YAC1C,OAAO,IAAI,CAAC;QACb,CAAC;QAED,OAAO,QAAQ,CAAC,IAAI,CAAC;IACtB,CAAC;YAAS,CAAC;QACV,MAAM,UAAU,CAAC,KAAK,EAAE,CAAC;IAC1B,CAAC;AAAA,CACD","sourcesContent":["import { open } from \"node:fs/promises\";\nimport { fileTypeFromBuffer } from \"file-type\";\n\nconst IMAGE_MIME_TYPES = new Set([\"image/jpeg\", \"image/png\", \"image/gif\", \"image/webp\"]);\n\nconst FILE_TYPE_SNIFF_BYTES = 4100;\n\nexport async function detectSupportedImageMimeTypeFromFile(filePath: string): Promise<string | null> {\n\tconst fileHandle = await open(filePath, \"r\");\n\ttry {\n\t\tconst buffer = Buffer.alloc(FILE_TYPE_SNIFF_BYTES);\n\t\tconst { bytesRead } = await fileHandle.read(buffer, 0, FILE_TYPE_SNIFF_BYTES, 0);\n\t\tif (bytesRead === 0) {\n\t\t\treturn null;\n\t\t}\n\n\t\tconst fileType = await fileTypeFromBuffer(buffer.subarray(0, bytesRead));\n\t\tif (!fileType) {\n\t\t\treturn null;\n\t\t}\n\n\t\tif (!IMAGE_MIME_TYPES.has(fileType.mime)) {\n\t\t\treturn null;\n\t\t}\n\n\t\treturn fileType.mime;\n\t} finally {\n\t\tawait fileHandle.close();\n\t}\n}\n"]}
@@ -0,0 +1,14 @@
1
+ /**
2
+ * Resolve a path to its canonical (real) form, following symlinks.
3
+ * Falls back to the raw path if resolution fails (e.g. the target does
4
+ * not exist yet), so that callers never crash on missing filesystem
5
+ * entries.
6
+ */
7
+ export declare function canonicalizePath(path: string): string;
8
+ /**
9
+ * Returns true if the value is NOT a package source (npm:, git:, etc.)
10
+ * or a URL protocol. Bare names and relative paths without ./ prefix
11
+ * are considered local.
12
+ */
13
+ export declare function isLocalPath(value: string): boolean;
14
+ //# sourceMappingURL=paths.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"paths.d.ts","sourceRoot":"","sources":["../../src/utils/paths.ts"],"names":[],"mappings":"AAEA;;;;;GAKG;AACH,wBAAgB,gBAAgB,CAAC,IAAI,EAAE,MAAM,GAAG,MAAM,CAMrD;AAED;;;;GAIG;AACH,wBAAgB,WAAW,CAAC,KAAK,EAAE,MAAM,GAAG,OAAO,CAclD","sourcesContent":["import { realpathSync } from \"node:fs\";\n\n/**\n * Resolve a path to its canonical (real) form, following symlinks.\n * Falls back to the raw path if resolution fails (e.g. the target does\n * not exist yet), so that callers never crash on missing filesystem\n * entries.\n */\nexport function canonicalizePath(path: string): string {\n\ttry {\n\t\treturn realpathSync(path);\n\t} catch {\n\t\treturn path;\n\t}\n}\n\n/**\n * Returns true if the value is NOT a package source (npm:, git:, etc.)\n * or a URL protocol. Bare names and relative paths without ./ prefix\n * are considered local.\n */\nexport function isLocalPath(value: string): boolean {\n\tconst trimmed = value.trim();\n\t// Known non-local prefixes\n\tif (\n\t\ttrimmed.startsWith(\"npm:\") ||\n\t\ttrimmed.startsWith(\"git:\") ||\n\t\ttrimmed.startsWith(\"github:\") ||\n\t\ttrimmed.startsWith(\"http:\") ||\n\t\ttrimmed.startsWith(\"https:\") ||\n\t\ttrimmed.startsWith(\"ssh:\")\n\t) {\n\t\treturn false;\n\t}\n\treturn true;\n}\n"]}
@@ -0,0 +1,34 @@
1
+ import { realpathSync } from "node:fs";
2
+ /**
3
+ * Resolve a path to its canonical (real) form, following symlinks.
4
+ * Falls back to the raw path if resolution fails (e.g. the target does
5
+ * not exist yet), so that callers never crash on missing filesystem
6
+ * entries.
7
+ */
8
+ export function canonicalizePath(path) {
9
+ try {
10
+ return realpathSync(path);
11
+ }
12
+ catch {
13
+ return path;
14
+ }
15
+ }
16
+ /**
17
+ * Returns true if the value is NOT a package source (npm:, git:, etc.)
18
+ * or a URL protocol. Bare names and relative paths without ./ prefix
19
+ * are considered local.
20
+ */
21
+ export function isLocalPath(value) {
22
+ const trimmed = value.trim();
23
+ // Known non-local prefixes
24
+ if (trimmed.startsWith("npm:") ||
25
+ trimmed.startsWith("git:") ||
26
+ trimmed.startsWith("github:") ||
27
+ trimmed.startsWith("http:") ||
28
+ trimmed.startsWith("https:") ||
29
+ trimmed.startsWith("ssh:")) {
30
+ return false;
31
+ }
32
+ return true;
33
+ }
34
+ //# sourceMappingURL=paths.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"paths.js","sourceRoot":"","sources":["../../src/utils/paths.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,SAAS,CAAC;AAEvC;;;;;GAKG;AACH,MAAM,UAAU,gBAAgB,CAAC,IAAY,EAAU;IACtD,IAAI,CAAC;QACJ,OAAO,YAAY,CAAC,IAAI,CAAC,CAAC;IAC3B,CAAC;IAAC,MAAM,CAAC;QACR,OAAO,IAAI,CAAC;IACb,CAAC;AAAA,CACD;AAED;;;;GAIG;AACH,MAAM,UAAU,WAAW,CAAC,KAAa,EAAW;IACnD,MAAM,OAAO,GAAG,KAAK,CAAC,IAAI,EAAE,CAAC;IAC7B,2BAA2B;IAC3B,IACC,OAAO,CAAC,UAAU,CAAC,MAAM,CAAC;QAC1B,OAAO,CAAC,UAAU,CAAC,MAAM,CAAC;QAC1B,OAAO,CAAC,UAAU,CAAC,SAAS,CAAC;QAC7B,OAAO,CAAC,UAAU,CAAC,OAAO,CAAC;QAC3B,OAAO,CAAC,UAAU,CAAC,QAAQ,CAAC;QAC5B,OAAO,CAAC,UAAU,CAAC,MAAM,CAAC,EACzB,CAAC;QACF,OAAO,KAAK,CAAC;IACd,CAAC;IACD,OAAO,IAAI,CAAC;AAAA,CACZ","sourcesContent":["import { realpathSync } from \"node:fs\";\n\n/**\n * Resolve a path to its canonical (real) form, following symlinks.\n * Falls back to the raw path if resolution fails (e.g. the target does\n * not exist yet), so that callers never crash on missing filesystem\n * entries.\n */\nexport function canonicalizePath(path: string): string {\n\ttry {\n\t\treturn realpathSync(path);\n\t} catch {\n\t\treturn path;\n\t}\n}\n\n/**\n * Returns true if the value is NOT a package source (npm:, git:, etc.)\n * or a URL protocol. Bare names and relative paths without ./ prefix\n * are considered local.\n */\nexport function isLocalPath(value: string): boolean {\n\tconst trimmed = value.trim();\n\t// Known non-local prefixes\n\tif (\n\t\ttrimmed.startsWith(\"npm:\") ||\n\t\ttrimmed.startsWith(\"git:\") ||\n\t\ttrimmed.startsWith(\"github:\") ||\n\t\ttrimmed.startsWith(\"http:\") ||\n\t\ttrimmed.startsWith(\"https:\") ||\n\t\ttrimmed.startsWith(\"ssh:\")\n\t) {\n\t\treturn false;\n\t}\n\treturn true;\n}\n"]}
@@ -0,0 +1,21 @@
1
+ /**
2
+ * Photon image processing wrapper.
3
+ *
4
+ * This module provides a unified interface to @silvia-odwyer/photon-node that works in:
5
+ * 1. Node.js (development, npm run build)
6
+ * 2. Bun compiled binaries (standalone distribution)
7
+ *
8
+ * The challenge: photon-node's CJS entry uses fs.readFileSync(__dirname + '/photon_rs_bg.wasm')
9
+ * which bakes the build machine's absolute path into Bun compiled binaries.
10
+ *
11
+ * Solution:
12
+ * 1. Patch fs.readFileSync to redirect missing photon_rs_bg.wasm reads
13
+ * 2. Copy photon_rs_bg.wasm next to the executable in build:binary
14
+ */
15
+ export type { PhotonImage as PhotonImageType } from "@silvia-odwyer/photon-node";
16
+ /**
17
+ * Load the photon module asynchronously.
18
+ * Returns cached module on subsequent calls.
19
+ */
20
+ export declare function loadPhoton(): Promise<typeof import("@silvia-odwyer/photon-node") | null>;
21
+ //# sourceMappingURL=photon.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"photon.d.ts","sourceRoot":"","sources":["../../src/utils/photon.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;GAaG;AAWH,YAAY,EAAE,WAAW,IAAI,eAAe,EAAE,MAAM,4BAA4B,CAAC;AAuFjF;;;GAGG;AACH,wBAAsB,UAAU,IAAI,OAAO,CAAC,cAAc,4BAA4B,CAAC,GAAG,IAAI,CAAC,CAuB9F","sourcesContent":["/**\n * Photon image processing wrapper.\n *\n * This module provides a unified interface to @silvia-odwyer/photon-node that works in:\n * 1. Node.js (development, npm run build)\n * 2. Bun compiled binaries (standalone distribution)\n *\n * The challenge: photon-node's CJS entry uses fs.readFileSync(__dirname + '/photon_rs_bg.wasm')\n * which bakes the build machine's absolute path into Bun compiled binaries.\n *\n * Solution:\n * 1. Patch fs.readFileSync to redirect missing photon_rs_bg.wasm reads\n * 2. Copy photon_rs_bg.wasm next to the executable in build:binary\n */\n\nimport type { PathOrFileDescriptor } from \"fs\";\nimport { createRequire } from \"module\";\nimport * as path from \"path\";\nimport { fileURLToPath } from \"url\";\n\nconst require = createRequire(import.meta.url);\nconst fs = require(\"fs\") as typeof import(\"fs\");\n\n// Re-export types from the main package\nexport type { PhotonImage as PhotonImageType } from \"@silvia-odwyer/photon-node\";\n\ntype ReadFileSync = typeof fs.readFileSync;\n\nconst WASM_FILENAME = \"photon_rs_bg.wasm\";\n\n// Lazy-loaded photon module\nlet photonModule: typeof import(\"@silvia-odwyer/photon-node\") | null = null;\nlet loadPromise: Promise<typeof import(\"@silvia-odwyer/photon-node\") | null> | null = null;\n\nfunction pathOrNull(file: PathOrFileDescriptor): string | null {\n\tif (typeof file === \"string\") {\n\t\treturn file;\n\t}\n\tif (file instanceof URL) {\n\t\treturn fileURLToPath(file);\n\t}\n\treturn null;\n}\n\nfunction getFallbackWasmPaths(): string[] {\n\tconst execDir = path.dirname(process.execPath);\n\treturn [\n\t\tpath.join(execDir, WASM_FILENAME),\n\t\tpath.join(execDir, \"photon\", WASM_FILENAME),\n\t\tpath.join(process.cwd(), WASM_FILENAME),\n\t];\n}\n\nfunction patchPhotonWasmRead(): () => void {\n\tconst originalReadFileSync: ReadFileSync = fs.readFileSync.bind(fs);\n\tconst fallbackPaths = getFallbackWasmPaths();\n\tconst mutableFs = fs as { readFileSync: ReadFileSync };\n\n\tconst patchedReadFileSync: ReadFileSync = ((...args: Parameters<ReadFileSync>) => {\n\t\tconst [file, options] = args;\n\t\tconst resolvedPath = pathOrNull(file);\n\n\t\tif (resolvedPath?.endsWith(WASM_FILENAME)) {\n\t\t\ttry {\n\t\t\t\treturn originalReadFileSync(...args);\n\t\t\t} catch (error) {\n\t\t\t\tconst err = error as NodeJS.ErrnoException;\n\t\t\t\tif (err?.code && err.code !== \"ENOENT\") {\n\t\t\t\t\tthrow error;\n\t\t\t\t}\n\n\t\t\t\tfor (const fallbackPath of fallbackPaths) {\n\t\t\t\t\tif (!fs.existsSync(fallbackPath)) {\n\t\t\t\t\t\tcontinue;\n\t\t\t\t\t}\n\t\t\t\t\tif (options === undefined) {\n\t\t\t\t\t\treturn originalReadFileSync(fallbackPath);\n\t\t\t\t\t}\n\t\t\t\t\treturn originalReadFileSync(fallbackPath, options);\n\t\t\t\t}\n\n\t\t\t\tthrow error;\n\t\t\t}\n\t\t}\n\n\t\treturn originalReadFileSync(...args);\n\t}) as ReadFileSync;\n\n\ttry {\n\t\tmutableFs.readFileSync = patchedReadFileSync;\n\t} catch {\n\t\tObject.defineProperty(fs, \"readFileSync\", {\n\t\t\tvalue: patchedReadFileSync,\n\t\t\twritable: true,\n\t\t\tconfigurable: true,\n\t\t});\n\t}\n\n\treturn () => {\n\t\ttry {\n\t\t\tmutableFs.readFileSync = originalReadFileSync;\n\t\t} catch {\n\t\t\tObject.defineProperty(fs, \"readFileSync\", {\n\t\t\t\tvalue: originalReadFileSync,\n\t\t\t\twritable: true,\n\t\t\t\tconfigurable: true,\n\t\t\t});\n\t\t}\n\t};\n}\n\n/**\n * Load the photon module asynchronously.\n * Returns cached module on subsequent calls.\n */\nexport async function loadPhoton(): Promise<typeof import(\"@silvia-odwyer/photon-node\") | null> {\n\tif (photonModule) {\n\t\treturn photonModule;\n\t}\n\n\tif (loadPromise) {\n\t\treturn loadPromise;\n\t}\n\n\tloadPromise = (async () => {\n\t\tconst restoreReadFileSync = patchPhotonWasmRead();\n\t\ttry {\n\t\t\tphotonModule = await import(\"@silvia-odwyer/photon-node\");\n\t\t\treturn photonModule;\n\t\t} catch {\n\t\t\tphotonModule = null;\n\t\t\treturn photonModule;\n\t\t} finally {\n\t\t\trestoreReadFileSync();\n\t\t}\n\t})();\n\n\treturn loadPromise;\n}\n"]}
@@ -0,0 +1,121 @@
1
+ /**
2
+ * Photon image processing wrapper.
3
+ *
4
+ * This module provides a unified interface to @silvia-odwyer/photon-node that works in:
5
+ * 1. Node.js (development, npm run build)
6
+ * 2. Bun compiled binaries (standalone distribution)
7
+ *
8
+ * The challenge: photon-node's CJS entry uses fs.readFileSync(__dirname + '/photon_rs_bg.wasm')
9
+ * which bakes the build machine's absolute path into Bun compiled binaries.
10
+ *
11
+ * Solution:
12
+ * 1. Patch fs.readFileSync to redirect missing photon_rs_bg.wasm reads
13
+ * 2. Copy photon_rs_bg.wasm next to the executable in build:binary
14
+ */
15
+ import { createRequire } from "module";
16
+ import * as path from "path";
17
+ import { fileURLToPath } from "url";
18
+ const require = createRequire(import.meta.url);
19
+ const fs = require("fs");
20
+ const WASM_FILENAME = "photon_rs_bg.wasm";
21
+ // Lazy-loaded photon module
22
+ let photonModule = null;
23
+ let loadPromise = null;
24
+ function pathOrNull(file) {
25
+ if (typeof file === "string") {
26
+ return file;
27
+ }
28
+ if (file instanceof URL) {
29
+ return fileURLToPath(file);
30
+ }
31
+ return null;
32
+ }
33
+ function getFallbackWasmPaths() {
34
+ const execDir = path.dirname(process.execPath);
35
+ return [
36
+ path.join(execDir, WASM_FILENAME),
37
+ path.join(execDir, "photon", WASM_FILENAME),
38
+ path.join(process.cwd(), WASM_FILENAME),
39
+ ];
40
+ }
41
+ function patchPhotonWasmRead() {
42
+ const originalReadFileSync = fs.readFileSync.bind(fs);
43
+ const fallbackPaths = getFallbackWasmPaths();
44
+ const mutableFs = fs;
45
+ const patchedReadFileSync = ((...args) => {
46
+ const [file, options] = args;
47
+ const resolvedPath = pathOrNull(file);
48
+ if (resolvedPath?.endsWith(WASM_FILENAME)) {
49
+ try {
50
+ return originalReadFileSync(...args);
51
+ }
52
+ catch (error) {
53
+ const err = error;
54
+ if (err?.code && err.code !== "ENOENT") {
55
+ throw error;
56
+ }
57
+ for (const fallbackPath of fallbackPaths) {
58
+ if (!fs.existsSync(fallbackPath)) {
59
+ continue;
60
+ }
61
+ if (options === undefined) {
62
+ return originalReadFileSync(fallbackPath);
63
+ }
64
+ return originalReadFileSync(fallbackPath, options);
65
+ }
66
+ throw error;
67
+ }
68
+ }
69
+ return originalReadFileSync(...args);
70
+ });
71
+ try {
72
+ mutableFs.readFileSync = patchedReadFileSync;
73
+ }
74
+ catch {
75
+ Object.defineProperty(fs, "readFileSync", {
76
+ value: patchedReadFileSync,
77
+ writable: true,
78
+ configurable: true,
79
+ });
80
+ }
81
+ return () => {
82
+ try {
83
+ mutableFs.readFileSync = originalReadFileSync;
84
+ }
85
+ catch {
86
+ Object.defineProperty(fs, "readFileSync", {
87
+ value: originalReadFileSync,
88
+ writable: true,
89
+ configurable: true,
90
+ });
91
+ }
92
+ };
93
+ }
94
+ /**
95
+ * Load the photon module asynchronously.
96
+ * Returns cached module on subsequent calls.
97
+ */
98
+ export async function loadPhoton() {
99
+ if (photonModule) {
100
+ return photonModule;
101
+ }
102
+ if (loadPromise) {
103
+ return loadPromise;
104
+ }
105
+ loadPromise = (async () => {
106
+ const restoreReadFileSync = patchPhotonWasmRead();
107
+ try {
108
+ photonModule = await import("@silvia-odwyer/photon-node");
109
+ return photonModule;
110
+ }
111
+ catch {
112
+ photonModule = null;
113
+ return photonModule;
114
+ }
115
+ finally {
116
+ restoreReadFileSync();
117
+ }
118
+ })();
119
+ return loadPromise;
120
+ }
121
+ //# sourceMappingURL=photon.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"photon.js","sourceRoot":"","sources":["../../src/utils/photon.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Photon image processing wrapper.\n *\n * This module provides a unified interface to @silvia-odwyer/photon-node that works in:\n * 1. Node.js (development, npm run build)\n * 2. Bun compiled binaries (standalone distribution)\n *\n * The challenge: photon-node's CJS entry uses fs.readFileSync(__dirname + '/photon_rs_bg.wasm')\n * which bakes the build machine's absolute path into Bun compiled binaries.\n *\n * Solution:\n * 1. Patch fs.readFileSync to redirect missing photon_rs_bg.wasm reads\n * 2. Copy photon_rs_bg.wasm next to the executable in build:binary\n */\n\nimport type { PathOrFileDescriptor } from \"fs\";\nimport { createRequire } from \"module\";\nimport * as path from \"path\";\nimport { fileURLToPath } from \"url\";\n\nconst require = createRequire(import.meta.url);\nconst fs = require(\"fs\") as typeof import(\"fs\");\n\n// Re-export types from the main package\nexport type { PhotonImage as PhotonImageType } from \"@silvia-odwyer/photon-node\";\n\ntype ReadFileSync = typeof fs.readFileSync;\n\nconst WASM_FILENAME = \"photon_rs_bg.wasm\";\n\n// Lazy-loaded photon module\nlet photonModule: typeof import(\"@silvia-odwyer/photon-node\") | null = null;\nlet loadPromise: Promise<typeof import(\"@silvia-odwyer/photon-node\") | null> | null = null;\n\nfunction pathOrNull(file: PathOrFileDescriptor): string | null {\n\tif (typeof file === \"string\") {\n\t\treturn file;\n\t}\n\tif (file instanceof URL) {\n\t\treturn fileURLToPath(file);\n\t}\n\treturn null;\n}\n\nfunction getFallbackWasmPaths(): string[] {\n\tconst execDir = path.dirname(process.execPath);\n\treturn [\n\t\tpath.join(execDir, WASM_FILENAME),\n\t\tpath.join(execDir, \"photon\", WASM_FILENAME),\n\t\tpath.join(process.cwd(), WASM_FILENAME),\n\t];\n}\n\nfunction patchPhotonWasmRead(): () => void {\n\tconst originalReadFileSync: ReadFileSync = fs.readFileSync.bind(fs);\n\tconst fallbackPaths = getFallbackWasmPaths();\n\tconst mutableFs = fs as { readFileSync: ReadFileSync };\n\n\tconst patchedReadFileSync: ReadFileSync = ((...args: Parameters<ReadFileSync>) => {\n\t\tconst [file, options] = args;\n\t\tconst resolvedPath = pathOrNull(file);\n\n\t\tif (resolvedPath?.endsWith(WASM_FILENAME)) {\n\t\t\ttry {\n\t\t\t\treturn originalReadFileSync(...args);\n\t\t\t} catch (error) {\n\t\t\t\tconst err = error as NodeJS.ErrnoException;\n\t\t\t\tif (err?.code && err.code !== \"ENOENT\") {\n\t\t\t\t\tthrow error;\n\t\t\t\t}\n\n\t\t\t\tfor (const fallbackPath of fallbackPaths) {\n\t\t\t\t\tif (!fs.existsSync(fallbackPath)) {\n\t\t\t\t\t\tcontinue;\n\t\t\t\t\t}\n\t\t\t\t\tif (options === undefined) {\n\t\t\t\t\t\treturn originalReadFileSync(fallbackPath);\n\t\t\t\t\t}\n\t\t\t\t\treturn originalReadFileSync(fallbackPath, options);\n\t\t\t\t}\n\n\t\t\t\tthrow error;\n\t\t\t}\n\t\t}\n\n\t\treturn originalReadFileSync(...args);\n\t}) as ReadFileSync;\n\n\ttry {\n\t\tmutableFs.readFileSync = patchedReadFileSync;\n\t} catch {\n\t\tObject.defineProperty(fs, \"readFileSync\", {\n\t\t\tvalue: patchedReadFileSync,\n\t\t\twritable: true,\n\t\t\tconfigurable: true,\n\t\t});\n\t}\n\n\treturn () => {\n\t\ttry {\n\t\t\tmutableFs.readFileSync = originalReadFileSync;\n\t\t} catch {\n\t\t\tObject.defineProperty(fs, \"readFileSync\", {\n\t\t\t\tvalue: originalReadFileSync,\n\t\t\t\twritable: true,\n\t\t\t\tconfigurable: true,\n\t\t\t});\n\t\t}\n\t};\n}\n\n/**\n * Load the photon module asynchronously.\n * Returns cached module on subsequent calls.\n */\nexport async function loadPhoton(): Promise<typeof import(\"@silvia-odwyer/photon-node\") | null> {\n\tif (photonModule) {\n\t\treturn photonModule;\n\t}\n\n\tif (loadPromise) {\n\t\treturn loadPromise;\n\t}\n\n\tloadPromise = (async () => {\n\t\tconst restoreReadFileSync = patchPhotonWasmRead();\n\t\ttry {\n\t\t\tphotonModule = await import(\"@silvia-odwyer/photon-node\");\n\t\t\treturn photonModule;\n\t\t} catch {\n\t\t\tphotonModule = null;\n\t\t\treturn photonModule;\n\t\t} finally {\n\t\t\trestoreReadFileSync();\n\t\t}\n\t})();\n\n\treturn loadPromise;\n}\n"]}
@@ -0,0 +1,30 @@
1
+ export interface ShellConfig {
2
+ shell: string;
3
+ args: string[];
4
+ }
5
+ /**
6
+ * Resolve shell configuration based on platform and an optional explicit shell path.
7
+ * Resolution order:
8
+ * 1. User-specified shellPath
9
+ * 2. On Windows: Git Bash in known locations, then bash on PATH
10
+ * 3. On Unix: /bin/bash, then bash on PATH, then fallback to sh
11
+ */
12
+ export declare function getShellConfig(customShellPath?: string): ShellConfig;
13
+ export declare function getShellEnv(): NodeJS.ProcessEnv;
14
+ /**
15
+ * Sanitize binary output for display/storage.
16
+ * Removes characters that crash string-width or cause display issues:
17
+ * - Control characters (except tab, newline, carriage return)
18
+ * - Lone surrogates
19
+ * - Unicode Format characters (crash string-width due to a bug)
20
+ * - Characters with undefined code points
21
+ */
22
+ export declare function sanitizeBinaryOutput(str: string): string;
23
+ export declare function trackDetachedChildPid(pid: number): void;
24
+ export declare function untrackDetachedChildPid(pid: number): void;
25
+ export declare function killTrackedDetachedChildren(): void;
26
+ /**
27
+ * Kill a process and all its children (cross-platform)
28
+ */
29
+ export declare function killProcessTree(pid: number): void;
30
+ //# sourceMappingURL=shell.d.ts.map