openvibe 0.57.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (636) hide show
  1. package/CHANGELOG.md +3041 -0
  2. package/README.md +569 -0
  3. package/dist/cli/args.d.ts +44 -0
  4. package/dist/cli/args.d.ts.map +1 -0
  5. package/dist/cli/args.js +272 -0
  6. package/dist/cli/args.js.map +1 -0
  7. package/dist/cli/config-selector.d.ts +10 -0
  8. package/dist/cli/config-selector.d.ts.map +1 -0
  9. package/dist/cli/config-selector.js +26 -0
  10. package/dist/cli/config-selector.js.map +1 -0
  11. package/dist/cli/file-processor.d.ts +10 -0
  12. package/dist/cli/file-processor.d.ts.map +1 -0
  13. package/dist/cli/file-processor.js +68 -0
  14. package/dist/cli/file-processor.js.map +1 -0
  15. package/dist/cli/list-models.d.ts +3 -0
  16. package/dist/cli/list-models.d.ts.map +1 -0
  17. package/dist/cli/list-models.js +30 -0
  18. package/dist/cli/list-models.js.map +1 -0
  19. package/dist/cli/session-picker.d.ts +5 -0
  20. package/dist/cli/session-picker.d.ts.map +1 -0
  21. package/dist/cli/session-picker.js +30 -0
  22. package/dist/cli/session-picker.js.map +1 -0
  23. package/dist/cli.d.ts +3 -0
  24. package/dist/cli.d.ts.map +1 -0
  25. package/dist/cli.js +31 -0
  26. package/dist/cli.js.map +1 -0
  27. package/dist/config.d.ts +29 -0
  28. package/dist/config.d.ts.map +1 -0
  29. package/dist/config.js +145 -0
  30. package/dist/config.js.map +1 -0
  31. package/dist/core/accelerated-client.d.ts +42 -0
  32. package/dist/core/accelerated-client.d.ts.map +1 -0
  33. package/dist/core/accelerated-client.js +97 -0
  34. package/dist/core/accelerated-client.js.map +1 -0
  35. package/dist/core/accelerated-stream.d.ts +41 -0
  36. package/dist/core/accelerated-stream.d.ts.map +1 -0
  37. package/dist/core/accelerated-stream.js +133 -0
  38. package/dist/core/accelerated-stream.js.map +1 -0
  39. package/dist/core/agent-session.d.ts +566 -0
  40. package/dist/core/agent-session.d.ts.map +1 -0
  41. package/dist/core/agent-session.js +2361 -0
  42. package/dist/core/agent-session.js.map +1 -0
  43. package/dist/core/api-concurrency.d.ts +37 -0
  44. package/dist/core/api-concurrency.d.ts.map +1 -0
  45. package/dist/core/api-concurrency.js +179 -0
  46. package/dist/core/api-concurrency.js.map +1 -0
  47. package/dist/core/auth-storage.d.ts +66 -0
  48. package/dist/core/auth-storage.d.ts.map +1 -0
  49. package/dist/core/auth-storage.js +336 -0
  50. package/dist/core/auth-storage.js.map +1 -0
  51. package/dist/core/bash-executor.d.ts +15 -0
  52. package/dist/core/bash-executor.d.ts.map +1 -0
  53. package/dist/core/bash-executor.js +166 -0
  54. package/dist/core/bash-executor.js.map +1 -0
  55. package/dist/core/branded-ai.d.ts +17 -0
  56. package/dist/core/branded-ai.d.ts.map +1 -0
  57. package/dist/core/branded-ai.js +123 -0
  58. package/dist/core/branded-ai.js.map +1 -0
  59. package/dist/core/compaction/branch-summarization.d.ts +37 -0
  60. package/dist/core/compaction/branch-summarization.d.ts.map +1 -0
  61. package/dist/core/compaction/branch-summarization.js +158 -0
  62. package/dist/core/compaction/branch-summarization.js.map +1 -0
  63. package/dist/core/compaction/compaction.d.ts +56 -0
  64. package/dist/core/compaction/compaction.d.ts.map +1 -0
  65. package/dist/core/compaction/compaction.js +501 -0
  66. package/dist/core/compaction/compaction.js.map +1 -0
  67. package/dist/core/compaction/index.d.ts +4 -0
  68. package/dist/core/compaction/index.d.ts.map +1 -0
  69. package/dist/core/compaction/index.js +4 -0
  70. package/dist/core/compaction/index.js.map +1 -0
  71. package/dist/core/compaction/utils.d.ts +17 -0
  72. package/dist/core/compaction/utils.d.ts.map +1 -0
  73. package/dist/core/compaction/utils.js +120 -0
  74. package/dist/core/compaction/utils.js.map +1 -0
  75. package/dist/core/defaults.d.ts +3 -0
  76. package/dist/core/defaults.d.ts.map +1 -0
  77. package/dist/core/defaults.js +2 -0
  78. package/dist/core/defaults.js.map +1 -0
  79. package/dist/core/diagnostics.d.ts +15 -0
  80. package/dist/core/diagnostics.d.ts.map +1 -0
  81. package/dist/core/diagnostics.js +2 -0
  82. package/dist/core/diagnostics.js.map +1 -0
  83. package/dist/core/event-bus.d.ts +9 -0
  84. package/dist/core/event-bus.d.ts.map +1 -0
  85. package/dist/core/event-bus.js +25 -0
  86. package/dist/core/event-bus.js.map +1 -0
  87. package/dist/core/exec.d.ts +13 -0
  88. package/dist/core/exec.d.ts.map +1 -0
  89. package/dist/core/exec.js +61 -0
  90. package/dist/core/exec.js.map +1 -0
  91. package/dist/core/export-html/ansi-to-html.d.ts +3 -0
  92. package/dist/core/export-html/ansi-to-html.d.ts.map +1 -0
  93. package/dist/core/export-html/ansi-to-html.js +193 -0
  94. package/dist/core/export-html/ansi-to-html.js.map +1 -0
  95. package/dist/core/export-html/index.d.ts +22 -0
  96. package/dist/core/export-html/index.d.ts.map +1 -0
  97. package/dist/core/export-html/index.js +193 -0
  98. package/dist/core/export-html/index.js.map +1 -0
  99. package/dist/core/export-html/template.css +971 -0
  100. package/dist/core/export-html/template.html +54 -0
  101. package/dist/core/export-html/template.js +1583 -0
  102. package/dist/core/export-html/tool-renderer.d.ts +21 -0
  103. package/dist/core/export-html/tool-renderer.d.ts.map +1 -0
  104. package/dist/core/export-html/tool-renderer.js +51 -0
  105. package/dist/core/export-html/tool-renderer.js.map +1 -0
  106. package/dist/core/export-html/vendor/highlight.min.js +1213 -0
  107. package/dist/core/export-html/vendor/marked.min.js +6 -0
  108. package/dist/core/extensions/index.d.ts +8 -0
  109. package/dist/core/extensions/index.d.ts.map +1 -0
  110. package/dist/core/extensions/index.js +5 -0
  111. package/dist/core/extensions/index.js.map +1 -0
  112. package/dist/core/extensions/loader.d.ts +7 -0
  113. package/dist/core/extensions/loader.d.ts.map +1 -0
  114. package/dist/core/extensions/loader.js +349 -0
  115. package/dist/core/extensions/loader.js.map +1 -0
  116. package/dist/core/extensions/runner.d.ts +124 -0
  117. package/dist/core/extensions/runner.d.ts.map +1 -0
  118. package/dist/core/extensions/runner.js +657 -0
  119. package/dist/core/extensions/runner.js.map +1 -0
  120. package/dist/core/extensions/types.d.ts +1032 -0
  121. package/dist/core/extensions/types.d.ts.map +1 -0
  122. package/dist/core/extensions/types.js +35 -0
  123. package/dist/core/extensions/types.js.map +1 -0
  124. package/dist/core/extensions/wrapper.d.ts +8 -0
  125. package/dist/core/extensions/wrapper.d.ts.map +1 -0
  126. package/dist/core/extensions/wrapper.js +79 -0
  127. package/dist/core/extensions/wrapper.js.map +1 -0
  128. package/dist/core/footer-data-provider.d.ts +19 -0
  129. package/dist/core/footer-data-provider.d.ts.map +1 -0
  130. package/dist/core/footer-data-provider.js +113 -0
  131. package/dist/core/footer-data-provider.js.map +1 -0
  132. package/dist/core/index.d.ts +13 -0
  133. package/dist/core/index.d.ts.map +1 -0
  134. package/dist/core/index.js +11 -0
  135. package/dist/core/index.js.map +1 -0
  136. package/dist/core/keybindings.d.ts +22 -0
  137. package/dist/core/keybindings.d.ts.map +1 -0
  138. package/dist/core/keybindings.js +124 -0
  139. package/dist/core/keybindings.js.map +1 -0
  140. package/dist/core/messages.d.ts +51 -0
  141. package/dist/core/messages.d.ts.map +1 -0
  142. package/dist/core/messages.js +102 -0
  143. package/dist/core/messages.js.map +1 -0
  144. package/dist/core/model-registry.d.ts +26 -0
  145. package/dist/core/model-registry.d.ts.map +1 -0
  146. package/dist/core/model-registry.js +61 -0
  147. package/dist/core/model-registry.js.map +1 -0
  148. package/dist/core/model-resolver.d.ts +20 -0
  149. package/dist/core/model-resolver.d.ts.map +1 -0
  150. package/dist/core/model-resolver.js +47 -0
  151. package/dist/core/model-resolver.js.map +1 -0
  152. package/dist/core/multi-gpu-executor.d.ts +50 -0
  153. package/dist/core/multi-gpu-executor.d.ts.map +1 -0
  154. package/dist/core/multi-gpu-executor.js +201 -0
  155. package/dist/core/multi-gpu-executor.js.map +1 -0
  156. package/dist/core/onboarding.d.ts +3 -0
  157. package/dist/core/onboarding.d.ts.map +1 -0
  158. package/dist/core/onboarding.js +109 -0
  159. package/dist/core/onboarding.js.map +1 -0
  160. package/dist/core/package-manager.d.ts +136 -0
  161. package/dist/core/package-manager.d.ts.map +1 -0
  162. package/dist/core/package-manager.js +1375 -0
  163. package/dist/core/package-manager.js.map +1 -0
  164. package/dist/core/prompt-templates.d.ts +18 -0
  165. package/dist/core/prompt-templates.d.ts.map +1 -0
  166. package/dist/core/prompt-templates.js +204 -0
  167. package/dist/core/prompt-templates.js.map +1 -0
  168. package/dist/core/resolve-config-value.d.ts +4 -0
  169. package/dist/core/resolve-config-value.d.ts.map +1 -0
  170. package/dist/core/resolve-config-value.js +45 -0
  171. package/dist/core/resolve-config-value.js.map +1 -0
  172. package/dist/core/resource-loader.d.ts +184 -0
  173. package/dist/core/resource-loader.d.ts.map +1 -0
  174. package/dist/core/resource-loader.js +661 -0
  175. package/dist/core/resource-loader.js.map +1 -0
  176. package/dist/core/response-accelerator.d.ts +51 -0
  177. package/dist/core/response-accelerator.d.ts.map +1 -0
  178. package/dist/core/response-accelerator.js +149 -0
  179. package/dist/core/response-accelerator.js.map +1 -0
  180. package/dist/core/sdk.d.ts +39 -0
  181. package/dist/core/sdk.d.ts.map +1 -0
  182. package/dist/core/sdk.js +151 -0
  183. package/dist/core/sdk.js.map +1 -0
  184. package/dist/core/session-manager.d.ts +160 -0
  185. package/dist/core/session-manager.d.ts.map +1 -0
  186. package/dist/core/session-manager.js +899 -0
  187. package/dist/core/session-manager.js.map +1 -0
  188. package/dist/core/settings-manager.d.ts +220 -0
  189. package/dist/core/settings-manager.d.ts.map +1 -0
  190. package/dist/core/settings-manager.js +673 -0
  191. package/dist/core/settings-manager.js.map +1 -0
  192. package/dist/core/skills.d.ts +33 -0
  193. package/dist/core/skills.d.ts.map +1 -0
  194. package/dist/core/skills.js +326 -0
  195. package/dist/core/skills.js.map +1 -0
  196. package/dist/core/slash-commands.d.ts +15 -0
  197. package/dist/core/slash-commands.d.ts.map +1 -0
  198. package/dist/core/slash-commands.js +19 -0
  199. package/dist/core/slash-commands.js.map +1 -0
  200. package/dist/core/system-prompt.d.ts +23 -0
  201. package/dist/core/system-prompt.d.ts.map +1 -0
  202. package/dist/core/system-prompt.js +154 -0
  203. package/dist/core/system-prompt.js.map +1 -0
  204. package/dist/core/timings.d.ts +3 -0
  205. package/dist/core/timings.d.ts.map +1 -0
  206. package/dist/core/timings.js +21 -0
  207. package/dist/core/timings.js.map +1 -0
  208. package/dist/core/tools/bash.d.ts +40 -0
  209. package/dist/core/tools/bash.d.ts.map +1 -0
  210. package/dist/core/tools/bash.js +212 -0
  211. package/dist/core/tools/bash.js.map +1 -0
  212. package/dist/core/tools/edit-diff.d.ts +29 -0
  213. package/dist/core/tools/edit-diff.d.ts.map +1 -0
  214. package/dist/core/tools/edit-diff.js +182 -0
  215. package/dist/core/tools/edit-diff.js.map +1 -0
  216. package/dist/core/tools/edit.d.ts +28 -0
  217. package/dist/core/tools/edit.d.ts.map +1 -0
  218. package/dist/core/tools/edit.js +129 -0
  219. package/dist/core/tools/edit.js.map +1 -0
  220. package/dist/core/tools/fast-executor.d.ts +33 -0
  221. package/dist/core/tools/fast-executor.d.ts.map +1 -0
  222. package/dist/core/tools/fast-executor.js +83 -0
  223. package/dist/core/tools/fast-executor.js.map +1 -0
  224. package/dist/core/tools/find.d.ts +28 -0
  225. package/dist/core/tools/find.d.ts.map +1 -0
  226. package/dist/core/tools/find.js +50 -0
  227. package/dist/core/tools/find.js.map +1 -0
  228. package/dist/core/tools/grep.d.ts +37 -0
  229. package/dist/core/tools/grep.d.ts.map +1 -0
  230. package/dist/core/tools/grep.js +233 -0
  231. package/dist/core/tools/grep.js.map +1 -0
  232. package/dist/core/tools/index.d.ts +63 -0
  233. package/dist/core/tools/index.d.ts.map +1 -0
  234. package/dist/core/tools/index.js +52 -0
  235. package/dist/core/tools/index.js.map +1 -0
  236. package/dist/core/tools/local-accelerator.d.ts +65 -0
  237. package/dist/core/tools/local-accelerator.d.ts.map +1 -0
  238. package/dist/core/tools/local-accelerator.js +198 -0
  239. package/dist/core/tools/local-accelerator.js.map +1 -0
  240. package/dist/core/tools/ls.d.ts +31 -0
  241. package/dist/core/tools/ls.d.ts.map +1 -0
  242. package/dist/core/tools/ls.js +109 -0
  243. package/dist/core/tools/ls.js.map +1 -0
  244. package/dist/core/tools/parallel-executor.d.ts +60 -0
  245. package/dist/core/tools/parallel-executor.d.ts.map +1 -0
  246. package/dist/core/tools/parallel-executor.js +257 -0
  247. package/dist/core/tools/parallel-executor.js.map +1 -0
  248. package/dist/core/tools/path-utils.d.ts +4 -0
  249. package/dist/core/tools/path-utils.d.ts.map +1 -0
  250. package/dist/core/tools/path-utils.js +70 -0
  251. package/dist/core/tools/path-utils.js.map +1 -0
  252. package/dist/core/tools/read.d.ts +29 -0
  253. package/dist/core/tools/read.d.ts.map +1 -0
  254. package/dist/core/tools/read.js +146 -0
  255. package/dist/core/tools/read.js.map +1 -0
  256. package/dist/core/tools/truncate.d.ts +28 -0
  257. package/dist/core/tools/truncate.d.ts.map +1 -0
  258. package/dist/core/tools/truncate.js +161 -0
  259. package/dist/core/tools/truncate.js.map +1 -0
  260. package/dist/core/tools/write.d.ts +21 -0
  261. package/dist/core/tools/write.d.ts.map +1 -0
  262. package/dist/core/tools/write.js +69 -0
  263. package/dist/core/tools/write.js.map +1 -0
  264. package/dist/core/user-config.d.ts +26 -0
  265. package/dist/core/user-config.d.ts.map +1 -0
  266. package/dist/core/user-config.js +75 -0
  267. package/dist/core/user-config.js.map +1 -0
  268. package/dist/index.d.ts +27 -0
  269. package/dist/index.d.ts.map +1 -0
  270. package/dist/index.js +42 -0
  271. package/dist/index.js.map +1 -0
  272. package/dist/main.d.ts +2 -0
  273. package/dist/main.d.ts.map +1 -0
  274. package/dist/main.js +609 -0
  275. package/dist/main.js.map +1 -0
  276. package/dist/migrations.d.ts +8 -0
  277. package/dist/migrations.d.ts.map +1 -0
  278. package/dist/migrations.js +197 -0
  279. package/dist/migrations.js.map +1 -0
  280. package/dist/modes/index.d.ts +6 -0
  281. package/dist/modes/index.d.ts.map +1 -0
  282. package/dist/modes/index.js +5 -0
  283. package/dist/modes/index.js.map +1 -0
  284. package/dist/modes/interactive/components/armin.d.ts +31 -0
  285. package/dist/modes/interactive/components/armin.d.ts.map +1 -0
  286. package/dist/modes/interactive/components/armin.js +306 -0
  287. package/dist/modes/interactive/components/armin.js.map +1 -0
  288. package/dist/modes/interactive/components/assistant-message.d.ts +13 -0
  289. package/dist/modes/interactive/components/assistant-message.d.ts.map +1 -0
  290. package/dist/modes/interactive/components/assistant-message.js +82 -0
  291. package/dist/modes/interactive/components/assistant-message.js.map +1 -0
  292. package/dist/modes/interactive/components/bash-execution.d.ts +23 -0
  293. package/dist/modes/interactive/components/bash-execution.d.ts.map +1 -0
  294. package/dist/modes/interactive/components/bash-execution.js +126 -0
  295. package/dist/modes/interactive/components/bash-execution.js.map +1 -0
  296. package/dist/modes/interactive/components/bordered-loader.d.ts +15 -0
  297. package/dist/modes/interactive/components/bordered-loader.d.ts.map +1 -0
  298. package/dist/modes/interactive/components/bordered-loader.js +50 -0
  299. package/dist/modes/interactive/components/bordered-loader.js.map +1 -0
  300. package/dist/modes/interactive/components/branch-summary-message.d.ts +12 -0
  301. package/dist/modes/interactive/components/branch-summary-message.d.ts.map +1 -0
  302. package/dist/modes/interactive/components/branch-summary-message.js +40 -0
  303. package/dist/modes/interactive/components/branch-summary-message.js.map +1 -0
  304. package/dist/modes/interactive/components/compaction-summary-message.d.ts +12 -0
  305. package/dist/modes/interactive/components/compaction-summary-message.d.ts.map +1 -0
  306. package/dist/modes/interactive/components/compaction-summary-message.js +41 -0
  307. package/dist/modes/interactive/components/compaction-summary-message.js.map +1 -0
  308. package/dist/modes/interactive/components/config-selector.d.ts +68 -0
  309. package/dist/modes/interactive/components/config-selector.d.ts.map +1 -0
  310. package/dist/modes/interactive/components/config-selector.js +451 -0
  311. package/dist/modes/interactive/components/config-selector.js.map +1 -0
  312. package/dist/modes/interactive/components/countdown-timer.d.ts +11 -0
  313. package/dist/modes/interactive/components/countdown-timer.d.ts.map +1 -0
  314. package/dist/modes/interactive/components/countdown-timer.js +30 -0
  315. package/dist/modes/interactive/components/countdown-timer.js.map +1 -0
  316. package/dist/modes/interactive/components/custom-editor.d.ts +14 -0
  317. package/dist/modes/interactive/components/custom-editor.d.ts.map +1 -0
  318. package/dist/modes/interactive/components/custom-editor.js +52 -0
  319. package/dist/modes/interactive/components/custom-editor.js.map +1 -0
  320. package/dist/modes/interactive/components/custom-message.d.ts +16 -0
  321. package/dist/modes/interactive/components/custom-message.d.ts.map +1 -0
  322. package/dist/modes/interactive/components/custom-message.js +66 -0
  323. package/dist/modes/interactive/components/custom-message.js.map +1 -0
  324. package/dist/modes/interactive/components/daxnuts.d.ts +18 -0
  325. package/dist/modes/interactive/components/daxnuts.d.ts.map +1 -0
  326. package/dist/modes/interactive/components/daxnuts.js +130 -0
  327. package/dist/modes/interactive/components/daxnuts.js.map +1 -0
  328. package/dist/modes/interactive/components/diff.d.ts +5 -0
  329. package/dist/modes/interactive/components/diff.d.ts.map +1 -0
  330. package/dist/modes/interactive/components/diff.js +106 -0
  331. package/dist/modes/interactive/components/diff.js.map +1 -0
  332. package/dist/modes/interactive/components/dynamic-border.d.ts +8 -0
  333. package/dist/modes/interactive/components/dynamic-border.d.ts.map +1 -0
  334. package/dist/modes/interactive/components/dynamic-border.js +12 -0
  335. package/dist/modes/interactive/components/dynamic-border.js.map +1 -0
  336. package/dist/modes/interactive/components/extension-editor.d.ts +16 -0
  337. package/dist/modes/interactive/components/extension-editor.d.ts.map +1 -0
  338. package/dist/modes/interactive/components/extension-editor.js +95 -0
  339. package/dist/modes/interactive/components/extension-editor.js.map +1 -0
  340. package/dist/modes/interactive/components/extension-input.d.ts +20 -0
  341. package/dist/modes/interactive/components/extension-input.d.ts.map +1 -0
  342. package/dist/modes/interactive/components/extension-input.js +57 -0
  343. package/dist/modes/interactive/components/extension-input.js.map +1 -0
  344. package/dist/modes/interactive/components/extension-selector.d.ts +20 -0
  345. package/dist/modes/interactive/components/extension-selector.d.ts.map +1 -0
  346. package/dist/modes/interactive/components/extension-selector.js +74 -0
  347. package/dist/modes/interactive/components/extension-selector.js.map +1 -0
  348. package/dist/modes/interactive/components/footer.d.ts +14 -0
  349. package/dist/modes/interactive/components/footer.d.ts.map +1 -0
  350. package/dist/modes/interactive/components/footer.js +78 -0
  351. package/dist/modes/interactive/components/footer.js.map +1 -0
  352. package/dist/modes/interactive/components/index.d.ts +28 -0
  353. package/dist/modes/interactive/components/index.d.ts.map +1 -0
  354. package/dist/modes/interactive/components/index.js +28 -0
  355. package/dist/modes/interactive/components/index.js.map +1 -0
  356. package/dist/modes/interactive/components/keybinding-hints.d.ts +8 -0
  357. package/dist/modes/interactive/components/keybinding-hints.d.ts.map +1 -0
  358. package/dist/modes/interactive/components/keybinding-hints.js +25 -0
  359. package/dist/modes/interactive/components/keybinding-hints.js.map +1 -0
  360. package/dist/modes/interactive/components/onboarding-wizard.d.ts +23 -0
  361. package/dist/modes/interactive/components/onboarding-wizard.d.ts.map +1 -0
  362. package/dist/modes/interactive/components/onboarding-wizard.js +250 -0
  363. package/dist/modes/interactive/components/onboarding-wizard.js.map +1 -0
  364. package/dist/modes/interactive/components/session-selector-search.d.ts +21 -0
  365. package/dist/modes/interactive/components/session-selector-search.d.ts.map +1 -0
  366. package/dist/modes/interactive/components/session-selector-search.js +149 -0
  367. package/dist/modes/interactive/components/session-selector-search.js.map +1 -0
  368. package/dist/modes/interactive/components/session-selector.d.ts +89 -0
  369. package/dist/modes/interactive/components/session-selector.d.ts.map +1 -0
  370. package/dist/modes/interactive/components/session-selector.js +786 -0
  371. package/dist/modes/interactive/components/session-selector.js.map +1 -0
  372. package/dist/modes/interactive/components/settings-selector.d.ts +55 -0
  373. package/dist/modes/interactive/components/settings-selector.d.ts.map +1 -0
  374. package/dist/modes/interactive/components/settings-selector.js +273 -0
  375. package/dist/modes/interactive/components/settings-selector.js.map +1 -0
  376. package/dist/modes/interactive/components/show-images-selector.d.ts +7 -0
  377. package/dist/modes/interactive/components/show-images-selector.d.ts.map +1 -0
  378. package/dist/modes/interactive/components/show-images-selector.js +28 -0
  379. package/dist/modes/interactive/components/show-images-selector.js.map +1 -0
  380. package/dist/modes/interactive/components/skill-invocation-message.d.ts +12 -0
  381. package/dist/modes/interactive/components/skill-invocation-message.d.ts.map +1 -0
  382. package/dist/modes/interactive/components/skill-invocation-message.js +40 -0
  383. package/dist/modes/interactive/components/skill-invocation-message.js.map +1 -0
  384. package/dist/modes/interactive/components/theme-selector.d.ts +8 -0
  385. package/dist/modes/interactive/components/theme-selector.d.ts.map +1 -0
  386. package/dist/modes/interactive/components/theme-selector.js +38 -0
  387. package/dist/modes/interactive/components/theme-selector.js.map +1 -0
  388. package/dist/modes/interactive/components/thinking-selector.d.ts +8 -0
  389. package/dist/modes/interactive/components/thinking-selector.d.ts.map +1 -0
  390. package/dist/modes/interactive/components/thinking-selector.js +40 -0
  391. package/dist/modes/interactive/components/thinking-selector.js.map +1 -0
  392. package/dist/modes/interactive/components/tool-execution.d.ts +54 -0
  393. package/dist/modes/interactive/components/tool-execution.d.ts.map +1 -0
  394. package/dist/modes/interactive/components/tool-execution.js +704 -0
  395. package/dist/modes/interactive/components/tool-execution.js.map +1 -0
  396. package/dist/modes/interactive/components/tree-selector.d.ts +59 -0
  397. package/dist/modes/interactive/components/tree-selector.d.ts.map +1 -0
  398. package/dist/modes/interactive/components/tree-selector.js +929 -0
  399. package/dist/modes/interactive/components/tree-selector.js.map +1 -0
  400. package/dist/modes/interactive/components/user-message-selector.d.ts +24 -0
  401. package/dist/modes/interactive/components/user-message-selector.d.ts.map +1 -0
  402. package/dist/modes/interactive/components/user-message-selector.js +89 -0
  403. package/dist/modes/interactive/components/user-message-selector.js.map +1 -0
  404. package/dist/modes/interactive/components/user-message.d.ts +6 -0
  405. package/dist/modes/interactive/components/user-message.d.ts.map +1 -0
  406. package/dist/modes/interactive/components/user-message.js +24 -0
  407. package/dist/modes/interactive/components/user-message.js.map +1 -0
  408. package/dist/modes/interactive/components/visual-truncate.d.ts +6 -0
  409. package/dist/modes/interactive/components/visual-truncate.d.ts.map +1 -0
  410. package/dist/modes/interactive/components/visual-truncate.js +15 -0
  411. package/dist/modes/interactive/components/visual-truncate.js.map +1 -0
  412. package/dist/modes/interactive/interactive-mode.d.ts +177 -0
  413. package/dist/modes/interactive/interactive-mode.d.ts.map +1 -0
  414. package/dist/modes/interactive/interactive-mode.js +3037 -0
  415. package/dist/modes/interactive/interactive-mode.js.map +1 -0
  416. package/dist/modes/interactive/theme/dark.json +85 -0
  417. package/dist/modes/interactive/theme/light.json +84 -0
  418. package/dist/modes/interactive/theme/theme-schema.json +335 -0
  419. package/dist/modes/interactive/theme/theme.d.ts +60 -0
  420. package/dist/modes/interactive/theme/theme.d.ts.map +1 -0
  421. package/dist/modes/interactive/theme/theme.js +852 -0
  422. package/dist/modes/interactive/theme/theme.js.map +1 -0
  423. package/dist/modes/print-mode.d.ts +10 -0
  424. package/dist/modes/print-mode.d.ts.map +1 -0
  425. package/dist/modes/print-mode.js +80 -0
  426. package/dist/modes/print-mode.js.map +1 -0
  427. package/dist/modes/rpc/jsonl.d.ts +4 -0
  428. package/dist/modes/rpc/jsonl.d.ts.map +1 -0
  429. package/dist/modes/rpc/jsonl.js +36 -0
  430. package/dist/modes/rpc/jsonl.js.map +1 -0
  431. package/dist/modes/rpc/rpc-client.d.ts +94 -0
  432. package/dist/modes/rpc/rpc-client.d.ts.map +1 -0
  433. package/dist/modes/rpc/rpc-client.js +262 -0
  434. package/dist/modes/rpc/rpc-client.js.map +1 -0
  435. package/dist/modes/rpc/rpc-mode.d.ts +3 -0
  436. package/dist/modes/rpc/rpc-mode.d.ts.map +1 -0
  437. package/dist/modes/rpc/rpc-mode.js +227 -0
  438. package/dist/modes/rpc/rpc-mode.js.map +1 -0
  439. package/dist/modes/rpc/rpc-types.d.ts +395 -0
  440. package/dist/modes/rpc/rpc-types.d.ts.map +1 -0
  441. package/dist/modes/rpc/rpc-types.js +2 -0
  442. package/dist/modes/rpc/rpc-types.js.map +1 -0
  443. package/dist/utils/changelog.d.ts +11 -0
  444. package/dist/utils/changelog.d.ts.map +1 -0
  445. package/dist/utils/changelog.js +69 -0
  446. package/dist/utils/changelog.js.map +1 -0
  447. package/dist/utils/clipboard-image.d.ts +11 -0
  448. package/dist/utils/clipboard-image.d.ts.map +1 -0
  449. package/dist/utils/clipboard-image.js +157 -0
  450. package/dist/utils/clipboard-image.js.map +1 -0
  451. package/dist/utils/clipboard-native.d.ts +7 -0
  452. package/dist/utils/clipboard-native.d.ts.map +1 -0
  453. package/dist/utils/clipboard-native.js +14 -0
  454. package/dist/utils/clipboard-native.js.map +1 -0
  455. package/dist/utils/clipboard.d.ts +2 -0
  456. package/dist/utils/clipboard.d.ts.map +1 -0
  457. package/dist/utils/clipboard.js +55 -0
  458. package/dist/utils/clipboard.js.map +1 -0
  459. package/dist/utils/frontmatter.d.ts +8 -0
  460. package/dist/utils/frontmatter.d.ts.map +1 -0
  461. package/dist/utils/frontmatter.js +26 -0
  462. package/dist/utils/frontmatter.js.map +1 -0
  463. package/dist/utils/git.d.ts +10 -0
  464. package/dist/utils/git.d.ts.map +1 -0
  465. package/dist/utils/git.js +156 -0
  466. package/dist/utils/git.js.map +1 -0
  467. package/dist/utils/image-convert.d.ts +5 -0
  468. package/dist/utils/image-convert.d.ts.map +1 -0
  469. package/dist/utils/image-convert.js +28 -0
  470. package/dist/utils/image-convert.js.map +1 -0
  471. package/dist/utils/image-resize.d.ts +19 -0
  472. package/dist/utils/image-resize.d.ts.map +1 -0
  473. package/dist/utils/image-resize.js +152 -0
  474. package/dist/utils/image-resize.js.map +1 -0
  475. package/dist/utils/mime.d.ts +2 -0
  476. package/dist/utils/mime.d.ts.map +1 -0
  477. package/dist/utils/mime.js +26 -0
  478. package/dist/utils/mime.js.map +1 -0
  479. package/dist/utils/photon.d.ts +3 -0
  480. package/dist/utils/photon.d.ts.map +1 -0
  481. package/dist/utils/photon.js +102 -0
  482. package/dist/utils/photon.js.map +1 -0
  483. package/dist/utils/shell.d.ts +8 -0
  484. package/dist/utils/shell.d.ts.map +1 -0
  485. package/dist/utils/shell.js +135 -0
  486. package/dist/utils/shell.js.map +1 -0
  487. package/dist/utils/sleep.d.ts +2 -0
  488. package/dist/utils/sleep.d.ts.map +1 -0
  489. package/dist/utils/sleep.js +14 -0
  490. package/dist/utils/sleep.js.map +1 -0
  491. package/dist/utils/tools-manager.d.ts +3 -0
  492. package/dist/utils/tools-manager.d.ts.map +1 -0
  493. package/dist/utils/tools-manager.js +227 -0
  494. package/dist/utils/tools-manager.js.map +1 -0
  495. package/docs/compaction.md +392 -0
  496. package/docs/custom-provider.md +592 -0
  497. package/docs/development.md +69 -0
  498. package/docs/extensions.md +2023 -0
  499. package/docs/images/doom-extension.png +0 -0
  500. package/docs/images/exy.png +0 -0
  501. package/docs/images/interactive-mode.png +0 -0
  502. package/docs/images/tree-view.png +0 -0
  503. package/docs/json.md +79 -0
  504. package/docs/keybindings.md +182 -0
  505. package/docs/models.md +297 -0
  506. package/docs/packages.md +209 -0
  507. package/docs/prompt-templates.md +67 -0
  508. package/docs/providers.md +188 -0
  509. package/docs/rpc.md +1354 -0
  510. package/docs/sdk.md +968 -0
  511. package/docs/session.md +412 -0
  512. package/docs/settings.md +225 -0
  513. package/docs/shell-aliases.md +13 -0
  514. package/docs/skills.md +231 -0
  515. package/docs/terminal-setup.md +87 -0
  516. package/docs/termux.md +127 -0
  517. package/docs/themes.md +295 -0
  518. package/docs/tmux.md +61 -0
  519. package/docs/tree.md +228 -0
  520. package/docs/tui.md +887 -0
  521. package/docs/windows.md +17 -0
  522. package/examples/README.md +25 -0
  523. package/examples/extensions/README.md +205 -0
  524. package/examples/extensions/antigravity-image-gen.ts +415 -0
  525. package/examples/extensions/auto-commit-on-exit.ts +49 -0
  526. package/examples/extensions/bash-spawn-hook.ts +30 -0
  527. package/examples/extensions/bookmark.ts +50 -0
  528. package/examples/extensions/built-in-tool-renderer.ts +246 -0
  529. package/examples/extensions/claude-rules.ts +86 -0
  530. package/examples/extensions/commands.ts +72 -0
  531. package/examples/extensions/confirm-destructive.ts +59 -0
  532. package/examples/extensions/custom-compaction.ts +114 -0
  533. package/examples/extensions/custom-footer.ts +64 -0
  534. package/examples/extensions/custom-header.ts +73 -0
  535. package/examples/extensions/custom-provider-anthropic/index.ts +604 -0
  536. package/examples/extensions/custom-provider-anthropic/package-lock.json +24 -0
  537. package/examples/extensions/custom-provider-anthropic/package.json +19 -0
  538. package/examples/extensions/custom-provider-gitlab-duo/index.ts +349 -0
  539. package/examples/extensions/custom-provider-gitlab-duo/package.json +16 -0
  540. package/examples/extensions/custom-provider-gitlab-duo/test.ts +82 -0
  541. package/examples/extensions/custom-provider-qwen-cli/index.ts +345 -0
  542. package/examples/extensions/custom-provider-qwen-cli/package.json +16 -0
  543. package/examples/extensions/dirty-repo-guard.ts +56 -0
  544. package/examples/extensions/doom-overlay/README.md +46 -0
  545. package/examples/extensions/doom-overlay/doom/build/doom.js +21 -0
  546. package/examples/extensions/doom-overlay/doom/build/doom.wasm +0 -0
  547. package/examples/extensions/doom-overlay/doom/build.sh +152 -0
  548. package/examples/extensions/doom-overlay/doom/doomgeneric_pi.c +72 -0
  549. package/examples/extensions/doom-overlay/doom-component.ts +132 -0
  550. package/examples/extensions/doom-overlay/doom-engine.ts +173 -0
  551. package/examples/extensions/doom-overlay/doom-keys.ts +104 -0
  552. package/examples/extensions/doom-overlay/index.ts +74 -0
  553. package/examples/extensions/doom-overlay/wad-finder.ts +51 -0
  554. package/examples/extensions/dynamic-resources/SKILL.md +8 -0
  555. package/examples/extensions/dynamic-resources/dynamic.json +79 -0
  556. package/examples/extensions/dynamic-resources/dynamic.md +5 -0
  557. package/examples/extensions/dynamic-resources/index.ts +15 -0
  558. package/examples/extensions/dynamic-tools.ts +74 -0
  559. package/examples/extensions/event-bus.ts +43 -0
  560. package/examples/extensions/file-trigger.ts +41 -0
  561. package/examples/extensions/git-checkpoint.ts +53 -0
  562. package/examples/extensions/handoff.ts +150 -0
  563. package/examples/extensions/hello.ts +25 -0
  564. package/examples/extensions/inline-bash.ts +94 -0
  565. package/examples/extensions/input-transform.ts +43 -0
  566. package/examples/extensions/interactive-shell.ts +196 -0
  567. package/examples/extensions/mac-system-theme.ts +47 -0
  568. package/examples/extensions/message-renderer.ts +59 -0
  569. package/examples/extensions/minimal-mode.ts +426 -0
  570. package/examples/extensions/modal-editor.ts +85 -0
  571. package/examples/extensions/model-status.ts +31 -0
  572. package/examples/extensions/notify.ts +55 -0
  573. package/examples/extensions/overlay-qa-tests.ts +1348 -0
  574. package/examples/extensions/overlay-test.ts +150 -0
  575. package/examples/extensions/permission-gate.ts +34 -0
  576. package/examples/extensions/pirate.ts +47 -0
  577. package/examples/extensions/plan-mode/README.md +65 -0
  578. package/examples/extensions/plan-mode/index.ts +340 -0
  579. package/examples/extensions/plan-mode/utils.ts +168 -0
  580. package/examples/extensions/preset.ts +398 -0
  581. package/examples/extensions/protected-paths.ts +30 -0
  582. package/examples/extensions/provider-payload.ts +14 -0
  583. package/examples/extensions/qna.ts +119 -0
  584. package/examples/extensions/question.ts +264 -0
  585. package/examples/extensions/questionnaire.ts +427 -0
  586. package/examples/extensions/rainbow-editor.ts +88 -0
  587. package/examples/extensions/reload-runtime.ts +37 -0
  588. package/examples/extensions/rpc-demo.ts +124 -0
  589. package/examples/extensions/sandbox/index.ts +318 -0
  590. package/examples/extensions/sandbox/package-lock.json +92 -0
  591. package/examples/extensions/sandbox/package.json +19 -0
  592. package/examples/extensions/send-user-message.ts +97 -0
  593. package/examples/extensions/session-name.ts +27 -0
  594. package/examples/extensions/shutdown-command.ts +63 -0
  595. package/examples/extensions/snake.ts +343 -0
  596. package/examples/extensions/space-invaders.ts +560 -0
  597. package/examples/extensions/ssh.ts +220 -0
  598. package/examples/extensions/status-line.ts +40 -0
  599. package/examples/extensions/subagent/README.md +172 -0
  600. package/examples/extensions/subagent/agents/planner.md +37 -0
  601. package/examples/extensions/subagent/agents/reviewer.md +35 -0
  602. package/examples/extensions/subagent/agents/scout.md +50 -0
  603. package/examples/extensions/subagent/agents/worker.md +24 -0
  604. package/examples/extensions/subagent/agents.ts +126 -0
  605. package/examples/extensions/subagent/index.ts +964 -0
  606. package/examples/extensions/subagent/prompts/implement-and-review.md +10 -0
  607. package/examples/extensions/subagent/prompts/implement.md +10 -0
  608. package/examples/extensions/subagent/prompts/scout-and-plan.md +9 -0
  609. package/examples/extensions/summarize.ts +195 -0
  610. package/examples/extensions/system-prompt-header.ts +17 -0
  611. package/examples/extensions/timed-confirm.ts +70 -0
  612. package/examples/extensions/titlebar-spinner.ts +58 -0
  613. package/examples/extensions/todo.ts +299 -0
  614. package/examples/extensions/tool-override.ts +143 -0
  615. package/examples/extensions/tools.ts +146 -0
  616. package/examples/extensions/trigger-compact.ts +40 -0
  617. package/examples/extensions/truncated-tool.ts +192 -0
  618. package/examples/extensions/widget-placement.ts +17 -0
  619. package/examples/extensions/with-deps/index.ts +32 -0
  620. package/examples/extensions/with-deps/package-lock.json +31 -0
  621. package/examples/extensions/with-deps/package.json +22 -0
  622. package/examples/rpc-extension-ui.ts +632 -0
  623. package/examples/sdk/01-minimal.ts +22 -0
  624. package/examples/sdk/02-custom-model.ts +49 -0
  625. package/examples/sdk/03-custom-prompt.ts +55 -0
  626. package/examples/sdk/04-skills.ts +46 -0
  627. package/examples/sdk/05-tools.ts +56 -0
  628. package/examples/sdk/06-extensions.ts +88 -0
  629. package/examples/sdk/07-context-files.ts +40 -0
  630. package/examples/sdk/08-prompt-templates.ts +47 -0
  631. package/examples/sdk/09-api-keys-and-oauth.ts +48 -0
  632. package/examples/sdk/10-settings.ts +51 -0
  633. package/examples/sdk/11-sessions.ts +48 -0
  634. package/examples/sdk/12-full-control.ts +82 -0
  635. package/examples/sdk/README.md +144 -0
  636. package/package.json +100 -0
@@ -0,0 +1,179 @@
1
+ class ConcurrencyManager {
2
+ maxConcurrent;
3
+ activeRequests = new Set();
4
+ queue = [];
5
+ requestStartTimes = new Map();
6
+ latencies = [];
7
+ constructor(maxConcurrent = 4) {
8
+ this.maxConcurrent = maxConcurrent;
9
+ }
10
+ updateConcurrency(newConcurrency) {
11
+ const old = this.maxConcurrent;
12
+ this.maxConcurrent = Math.max(1, newConcurrency);
13
+ this.processQueue();
14
+ }
15
+ async execute(requestId, task, signal) {
16
+ if (this.activeRequests.size >= this.maxConcurrent) {
17
+ await this.waitForSlot(requestId, signal);
18
+ }
19
+ if (signal?.aborted) {
20
+ throw new Error("Request aborted");
21
+ }
22
+ this.activeRequests.add(requestId);
23
+ this.requestStartTimes.set(requestId, Date.now());
24
+ try {
25
+ const result = await task();
26
+ return result;
27
+ }
28
+ finally {
29
+ const startTime = this.requestStartTimes.get(requestId);
30
+ if (startTime) {
31
+ const latency = Date.now() - startTime;
32
+ this.latencies.push(latency);
33
+ if (this.latencies.length > 100) {
34
+ this.latencies.shift();
35
+ }
36
+ this.requestStartTimes.delete(requestId);
37
+ }
38
+ this.activeRequests.delete(requestId);
39
+ this.processQueue();
40
+ }
41
+ }
42
+ async waitForSlot(requestId, signal) {
43
+ return new Promise((resolve, reject) => {
44
+ const checkQueue = () => {
45
+ if (this.activeRequests.size < this.maxConcurrent) {
46
+ resolve();
47
+ return true;
48
+ }
49
+ if (signal?.aborted) {
50
+ reject(new Error("Request aborted"));
51
+ return true;
52
+ }
53
+ return false;
54
+ };
55
+ if (!checkQueue()) {
56
+ const interval = setInterval(() => {
57
+ if (checkQueue()) {
58
+ clearInterval(interval);
59
+ }
60
+ }, 10);
61
+ }
62
+ });
63
+ }
64
+ processQueue() {
65
+ while (this.queue.length > 0 && this.activeRequests.size < this.maxConcurrent) {
66
+ const next = this.queue.shift();
67
+ if (next) {
68
+ this.executeRequest(next);
69
+ }
70
+ }
71
+ }
72
+ async executeRequest(pending) {
73
+ try {
74
+ const stream = await import("@mariozechner/pi-ai");
75
+ const response = await stream.streamSimple(pending.model, pending.context, pending.options);
76
+ const result = await response.result();
77
+ pending.resolve(result);
78
+ }
79
+ catch (error) {
80
+ pending.reject(error instanceof Error ? error : new Error(String(error)));
81
+ }
82
+ }
83
+ getStats() {
84
+ const avgLatency = this.latencies.length > 0 ? this.latencies.reduce((a, b) => a + b, 0) / this.latencies.length : 0;
85
+ return {
86
+ activeRequests: this.activeRequests.size,
87
+ queuedRequests: this.queue.length,
88
+ totalCompleted: this.latencies.length,
89
+ totalErrors: 0,
90
+ averageLatency: avgLatency,
91
+ };
92
+ }
93
+ clear() {
94
+ this.queue.forEach((req) => {
95
+ req.reject(new Error("Queue cleared"));
96
+ });
97
+ this.queue = [];
98
+ }
99
+ }
100
+ function detectOptimalConcurrency() {
101
+ try {
102
+ const os = require("os");
103
+ const cpuCores = os.cpus().length;
104
+ try {
105
+ const result = require("child_process").execSync("wmic path win32_VideoController get name", {
106
+ encoding: "utf-8",
107
+ });
108
+ const gpuCount = (result.match(/Name/g) || []).length;
109
+ return Math.max(2, gpuCount || cpuCores);
110
+ }
111
+ catch {
112
+ return Math.max(2, Math.floor(cpuCores / 2));
113
+ }
114
+ }
115
+ catch {
116
+ return 4;
117
+ }
118
+ }
119
+ const globalConcurrencyManager = new ConcurrencyManager(detectOptimalConcurrency());
120
+ export class APIParallelExecutor {
121
+ concurrencyManager;
122
+ constructor(concurrency) {
123
+ this.concurrencyManager = concurrency ? new ConcurrencyManager(concurrency) : globalConcurrencyManager;
124
+ }
125
+ async batchRequest(requests, options) {
126
+ const promises = requests.map((req, index) => {
127
+ return this.concurrencyManager.execute(`req-${Date.now()}-${index}`, async () => {
128
+ const stream = await import("@mariozechner/pi-ai");
129
+ const response = await stream.streamSimple(req.model, req.context, req.options);
130
+ return response.result();
131
+ }, options?.signal);
132
+ });
133
+ return Promise.all(promises);
134
+ }
135
+ async *streamBatch(requests, options) {
136
+ const promises = requests.map((req, index) => {
137
+ return this.concurrencyManager
138
+ .execute(`stream-req-${Date.now()}-${index}`, async () => {
139
+ const stream = await import("@mariozechner/pi-ai");
140
+ const response = await stream.streamSimple(req.model, req.context, req.options);
141
+ return response.result();
142
+ }, options?.signal)
143
+ .then((result) => ({ index, result }));
144
+ });
145
+ const remaining = new Set(promises.map((p, i) => i));
146
+ while (remaining.size > 0) {
147
+ if (options?.signal?.aborted) {
148
+ throw new Error("Stream batch aborted");
149
+ }
150
+ for (const idx of remaining) {
151
+ const promise = promises[idx];
152
+ if (promise && (await Promise.race([promise, Promise.resolve(null)]))) {
153
+ const result = await promise;
154
+ yield result;
155
+ remaining.delete(idx);
156
+ break;
157
+ }
158
+ }
159
+ if (remaining.size > 0) {
160
+ await new Promise((resolve) => setTimeout(resolve, 5));
161
+ }
162
+ }
163
+ }
164
+ getStats() {
165
+ return this.concurrencyManager.getStats();
166
+ }
167
+ setConcurrency(concurrency) {
168
+ this.concurrencyManager.updateConcurrency(concurrency);
169
+ }
170
+ getOptimalConcurrency() {
171
+ return detectOptimalConcurrency();
172
+ }
173
+ clear() {
174
+ this.concurrencyManager.clear();
175
+ }
176
+ }
177
+ export const globalParallelExecutor = new APIParallelExecutor();
178
+ export { detectOptimalConcurrency };
179
+ //# sourceMappingURL=api-concurrency.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"api-concurrency.js","sourceRoot":"","sources":["../../src/core/api-concurrency.ts"],"names":[],"mappings":"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","sourcesContent":["import type { Api, AssistantMessage, Context, Model } from \"@mariozechner/pi-ai\";\n\ninterface PendingRequest {\n\tid: string;\n\tmodel: Model<any>;\n\tcontext: Context;\n\tresolve: (value: AssistantMessage) => void;\n\treject: (error: Error) => void;\n\tsignal?: AbortSignal;\n\toptions?: any;\n}\ninterface ConcurrencyStats {\n\tactiveRequests: number;\n\tqueuedRequests: number;\n\ttotalCompleted: number;\n\ttotalErrors: number;\n\taverageLatency: number;\n}\nclass ConcurrencyManager {\n\tprivate maxConcurrent: number;\n\tprivate activeRequests: Set<string> = new Set();\n\tprivate queue: PendingRequest[] = [];\n\tprivate requestStartTimes: Map<string, number> = new Map();\n\tprivate latencies: number[] = [];\n\tconstructor(maxConcurrent: number = 4) {\n\t\tthis.maxConcurrent = maxConcurrent;\n\t}\n\tupdateConcurrency(newConcurrency: number): void {\n\t\tconst old = this.maxConcurrent;\n\t\tthis.maxConcurrent = Math.max(1, newConcurrency);\n\t\tthis.processQueue();\n\t}\n\tasync execute<T>(requestId: string, task: () => Promise<T>, signal?: AbortSignal): Promise<T> {\n\t\tif (this.activeRequests.size >= this.maxConcurrent) {\n\t\t\tawait this.waitForSlot(requestId, signal);\n\t\t}\n\t\tif (signal?.aborted) {\n\t\t\tthrow new Error(\"Request aborted\");\n\t\t}\n\t\tthis.activeRequests.add(requestId);\n\t\tthis.requestStartTimes.set(requestId, Date.now());\n\t\ttry {\n\t\t\tconst result = await task();\n\t\t\treturn result;\n\t\t} finally {\n\t\t\tconst startTime = this.requestStartTimes.get(requestId);\n\t\t\tif (startTime) {\n\t\t\t\tconst latency = Date.now() - startTime;\n\t\t\t\tthis.latencies.push(latency);\n\t\t\t\tif (this.latencies.length > 100) {\n\t\t\t\t\tthis.latencies.shift();\n\t\t\t\t}\n\t\t\t\tthis.requestStartTimes.delete(requestId);\n\t\t\t}\n\t\t\tthis.activeRequests.delete(requestId);\n\t\t\tthis.processQueue();\n\t\t}\n\t}\n\tprivate async waitForSlot(requestId: string, signal?: AbortSignal): Promise<void> {\n\t\treturn new Promise((resolve, reject) => {\n\t\t\tconst checkQueue = () => {\n\t\t\t\tif (this.activeRequests.size < this.maxConcurrent) {\n\t\t\t\t\tresolve();\n\t\t\t\t\treturn true;\n\t\t\t\t}\n\t\t\t\tif (signal?.aborted) {\n\t\t\t\t\treject(new Error(\"Request aborted\"));\n\t\t\t\t\treturn true;\n\t\t\t\t}\n\t\t\t\treturn false;\n\t\t\t};\n\t\t\tif (!checkQueue()) {\n\t\t\t\tconst interval = setInterval(() => {\n\t\t\t\t\tif (checkQueue()) {\n\t\t\t\t\t\tclearInterval(interval);\n\t\t\t\t\t}\n\t\t\t\t}, 10);\n\t\t\t}\n\t\t});\n\t}\n\tprivate processQueue(): void {\n\t\twhile (this.queue.length > 0 && this.activeRequests.size < this.maxConcurrent) {\n\t\t\tconst next = this.queue.shift();\n\t\t\tif (next) {\n\t\t\t\tthis.executeRequest(next);\n\t\t\t}\n\t\t}\n\t}\n\tprivate async executeRequest(pending: PendingRequest): Promise<void> {\n\t\ttry {\n\t\t\tconst stream = await import(\"@mariozechner/pi-ai\");\n\t\t\tconst response = await stream.streamSimple(pending.model, pending.context, pending.options);\n\t\t\tconst result = await response.result();\n\t\t\tpending.resolve(result);\n\t\t} catch (error) {\n\t\t\tpending.reject(error instanceof Error ? error : new Error(String(error)));\n\t\t}\n\t}\n\tgetStats(): ConcurrencyStats {\n\t\tconst avgLatency =\n\t\t\tthis.latencies.length > 0 ? this.latencies.reduce((a, b) => a + b, 0) / this.latencies.length : 0;\n\t\treturn {\n\t\t\tactiveRequests: this.activeRequests.size,\n\t\t\tqueuedRequests: this.queue.length,\n\t\t\ttotalCompleted: this.latencies.length,\n\t\t\ttotalErrors: 0,\n\t\t\taverageLatency: avgLatency,\n\t\t};\n\t}\n\tclear(): void {\n\t\tthis.queue.forEach((req) => {\n\t\t\treq.reject(new Error(\"Queue cleared\"));\n\t\t});\n\t\tthis.queue = [];\n\t}\n}\nfunction detectOptimalConcurrency(): number {\n\ttry {\n\t\tconst os = require(\"os\");\n\t\tconst cpuCores = os.cpus().length;\n\t\ttry {\n\t\t\tconst result = require(\"child_process\").execSync(\"wmic path win32_VideoController get name\", {\n\t\t\t\tencoding: \"utf-8\",\n\t\t\t});\n\t\t\tconst gpuCount = (result.match(/Name/g) || []).length;\n\t\t\treturn Math.max(2, gpuCount || cpuCores);\n\t\t} catch {\n\t\t\treturn Math.max(2, Math.floor(cpuCores / 2));\n\t\t}\n\t} catch {\n\t\treturn 4;\n\t}\n}\nconst globalConcurrencyManager = new ConcurrencyManager(detectOptimalConcurrency());\nexport class APIParallelExecutor {\n\tprivate concurrencyManager: ConcurrencyManager;\n\tconstructor(concurrency?: number) {\n\t\tthis.concurrencyManager = concurrency ? new ConcurrencyManager(concurrency) : globalConcurrencyManager;\n\t}\n\tasync batchRequest(\n\t\trequests: Array<{\n\t\t\tmodel: Model<any>;\n\t\t\tcontext: Context;\n\t\t\toptions?: any;\n\t\t}>,\n\t\toptions?: { signal?: AbortSignal },\n\t): Promise<AssistantMessage[]> {\n\t\tconst promises = requests.map((req, index) => {\n\t\t\treturn this.concurrencyManager.execute(\n\t\t\t\t`req-${Date.now()}-${index}`,\n\t\t\t\tasync () => {\n\t\t\t\t\tconst stream = await import(\"@mariozechner/pi-ai\");\n\t\t\t\t\tconst response = await stream.streamSimple(req.model, req.context, req.options);\n\t\t\t\t\treturn response.result();\n\t\t\t\t},\n\t\t\t\toptions?.signal,\n\t\t\t);\n\t\t});\n\t\treturn Promise.all(promises);\n\t}\n\tasync *streamBatch(\n\t\trequests: Array<{\n\t\t\tmodel: Model<any>;\n\t\t\tcontext: Context;\n\t\t\toptions?: any;\n\t\t}>,\n\t\toptions?: { signal?: AbortSignal },\n\t): AsyncGenerator<{ index: number; result: AssistantMessage }, void, void> {\n\t\tconst promises = requests.map((req, index) => {\n\t\t\treturn this.concurrencyManager\n\t\t\t\t.execute(\n\t\t\t\t\t`stream-req-${Date.now()}-${index}`,\n\t\t\t\t\tasync () => {\n\t\t\t\t\t\tconst stream = await import(\"@mariozechner/pi-ai\");\n\t\t\t\t\t\tconst response = await stream.streamSimple(req.model, req.context, req.options);\n\t\t\t\t\t\treturn response.result();\n\t\t\t\t\t},\n\t\t\t\t\toptions?.signal,\n\t\t\t\t)\n\t\t\t\t.then((result) => ({ index, result }));\n\t\t});\n\t\tconst remaining = new Set(promises.map((p, i) => i));\n\t\twhile (remaining.size > 0) {\n\t\t\tif (options?.signal?.aborted) {\n\t\t\t\tthrow new Error(\"Stream batch aborted\");\n\t\t\t}\n\t\t\tfor (const idx of remaining) {\n\t\t\t\tconst promise = promises[idx];\n\t\t\t\tif (promise && (await Promise.race([promise, Promise.resolve(null)]))) {\n\t\t\t\t\tconst result = await promise;\n\t\t\t\t\tyield result;\n\t\t\t\t\tremaining.delete(idx);\n\t\t\t\t\tbreak;\n\t\t\t\t}\n\t\t\t}\n\t\t\tif (remaining.size > 0) {\n\t\t\t\tawait new Promise((resolve) => setTimeout(resolve, 5));\n\t\t\t}\n\t\t}\n\t}\n\tgetStats(): ConcurrencyStats {\n\t\treturn this.concurrencyManager.getStats();\n\t}\n\tsetConcurrency(concurrency: number): void {\n\t\tthis.concurrencyManager.updateConcurrency(concurrency);\n\t}\n\tgetOptimalConcurrency(): number {\n\t\treturn detectOptimalConcurrency();\n\t}\n\tclear(): void {\n\t\tthis.concurrencyManager.clear();\n\t}\n}\nexport const globalParallelExecutor = new APIParallelExecutor();\nexport { detectOptimalConcurrency, type ConcurrencyStats };\n"]}
@@ -0,0 +1,66 @@
1
+ import { type OAuthCredentials, type OAuthLoginCallbacks, type OAuthProviderId } from "@mariozechner/pi-ai";
2
+ export type ApiKeyCredential = {
3
+ type: "api_key";
4
+ key: string;
5
+ };
6
+ export type OAuthCredential = {
7
+ type: "oauth";
8
+ } & OAuthCredentials;
9
+ export type AuthCredential = ApiKeyCredential | OAuthCredential;
10
+ export type AuthStorageData = Record<string, AuthCredential>;
11
+ type LockResult<T> = {
12
+ result: T;
13
+ next?: string;
14
+ };
15
+ export interface AuthStorageBackend {
16
+ withLock<T>(fn: (current: string | undefined) => LockResult<T>): T;
17
+ withLockAsync<T>(fn: (current: string | undefined) => Promise<LockResult<T>>): Promise<T>;
18
+ }
19
+ export declare class FileAuthStorageBackend implements AuthStorageBackend {
20
+ private authPath;
21
+ constructor(authPath?: string);
22
+ private ensureParentDir;
23
+ private ensureFileExists;
24
+ private acquireLockSyncWithRetry;
25
+ withLock<T>(fn: (current: string | undefined) => LockResult<T>): T;
26
+ withLockAsync<T>(fn: (current: string | undefined) => Promise<LockResult<T>>): Promise<T>;
27
+ }
28
+ export declare class InMemoryAuthStorageBackend implements AuthStorageBackend {
29
+ private value;
30
+ withLock<T>(fn: (current: string | undefined) => LockResult<T>): T;
31
+ withLockAsync<T>(fn: (current: string | undefined) => Promise<LockResult<T>>): Promise<T>;
32
+ }
33
+ export declare class AuthStorage {
34
+ private storage;
35
+ private data;
36
+ private runtimeOverrides;
37
+ private fallbackResolver?;
38
+ private loadError;
39
+ private errors;
40
+ private constructor();
41
+ static create(authPath?: string): AuthStorage;
42
+ static fromStorage(storage: AuthStorageBackend): AuthStorage;
43
+ static inMemory(data?: AuthStorageData): AuthStorage;
44
+ setRuntimeApiKey(provider: string, apiKey: string): void;
45
+ removeRuntimeApiKey(provider: string): void;
46
+ setFallbackResolver(resolver: (provider: string) => string | undefined): void;
47
+ private recordError;
48
+ private parseStorageData;
49
+ reload(): void;
50
+ private persistProviderChange;
51
+ get(provider: string): AuthCredential | undefined;
52
+ set(provider: string, credential: AuthCredential): void;
53
+ remove(provider: string): void;
54
+ list(): string[];
55
+ has(provider: string): boolean;
56
+ hasAuth(provider: string): boolean;
57
+ getAll(): AuthStorageData;
58
+ drainErrors(): Error[];
59
+ login(providerId: OAuthProviderId, callbacks: OAuthLoginCallbacks): Promise<void>;
60
+ logout(provider: string): void;
61
+ private refreshOAuthTokenWithLock;
62
+ getApiKey(providerId: string): Promise<string | undefined>;
63
+ getOAuthProviders(): import("@mariozechner/pi-ai").OAuthProviderInterface[];
64
+ }
65
+ export {};
66
+ //# sourceMappingURL=auth-storage.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"auth-storage.d.ts","sourceRoot":"","sources":["../../src/core/auth-storage.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n\tgetEnvApiKey,\n\ttype OAuthCredentials,\n\ttype OAuthLoginCallbacks,\n\ttype OAuthProviderId,\n} from \"@mariozechner/pi-ai\";\nimport { getOAuthApiKey, getOAuthProvider, getOAuthProviders } from \"@mariozechner/pi-ai/oauth\";\nimport { chmodSync, existsSync, mkdirSync, readFileSync, writeFileSync } from \"fs\";\nimport { dirname, join } from \"path\";\nimport lockfile from \"proper-lockfile\";\nimport { getAgentDir } from \"../config.js\";\nimport { resolveConfigValue } from \"./resolve-config-value.js\";\nexport type ApiKeyCredential = {\n\ttype: \"api_key\";\n\tkey: string;\n};\nexport type OAuthCredential = {\n\ttype: \"oauth\";\n} & OAuthCredentials;\nexport type AuthCredential = ApiKeyCredential | OAuthCredential;\nexport type AuthStorageData = Record<string, AuthCredential>;\ntype LockResult<T> = {\n\tresult: T;\n\tnext?: string;\n};\nexport interface AuthStorageBackend {\n\twithLock<T>(fn: (current: string | undefined) => LockResult<T>): T;\n\twithLockAsync<T>(fn: (current: string | undefined) => Promise<LockResult<T>>): Promise<T>;\n}\nexport class FileAuthStorageBackend implements AuthStorageBackend {\n\tconstructor(private authPath: string = join(getAgentDir(), \"auth.json\")) {}\n\tprivate ensureParentDir(): void {\n\t\tconst dir = dirname(this.authPath);\n\t\tif (!existsSync(dir)) {\n\t\t\tmkdirSync(dir, { recursive: true, mode: 0o700 });\n\t\t}\n\t}\n\tprivate ensureFileExists(): void {\n\t\tif (!existsSync(this.authPath)) {\n\t\t\twriteFileSync(this.authPath, \"{}\", \"utf-8\");\n\t\t\tchmodSync(this.authPath, 0o600);\n\t\t}\n\t}\n\tprivate acquireLockSyncWithRetry(path: string): () => void {\n\t\tconst maxAttempts = 10;\n\t\tconst delayMs = 20;\n\t\tlet lastError: unknown;\n\t\tfor (let attempt = 1; attempt <= maxAttempts; attempt++) {\n\t\t\ttry {\n\t\t\t\treturn lockfile.lockSync(path, { realpath: false });\n\t\t\t} catch (error) {\n\t\t\t\tconst code =\n\t\t\t\t\ttypeof error === \"object\" && error !== null && \"code\" in error\n\t\t\t\t\t\t? String((error as { code?: unknown }).code)\n\t\t\t\t\t\t: undefined;\n\t\t\t\tif (code !== \"ELOCKED\" || attempt === maxAttempts) {\n\t\t\t\t\tthrow error;\n\t\t\t\t}\n\t\t\t\tlastError = error;\n\t\t\t\tconst start = Date.now();\n\t\t\t\twhile (Date.now() - start < delayMs) {}\n\t\t\t}\n\t\t}\n\t\tthrow (lastError as Error) ?? new Error(\"Failed to acquire auth storage lock\");\n\t}\n\twithLock<T>(fn: (current: string | undefined) => LockResult<T>): T {\n\t\tthis.ensureParentDir();\n\t\tthis.ensureFileExists();\n\t\tlet release: (() => void) | undefined;\n\t\ttry {\n\t\t\trelease = this.acquireLockSyncWithRetry(this.authPath);\n\t\t\tconst current = existsSync(this.authPath) ? readFileSync(this.authPath, \"utf-8\") : undefined;\n\t\t\tconst { result, next } = fn(current);\n\t\t\tif (next !== undefined) {\n\t\t\t\twriteFileSync(this.authPath, next, \"utf-8\");\n\t\t\t\tchmodSync(this.authPath, 0o600);\n\t\t\t}\n\t\t\treturn result;\n\t\t} finally {\n\t\t\tif (release) {\n\t\t\t\trelease();\n\t\t\t}\n\t\t}\n\t}\n\tasync withLockAsync<T>(fn: (current: string | undefined) => Promise<LockResult<T>>): Promise<T> {\n\t\tthis.ensureParentDir();\n\t\tthis.ensureFileExists();\n\t\tlet release: (() => Promise<void>) | undefined;\n\t\tlet lockCompromised = false;\n\t\tlet lockCompromisedError: Error | undefined;\n\t\tconst throwIfCompromised = () => {\n\t\t\tif (lockCompromised) {\n\t\t\t\tthrow lockCompromisedError ?? new Error(\"Auth storage lock was compromised\");\n\t\t\t}\n\t\t};\n\t\ttry {\n\t\t\trelease = await lockfile.lock(this.authPath, {\n\t\t\t\tretries: {\n\t\t\t\t\tretries: 10,\n\t\t\t\t\tfactor: 2,\n\t\t\t\t\tminTimeout: 100,\n\t\t\t\t\tmaxTimeout: 10000,\n\t\t\t\t\trandomize: true,\n\t\t\t\t},\n\t\t\t\tstale: 30000,\n\t\t\t\tonCompromised: (err) => {\n\t\t\t\t\tlockCompromised = true;\n\t\t\t\t\tlockCompromisedError = err;\n\t\t\t\t},\n\t\t\t});\n\t\t\tthrowIfCompromised();\n\t\t\tconst current = existsSync(this.authPath) ? readFileSync(this.authPath, \"utf-8\") : undefined;\n\t\t\tconst { result, next } = await fn(current);\n\t\t\tthrowIfCompromised();\n\t\t\tif (next !== undefined) {\n\t\t\t\twriteFileSync(this.authPath, next, \"utf-8\");\n\t\t\t\tchmodSync(this.authPath, 0o600);\n\t\t\t}\n\t\t\tthrowIfCompromised();\n\t\t\treturn result;\n\t\t} finally {\n\t\t\tif (release) {\n\t\t\t\ttry {\n\t\t\t\t\tawait release();\n\t\t\t\t} catch {}\n\t\t\t}\n\t\t}\n\t}\n}\nexport class InMemoryAuthStorageBackend implements AuthStorageBackend {\n\tprivate value: string | undefined;\n\twithLock<T>(fn: (current: string | undefined) => LockResult<T>): T {\n\t\tconst { result, next } = fn(this.value);\n\t\tif (next !== undefined) {\n\t\t\tthis.value = next;\n\t\t}\n\t\treturn result;\n\t}\n\tasync withLockAsync<T>(fn: (current: string | undefined) => Promise<LockResult<T>>): Promise<T> {\n\t\tconst { result, next } = await fn(this.value);\n\t\tif (next !== undefined) {\n\t\t\tthis.value = next;\n\t\t}\n\t\treturn result;\n\t}\n}\nexport class AuthStorage {\n\tprivate data: AuthStorageData = {};\n\tprivate runtimeOverrides: Map<string, string> = new Map();\n\tprivate fallbackResolver?: (provider: string) => string | undefined;\n\tprivate loadError: Error | null = null;\n\tprivate errors: Error[] = [];\n\tprivate constructor(private storage: AuthStorageBackend) {\n\t\tthis.reload();\n\t}\n\tstatic create(authPath?: string): AuthStorage {\n\t\treturn new AuthStorage(new FileAuthStorageBackend(authPath ?? join(getAgentDir(), \"auth.json\")));\n\t}\n\tstatic fromStorage(storage: AuthStorageBackend): AuthStorage {\n\t\treturn new AuthStorage(storage);\n\t}\n\tstatic inMemory(data: AuthStorageData = {}): AuthStorage {\n\t\tconst storage = new InMemoryAuthStorageBackend();\n\t\tstorage.withLock(() => ({ result: undefined, next: JSON.stringify(data, null, 2) }));\n\t\treturn AuthStorage.fromStorage(storage);\n\t}\n\tsetRuntimeApiKey(provider: string, apiKey: string): void {\n\t\tthis.runtimeOverrides.set(provider, apiKey);\n\t}\n\tremoveRuntimeApiKey(provider: string): void {\n\t\tthis.runtimeOverrides.delete(provider);\n\t}\n\tsetFallbackResolver(resolver: (provider: string) => string | undefined): void {\n\t\tthis.fallbackResolver = resolver;\n\t}\n\tprivate recordError(error: unknown): void {\n\t\tconst normalizedError = error instanceof Error ? error : new Error(String(error));\n\t\tthis.errors.push(normalizedError);\n\t}\n\tprivate parseStorageData(content: string | undefined): AuthStorageData {\n\t\tif (!content) {\n\t\t\treturn {};\n\t\t}\n\t\treturn JSON.parse(content) as AuthStorageData;\n\t}\n\treload(): void {\n\t\tlet content: string | undefined;\n\t\ttry {\n\t\t\tthis.storage.withLock((current) => {\n\t\t\t\tcontent = current;\n\t\t\t\treturn { result: undefined };\n\t\t\t});\n\t\t\tthis.data = this.parseStorageData(content);\n\t\t\tthis.loadError = null;\n\t\t} catch (error) {\n\t\t\tthis.loadError = error as Error;\n\t\t\tthis.recordError(error);\n\t\t}\n\t}\n\tprivate persistProviderChange(provider: string, credential: AuthCredential | undefined): void {\n\t\tif (this.loadError) {\n\t\t\treturn;\n\t\t}\n\t\ttry {\n\t\t\tthis.storage.withLock((current) => {\n\t\t\t\tconst currentData = this.parseStorageData(current);\n\t\t\t\tconst merged: AuthStorageData = { ...currentData };\n\t\t\t\tif (credential) {\n\t\t\t\t\tmerged[provider] = credential;\n\t\t\t\t} else {\n\t\t\t\t\tdelete merged[provider];\n\t\t\t\t}\n\t\t\t\treturn { result: undefined, next: JSON.stringify(merged, null, 2) };\n\t\t\t});\n\t\t} catch (error) {\n\t\t\tthis.recordError(error);\n\t\t}\n\t}\n\tget(provider: string): AuthCredential | undefined {\n\t\treturn this.data[provider] ?? undefined;\n\t}\n\tset(provider: string, credential: AuthCredential): void {\n\t\tthis.data[provider] = credential;\n\t\tthis.persistProviderChange(provider, credential);\n\t}\n\tremove(provider: string): void {\n\t\tdelete this.data[provider];\n\t\tthis.persistProviderChange(provider, undefined);\n\t}\n\tlist(): string[] {\n\t\treturn Object.keys(this.data);\n\t}\n\thas(provider: string): boolean {\n\t\treturn provider in this.data;\n\t}\n\thasAuth(provider: string): boolean {\n\t\tif (this.runtimeOverrides.has(provider)) return true;\n\t\tif (this.data[provider]) return true;\n\t\tif (getEnvApiKey(provider)) return true;\n\t\tif (this.fallbackResolver?.(provider)) return true;\n\t\treturn false;\n\t}\n\tgetAll(): AuthStorageData {\n\t\treturn { ...this.data };\n\t}\n\tdrainErrors(): Error[] {\n\t\tconst drained = [...this.errors];\n\t\tthis.errors = [];\n\t\treturn drained;\n\t}\n\tasync login(providerId: OAuthProviderId, callbacks: OAuthLoginCallbacks): Promise<void> {\n\t\tconst provider = getOAuthProvider(providerId);\n\t\tif (!provider) {\n\t\t\tthrow new Error(`Unknown OAuth provider: ${providerId}`);\n\t\t}\n\t\tconst credentials = await provider.login(callbacks);\n\t\tthis.set(providerId, { type: \"oauth\", ...credentials });\n\t}\n\tlogout(provider: string): void {\n\t\tthis.remove(provider);\n\t}\n\tprivate async refreshOAuthTokenWithLock(\n\t\tproviderId: OAuthProviderId,\n\t): Promise<{ apiKey: string; newCredentials: OAuthCredentials } | null> {\n\t\tconst provider = getOAuthProvider(providerId);\n\t\tif (!provider) {\n\t\t\treturn null;\n\t\t}\n\t\tconst result = await this.storage.withLockAsync(async (current) => {\n\t\t\tconst currentData = this.parseStorageData(current);\n\t\t\tthis.data = currentData;\n\t\t\tthis.loadError = null;\n\t\t\tconst cred = currentData[providerId];\n\t\t\tif (cred?.type !== \"oauth\") {\n\t\t\t\treturn { result: null };\n\t\t\t}\n\t\t\tif (Date.now() < cred.expires) {\n\t\t\t\treturn { result: { apiKey: provider.getApiKey(cred), newCredentials: cred } };\n\t\t\t}\n\t\t\tconst oauthCreds: Record<string, OAuthCredentials> = {};\n\t\t\tfor (const [key, value] of Object.entries(currentData)) {\n\t\t\t\tif (value.type === \"oauth\") {\n\t\t\t\t\toauthCreds[key] = value;\n\t\t\t\t}\n\t\t\t}\n\t\t\tconst refreshed = await getOAuthApiKey(providerId, oauthCreds);\n\t\t\tif (!refreshed) {\n\t\t\t\treturn { result: null };\n\t\t\t}\n\t\t\tconst merged: AuthStorageData = {\n\t\t\t\t...currentData,\n\t\t\t\t[providerId]: { type: \"oauth\", ...refreshed.newCredentials },\n\t\t\t};\n\t\t\tthis.data = merged;\n\t\t\tthis.loadError = null;\n\t\t\treturn { result: refreshed, next: JSON.stringify(merged, null, 2) };\n\t\t});\n\t\treturn result;\n\t}\n\tasync getApiKey(providerId: string): Promise<string | undefined> {\n\t\tconst runtimeKey = this.runtimeOverrides.get(providerId);\n\t\tif (runtimeKey) {\n\t\t\treturn runtimeKey;\n\t\t}\n\t\tconst cred = this.data[providerId];\n\t\tif (cred?.type === \"api_key\") {\n\t\t\treturn resolveConfigValue(cred.key);\n\t\t}\n\t\tif (cred?.type === \"oauth\") {\n\t\t\tconst provider = getOAuthProvider(providerId);\n\t\t\tif (!provider) {\n\t\t\t\treturn undefined;\n\t\t\t}\n\t\t\tconst needsRefresh = Date.now() >= cred.expires;\n\t\t\tif (needsRefresh) {\n\t\t\t\ttry {\n\t\t\t\t\tconst result = await this.refreshOAuthTokenWithLock(providerId);\n\t\t\t\t\tif (result) {\n\t\t\t\t\t\treturn result.apiKey;\n\t\t\t\t\t}\n\t\t\t\t} catch (error) {\n\t\t\t\t\tthis.recordError(error);\n\t\t\t\t\tthis.reload();\n\t\t\t\t\tconst updatedCred = this.data[providerId];\n\t\t\t\t\tif (updatedCred?.type === \"oauth\" && Date.now() < updatedCred.expires) {\n\t\t\t\t\t\treturn provider.getApiKey(updatedCred);\n\t\t\t\t\t}\n\t\t\t\t\treturn undefined;\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\treturn provider.getApiKey(cred);\n\t\t\t}\n\t\t}\n\t\tconst envKey = getEnvApiKey(providerId);\n\t\tif (envKey) return envKey;\n\t\treturn this.fallbackResolver?.(providerId) ?? undefined;\n\t}\n\tgetOAuthProviders() {\n\t\treturn getOAuthProviders();\n\t}\n}\n"]}
@@ -0,0 +1,336 @@
1
+ import { getEnvApiKey, } from "@mariozechner/pi-ai";
2
+ import { getOAuthApiKey, getOAuthProvider, getOAuthProviders } from "@mariozechner/pi-ai/oauth";
3
+ import { chmodSync, existsSync, mkdirSync, readFileSync, writeFileSync } from "fs";
4
+ import { dirname, join } from "path";
5
+ import lockfile from "proper-lockfile";
6
+ import { getAgentDir } from "../config.js";
7
+ import { resolveConfigValue } from "./resolve-config-value.js";
8
+ export class FileAuthStorageBackend {
9
+ authPath;
10
+ constructor(authPath = join(getAgentDir(), "auth.json")) {
11
+ this.authPath = authPath;
12
+ }
13
+ ensureParentDir() {
14
+ const dir = dirname(this.authPath);
15
+ if (!existsSync(dir)) {
16
+ mkdirSync(dir, { recursive: true, mode: 0o700 });
17
+ }
18
+ }
19
+ ensureFileExists() {
20
+ if (!existsSync(this.authPath)) {
21
+ writeFileSync(this.authPath, "{}", "utf-8");
22
+ chmodSync(this.authPath, 0o600);
23
+ }
24
+ }
25
+ acquireLockSyncWithRetry(path) {
26
+ const maxAttempts = 10;
27
+ const delayMs = 20;
28
+ let lastError;
29
+ for (let attempt = 1; attempt <= maxAttempts; attempt++) {
30
+ try {
31
+ return lockfile.lockSync(path, { realpath: false });
32
+ }
33
+ catch (error) {
34
+ const code = typeof error === "object" && error !== null && "code" in error
35
+ ? String(error.code)
36
+ : undefined;
37
+ if (code !== "ELOCKED" || attempt === maxAttempts) {
38
+ throw error;
39
+ }
40
+ lastError = error;
41
+ const start = Date.now();
42
+ while (Date.now() - start < delayMs) { }
43
+ }
44
+ }
45
+ throw lastError ?? new Error("Failed to acquire auth storage lock");
46
+ }
47
+ withLock(fn) {
48
+ this.ensureParentDir();
49
+ this.ensureFileExists();
50
+ let release;
51
+ try {
52
+ release = this.acquireLockSyncWithRetry(this.authPath);
53
+ const current = existsSync(this.authPath) ? readFileSync(this.authPath, "utf-8") : undefined;
54
+ const { result, next } = fn(current);
55
+ if (next !== undefined) {
56
+ writeFileSync(this.authPath, next, "utf-8");
57
+ chmodSync(this.authPath, 0o600);
58
+ }
59
+ return result;
60
+ }
61
+ finally {
62
+ if (release) {
63
+ release();
64
+ }
65
+ }
66
+ }
67
+ async withLockAsync(fn) {
68
+ this.ensureParentDir();
69
+ this.ensureFileExists();
70
+ let release;
71
+ let lockCompromised = false;
72
+ let lockCompromisedError;
73
+ const throwIfCompromised = () => {
74
+ if (lockCompromised) {
75
+ throw lockCompromisedError ?? new Error("Auth storage lock was compromised");
76
+ }
77
+ };
78
+ try {
79
+ release = await lockfile.lock(this.authPath, {
80
+ retries: {
81
+ retries: 10,
82
+ factor: 2,
83
+ minTimeout: 100,
84
+ maxTimeout: 10000,
85
+ randomize: true,
86
+ },
87
+ stale: 30000,
88
+ onCompromised: (err) => {
89
+ lockCompromised = true;
90
+ lockCompromisedError = err;
91
+ },
92
+ });
93
+ throwIfCompromised();
94
+ const current = existsSync(this.authPath) ? readFileSync(this.authPath, "utf-8") : undefined;
95
+ const { result, next } = await fn(current);
96
+ throwIfCompromised();
97
+ if (next !== undefined) {
98
+ writeFileSync(this.authPath, next, "utf-8");
99
+ chmodSync(this.authPath, 0o600);
100
+ }
101
+ throwIfCompromised();
102
+ return result;
103
+ }
104
+ finally {
105
+ if (release) {
106
+ try {
107
+ await release();
108
+ }
109
+ catch { }
110
+ }
111
+ }
112
+ }
113
+ }
114
+ export class InMemoryAuthStorageBackend {
115
+ value;
116
+ withLock(fn) {
117
+ const { result, next } = fn(this.value);
118
+ if (next !== undefined) {
119
+ this.value = next;
120
+ }
121
+ return result;
122
+ }
123
+ async withLockAsync(fn) {
124
+ const { result, next } = await fn(this.value);
125
+ if (next !== undefined) {
126
+ this.value = next;
127
+ }
128
+ return result;
129
+ }
130
+ }
131
+ export class AuthStorage {
132
+ storage;
133
+ data = {};
134
+ runtimeOverrides = new Map();
135
+ fallbackResolver;
136
+ loadError = null;
137
+ errors = [];
138
+ constructor(storage) {
139
+ this.storage = storage;
140
+ this.reload();
141
+ }
142
+ static create(authPath) {
143
+ return new AuthStorage(new FileAuthStorageBackend(authPath ?? join(getAgentDir(), "auth.json")));
144
+ }
145
+ static fromStorage(storage) {
146
+ return new AuthStorage(storage);
147
+ }
148
+ static inMemory(data = {}) {
149
+ const storage = new InMemoryAuthStorageBackend();
150
+ storage.withLock(() => ({ result: undefined, next: JSON.stringify(data, null, 2) }));
151
+ return AuthStorage.fromStorage(storage);
152
+ }
153
+ setRuntimeApiKey(provider, apiKey) {
154
+ this.runtimeOverrides.set(provider, apiKey);
155
+ }
156
+ removeRuntimeApiKey(provider) {
157
+ this.runtimeOverrides.delete(provider);
158
+ }
159
+ setFallbackResolver(resolver) {
160
+ this.fallbackResolver = resolver;
161
+ }
162
+ recordError(error) {
163
+ const normalizedError = error instanceof Error ? error : new Error(String(error));
164
+ this.errors.push(normalizedError);
165
+ }
166
+ parseStorageData(content) {
167
+ if (!content) {
168
+ return {};
169
+ }
170
+ return JSON.parse(content);
171
+ }
172
+ reload() {
173
+ let content;
174
+ try {
175
+ this.storage.withLock((current) => {
176
+ content = current;
177
+ return { result: undefined };
178
+ });
179
+ this.data = this.parseStorageData(content);
180
+ this.loadError = null;
181
+ }
182
+ catch (error) {
183
+ this.loadError = error;
184
+ this.recordError(error);
185
+ }
186
+ }
187
+ persistProviderChange(provider, credential) {
188
+ if (this.loadError) {
189
+ return;
190
+ }
191
+ try {
192
+ this.storage.withLock((current) => {
193
+ const currentData = this.parseStorageData(current);
194
+ const merged = { ...currentData };
195
+ if (credential) {
196
+ merged[provider] = credential;
197
+ }
198
+ else {
199
+ delete merged[provider];
200
+ }
201
+ return { result: undefined, next: JSON.stringify(merged, null, 2) };
202
+ });
203
+ }
204
+ catch (error) {
205
+ this.recordError(error);
206
+ }
207
+ }
208
+ get(provider) {
209
+ return this.data[provider] ?? undefined;
210
+ }
211
+ set(provider, credential) {
212
+ this.data[provider] = credential;
213
+ this.persistProviderChange(provider, credential);
214
+ }
215
+ remove(provider) {
216
+ delete this.data[provider];
217
+ this.persistProviderChange(provider, undefined);
218
+ }
219
+ list() {
220
+ return Object.keys(this.data);
221
+ }
222
+ has(provider) {
223
+ return provider in this.data;
224
+ }
225
+ hasAuth(provider) {
226
+ if (this.runtimeOverrides.has(provider))
227
+ return true;
228
+ if (this.data[provider])
229
+ return true;
230
+ if (getEnvApiKey(provider))
231
+ return true;
232
+ if (this.fallbackResolver?.(provider))
233
+ return true;
234
+ return false;
235
+ }
236
+ getAll() {
237
+ return { ...this.data };
238
+ }
239
+ drainErrors() {
240
+ const drained = [...this.errors];
241
+ this.errors = [];
242
+ return drained;
243
+ }
244
+ async login(providerId, callbacks) {
245
+ const provider = getOAuthProvider(providerId);
246
+ if (!provider) {
247
+ throw new Error(`Unknown OAuth provider: ${providerId}`);
248
+ }
249
+ const credentials = await provider.login(callbacks);
250
+ this.set(providerId, { type: "oauth", ...credentials });
251
+ }
252
+ logout(provider) {
253
+ this.remove(provider);
254
+ }
255
+ async refreshOAuthTokenWithLock(providerId) {
256
+ const provider = getOAuthProvider(providerId);
257
+ if (!provider) {
258
+ return null;
259
+ }
260
+ const result = await this.storage.withLockAsync(async (current) => {
261
+ const currentData = this.parseStorageData(current);
262
+ this.data = currentData;
263
+ this.loadError = null;
264
+ const cred = currentData[providerId];
265
+ if (cred?.type !== "oauth") {
266
+ return { result: null };
267
+ }
268
+ if (Date.now() < cred.expires) {
269
+ return { result: { apiKey: provider.getApiKey(cred), newCredentials: cred } };
270
+ }
271
+ const oauthCreds = {};
272
+ for (const [key, value] of Object.entries(currentData)) {
273
+ if (value.type === "oauth") {
274
+ oauthCreds[key] = value;
275
+ }
276
+ }
277
+ const refreshed = await getOAuthApiKey(providerId, oauthCreds);
278
+ if (!refreshed) {
279
+ return { result: null };
280
+ }
281
+ const merged = {
282
+ ...currentData,
283
+ [providerId]: { type: "oauth", ...refreshed.newCredentials },
284
+ };
285
+ this.data = merged;
286
+ this.loadError = null;
287
+ return { result: refreshed, next: JSON.stringify(merged, null, 2) };
288
+ });
289
+ return result;
290
+ }
291
+ async getApiKey(providerId) {
292
+ const runtimeKey = this.runtimeOverrides.get(providerId);
293
+ if (runtimeKey) {
294
+ return runtimeKey;
295
+ }
296
+ const cred = this.data[providerId];
297
+ if (cred?.type === "api_key") {
298
+ return resolveConfigValue(cred.key);
299
+ }
300
+ if (cred?.type === "oauth") {
301
+ const provider = getOAuthProvider(providerId);
302
+ if (!provider) {
303
+ return undefined;
304
+ }
305
+ const needsRefresh = Date.now() >= cred.expires;
306
+ if (needsRefresh) {
307
+ try {
308
+ const result = await this.refreshOAuthTokenWithLock(providerId);
309
+ if (result) {
310
+ return result.apiKey;
311
+ }
312
+ }
313
+ catch (error) {
314
+ this.recordError(error);
315
+ this.reload();
316
+ const updatedCred = this.data[providerId];
317
+ if (updatedCred?.type === "oauth" && Date.now() < updatedCred.expires) {
318
+ return provider.getApiKey(updatedCred);
319
+ }
320
+ return undefined;
321
+ }
322
+ }
323
+ else {
324
+ return provider.getApiKey(cred);
325
+ }
326
+ }
327
+ const envKey = getEnvApiKey(providerId);
328
+ if (envKey)
329
+ return envKey;
330
+ return this.fallbackResolver?.(providerId) ?? undefined;
331
+ }
332
+ getOAuthProviders() {
333
+ return getOAuthProviders();
334
+ }
335
+ }
336
+ //# sourceMappingURL=auth-storage.js.map