shortcutxl 0.2.0

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 (423) hide show
  1. package/README.md +59 -0
  2. package/agent-docs/README.md +397 -0
  3. package/agent-docs/docs/compaction.md +390 -0
  4. package/agent-docs/docs/custom-provider.md +580 -0
  5. package/agent-docs/docs/development.md +69 -0
  6. package/agent-docs/docs/extensions.md +1971 -0
  7. package/agent-docs/docs/json.md +79 -0
  8. package/agent-docs/docs/keybindings.md +174 -0
  9. package/agent-docs/docs/models.md +293 -0
  10. package/agent-docs/docs/packages.md +209 -0
  11. package/agent-docs/docs/prompt-templates.md +67 -0
  12. package/agent-docs/docs/providers.md +186 -0
  13. package/agent-docs/docs/rpc.md +1317 -0
  14. package/agent-docs/docs/sdk.md +962 -0
  15. package/agent-docs/docs/session.md +412 -0
  16. package/agent-docs/docs/settings.md +223 -0
  17. package/agent-docs/docs/shell-aliases.md +13 -0
  18. package/agent-docs/docs/skills.md +231 -0
  19. package/agent-docs/docs/terminal-setup.md +70 -0
  20. package/agent-docs/docs/termux.md +127 -0
  21. package/agent-docs/docs/themes.md +295 -0
  22. package/agent-docs/docs/tree.md +219 -0
  23. package/agent-docs/docs/tui.md +887 -0
  24. package/agent-docs/docs/windows.md +17 -0
  25. package/agent-docs/examples/README.md +25 -0
  26. package/agent-docs/examples/extensions/README.md +205 -0
  27. package/agent-docs/examples/extensions/antigravity-image-gen.ts +447 -0
  28. package/agent-docs/examples/extensions/auto-commit-on-exit.ts +49 -0
  29. package/agent-docs/examples/extensions/bash-spawn-hook.ts +30 -0
  30. package/agent-docs/examples/extensions/bookmark.ts +50 -0
  31. package/agent-docs/examples/extensions/built-in-tool-renderer.ts +256 -0
  32. package/agent-docs/examples/extensions/claude-rules.ts +86 -0
  33. package/agent-docs/examples/extensions/commands.ts +75 -0
  34. package/agent-docs/examples/extensions/confirm-destructive.ts +59 -0
  35. package/agent-docs/examples/extensions/custom-compaction.ts +126 -0
  36. package/agent-docs/examples/extensions/custom-footer.ts +63 -0
  37. package/agent-docs/examples/extensions/custom-header.ts +73 -0
  38. package/agent-docs/examples/extensions/custom-provider-anthropic/index.ts +660 -0
  39. package/agent-docs/examples/extensions/custom-provider-anthropic/package-lock.json +24 -0
  40. package/agent-docs/examples/extensions/custom-provider-anthropic/package.json +19 -0
  41. package/agent-docs/examples/extensions/custom-provider-gitlab-duo/index.ts +362 -0
  42. package/agent-docs/examples/extensions/custom-provider-gitlab-duo/package.json +16 -0
  43. package/agent-docs/examples/extensions/custom-provider-gitlab-duo/test.ts +88 -0
  44. package/agent-docs/examples/extensions/custom-provider-qwen-cli/index.ts +349 -0
  45. package/agent-docs/examples/extensions/custom-provider-qwen-cli/package.json +16 -0
  46. package/agent-docs/examples/extensions/dirty-repo-guard.ts +56 -0
  47. package/agent-docs/examples/extensions/doom-overlay/README.md +46 -0
  48. package/agent-docs/examples/extensions/doom-overlay/doom/build.sh +152 -0
  49. package/agent-docs/examples/extensions/doom-overlay/doom/doomgeneric_pi.c +72 -0
  50. package/agent-docs/examples/extensions/doom-overlay/doom-component.ts +133 -0
  51. package/agent-docs/examples/extensions/doom-overlay/doom-engine.ts +186 -0
  52. package/agent-docs/examples/extensions/doom-overlay/doom-keys.ts +108 -0
  53. package/agent-docs/examples/extensions/doom-overlay/index.ts +74 -0
  54. package/agent-docs/examples/extensions/doom-overlay/wad-finder.ts +51 -0
  55. package/agent-docs/examples/extensions/dynamic-resources/SKILL.md +8 -0
  56. package/agent-docs/examples/extensions/dynamic-resources/dynamic.json +79 -0
  57. package/agent-docs/examples/extensions/dynamic-resources/dynamic.md +5 -0
  58. package/agent-docs/examples/extensions/dynamic-resources/index.ts +15 -0
  59. package/agent-docs/examples/extensions/dynamic-tools.ts +77 -0
  60. package/agent-docs/examples/extensions/event-bus.ts +43 -0
  61. package/agent-docs/examples/extensions/file-trigger.ts +41 -0
  62. package/agent-docs/examples/extensions/git-checkpoint.ts +53 -0
  63. package/agent-docs/examples/extensions/handoff.ts +155 -0
  64. package/agent-docs/examples/extensions/hello.ts +25 -0
  65. package/agent-docs/examples/extensions/inline-bash.ts +94 -0
  66. package/agent-docs/examples/extensions/input-transform.ts +43 -0
  67. package/agent-docs/examples/extensions/interactive-shell.ts +209 -0
  68. package/agent-docs/examples/extensions/mac-system-theme.ts +47 -0
  69. package/agent-docs/examples/extensions/message-renderer.ts +59 -0
  70. package/agent-docs/examples/extensions/minimal-mode.ts +430 -0
  71. package/agent-docs/examples/extensions/modal-editor.ts +90 -0
  72. package/agent-docs/examples/extensions/model-status.ts +31 -0
  73. package/agent-docs/examples/extensions/notify.ts +55 -0
  74. package/agent-docs/examples/extensions/overlay-qa-tests.ts +936 -0
  75. package/agent-docs/examples/extensions/overlay-test.ts +159 -0
  76. package/agent-docs/examples/extensions/permission-gate.ts +37 -0
  77. package/agent-docs/examples/extensions/pirate.ts +47 -0
  78. package/agent-docs/examples/extensions/plan-mode/README.md +65 -0
  79. package/agent-docs/examples/extensions/plan-mode/index.ts +363 -0
  80. package/agent-docs/examples/extensions/plan-mode/utils.ts +173 -0
  81. package/agent-docs/examples/extensions/preset.ts +418 -0
  82. package/agent-docs/examples/extensions/protected-paths.ts +30 -0
  83. package/agent-docs/examples/extensions/qna.ts +122 -0
  84. package/agent-docs/examples/extensions/question.ts +278 -0
  85. package/agent-docs/examples/extensions/questionnaire.ts +440 -0
  86. package/agent-docs/examples/extensions/rainbow-editor.ts +90 -0
  87. package/agent-docs/examples/extensions/reload-runtime.ts +37 -0
  88. package/agent-docs/examples/extensions/rpc-demo.ts +124 -0
  89. package/agent-docs/examples/extensions/sandbox/index.ts +324 -0
  90. package/agent-docs/examples/extensions/sandbox/package-lock.json +92 -0
  91. package/agent-docs/examples/extensions/sandbox/package.json +19 -0
  92. package/agent-docs/examples/extensions/send-user-message.ts +97 -0
  93. package/agent-docs/examples/extensions/session-name.ts +27 -0
  94. package/agent-docs/examples/extensions/shutdown-command.ts +69 -0
  95. package/agent-docs/examples/extensions/snake.ts +343 -0
  96. package/agent-docs/examples/extensions/space-invaders.ts +566 -0
  97. package/agent-docs/examples/extensions/ssh.ts +233 -0
  98. package/agent-docs/examples/extensions/status-line.ts +40 -0
  99. package/agent-docs/examples/extensions/subagent/README.md +172 -0
  100. package/agent-docs/examples/extensions/subagent/agents/planner.md +37 -0
  101. package/agent-docs/examples/extensions/subagent/agents/reviewer.md +35 -0
  102. package/agent-docs/examples/extensions/subagent/agents/scout.md +50 -0
  103. package/agent-docs/examples/extensions/subagent/agents/worker.md +24 -0
  104. package/agent-docs/examples/extensions/subagent/agents.ts +130 -0
  105. package/agent-docs/examples/extensions/subagent/index.ts +1068 -0
  106. package/agent-docs/examples/extensions/subagent/prompts/implement-and-review.md +10 -0
  107. package/agent-docs/examples/extensions/subagent/prompts/implement.md +10 -0
  108. package/agent-docs/examples/extensions/subagent/prompts/scout-and-plan.md +9 -0
  109. package/agent-docs/examples/extensions/summarize.ts +206 -0
  110. package/agent-docs/examples/extensions/system-prompt-header.ts +17 -0
  111. package/agent-docs/examples/extensions/timed-confirm.ts +72 -0
  112. package/agent-docs/examples/extensions/titlebar-spinner.ts +58 -0
  113. package/agent-docs/examples/extensions/todo.ts +314 -0
  114. package/agent-docs/examples/extensions/tool-override.ts +146 -0
  115. package/agent-docs/examples/extensions/tools.ts +145 -0
  116. package/agent-docs/examples/extensions/trigger-compact.ts +40 -0
  117. package/agent-docs/examples/extensions/truncated-tool.ts +194 -0
  118. package/agent-docs/examples/extensions/widget-placement.ts +17 -0
  119. package/agent-docs/examples/extensions/with-deps/index.ts +37 -0
  120. package/agent-docs/examples/extensions/with-deps/package-lock.json +31 -0
  121. package/agent-docs/examples/extensions/with-deps/package.json +22 -0
  122. package/agent-docs/examples/rpc-extension-ui.ts +654 -0
  123. package/agent-docs/examples/sdk/01-minimal.ts +22 -0
  124. package/agent-docs/examples/sdk/02-custom-model.ts +48 -0
  125. package/agent-docs/examples/sdk/03-custom-prompt.ts +55 -0
  126. package/agent-docs/examples/sdk/04-skills.ts +53 -0
  127. package/agent-docs/examples/sdk/05-tools.ts +56 -0
  128. package/agent-docs/examples/sdk/06-extensions.ts +88 -0
  129. package/agent-docs/examples/sdk/07-context-files.ts +40 -0
  130. package/agent-docs/examples/sdk/08-prompt-templates.ts +47 -0
  131. package/agent-docs/examples/sdk/09-api-keys-and-oauth.ts +48 -0
  132. package/agent-docs/examples/sdk/10-settings.ts +54 -0
  133. package/agent-docs/examples/sdk/11-sessions.ts +48 -0
  134. package/agent-docs/examples/sdk/12-full-control.ts +82 -0
  135. package/agent-docs/examples/sdk/README.md +144 -0
  136. package/agent-docs/xll-skill.md +61 -0
  137. package/agent-docs/xll-spec.md +110 -0
  138. package/dist/cli/args.js +290 -0
  139. package/dist/cli/config-selector.js +31 -0
  140. package/dist/cli/file-processor.js +79 -0
  141. package/dist/cli/list-models.js +92 -0
  142. package/dist/cli/package-commands.js +210 -0
  143. package/dist/cli/report-settings-errors.js +11 -0
  144. package/dist/cli/session-picker.js +34 -0
  145. package/dist/cli.js +19 -0
  146. package/dist/config.js +288 -0
  147. package/dist/core/abort.js +15 -0
  148. package/dist/core/agent-loop.js +352 -0
  149. package/dist/core/agent-session.js +2019 -0
  150. package/dist/core/agent.js +410 -0
  151. package/dist/core/auth-storage.js +456 -0
  152. package/dist/core/bash-executor.js +222 -0
  153. package/dist/core/compaction/branch-summarization.js +242 -0
  154. package/dist/core/compaction/compaction.js +610 -0
  155. package/dist/core/compaction/index.js +7 -0
  156. package/dist/core/compaction/utils.js +139 -0
  157. package/dist/core/defaults.js +6 -0
  158. package/dist/core/diagnostics.js +2 -0
  159. package/dist/core/event-bus.js +25 -0
  160. package/dist/core/exec.js +71 -0
  161. package/dist/core/export-html/ansi-to-html.js +256 -0
  162. package/dist/core/export-html/index.js +238 -0
  163. package/dist/core/export-html/session-view-model.js +342 -0
  164. package/dist/core/export-html/template.css +1110 -0
  165. package/dist/core/export-html/template.html +76 -0
  166. package/dist/core/export-html/template.js +1990 -0
  167. package/dist/core/export-html/tool-renderer.js +63 -0
  168. package/dist/core/export-html/vendor/highlight.min.js +7725 -0
  169. package/dist/core/export-html/vendor/marked.min.js +1803 -0
  170. package/dist/core/extensions/index.js +9 -0
  171. package/dist/core/extensions/loader.js +422 -0
  172. package/dist/core/extensions/runner.js +651 -0
  173. package/dist/core/extensions/types.js +35 -0
  174. package/dist/core/extensions/wrapper.js +102 -0
  175. package/dist/core/footer-data-provider.js +162 -0
  176. package/dist/core/index.js +9 -0
  177. package/dist/core/keybindings.js +153 -0
  178. package/dist/core/messages.js +133 -0
  179. package/dist/core/model-registry.js +539 -0
  180. package/dist/core/model-resolver.js +370 -0
  181. package/dist/core/package-manager.js +1485 -0
  182. package/dist/core/prompt-templates.js +253 -0
  183. package/dist/core/resolve-config-value.js +59 -0
  184. package/dist/core/resource-loader.js +700 -0
  185. package/dist/core/sdk.js +197 -0
  186. package/dist/core/session-bash.js +99 -0
  187. package/dist/core/session-compaction.js +165 -0
  188. package/dist/core/session-manager.js +1153 -0
  189. package/dist/core/session-models.js +99 -0
  190. package/dist/core/session-retry.js +155 -0
  191. package/dist/core/settings-manager.js +572 -0
  192. package/dist/core/skills.js +382 -0
  193. package/dist/core/slash-commands.js +31 -0
  194. package/dist/core/system-prompt.js +161 -0
  195. package/dist/core/theme.js +770 -0
  196. package/dist/core/timings.js +26 -0
  197. package/dist/core/tools/bash.js +258 -0
  198. package/dist/core/tools/edit-diff.js +245 -0
  199. package/dist/core/tools/edit.js +148 -0
  200. package/dist/core/tools/find.js +208 -0
  201. package/dist/core/tools/grep.js +246 -0
  202. package/dist/core/tools/index.js +67 -0
  203. package/dist/core/tools/ls.js +123 -0
  204. package/dist/core/tools/path-utils.js +81 -0
  205. package/dist/core/tools/read.js +160 -0
  206. package/dist/core/tools/truncate.js +70 -0
  207. package/dist/core/tools/write.js +82 -0
  208. package/dist/custom/agents/action.js +13 -0
  209. package/dist/custom/agents/document-reader.js +70 -0
  210. package/dist/custom/agents/general.js +26 -0
  211. package/dist/custom/agents/index.js +49 -0
  212. package/dist/custom/agents/installation.js +13 -0
  213. package/dist/custom/agents/types.js +7 -0
  214. package/dist/custom/auth/refresh-timer.js +33 -0
  215. package/dist/custom/auth/shortcut-oauth.js +145 -0
  216. package/dist/custom/constants.js +21 -0
  217. package/dist/custom/context/workbook-summary.js +73 -0
  218. package/dist/custom/credits/shortcut-credits.js +29 -0
  219. package/dist/custom/cron/cron-daemon-entry.js +18 -0
  220. package/dist/custom/cron/daemon-ipc.js +131 -0
  221. package/dist/custom/cron/daemon.js +224 -0
  222. package/dist/custom/cron/jobs.js +226 -0
  223. package/dist/custom/cron/run-log.js +51 -0
  224. package/dist/custom/cron/schedule.js +72 -0
  225. package/dist/custom/cron/status-line.js +98 -0
  226. package/dist/custom/cron/store.js +87 -0
  227. package/dist/custom/cron/types.js +8 -0
  228. package/dist/custom/dev/index.js +59 -0
  229. package/dist/custom/dev/trace-export.js +58 -0
  230. package/dist/custom/ensure-excel.js +63 -0
  231. package/dist/custom/excel-config.js +36 -0
  232. package/dist/custom/preflight.js +422 -0
  233. package/dist/custom/prompts/action.js +100 -0
  234. package/dist/custom/prompts/api.js +66 -0
  235. package/dist/custom/prompts/installation.js +124 -0
  236. package/dist/custom/prompts/shared.js +138 -0
  237. package/dist/custom/providers/llm-usage.js +42 -0
  238. package/dist/custom/providers/message-converter.js +74 -0
  239. package/dist/custom/providers/provider-ids.js +9 -0
  240. package/dist/custom/providers/register-openai-codex-provider.js +27 -0
  241. package/dist/custom/providers/register-shortcut-provider.js +52 -0
  242. package/dist/custom/providers/shortcut-invoke.js +117 -0
  243. package/dist/custom/providers/shortcut-stream.js +252 -0
  244. package/dist/custom/providers/sse-protocol.js +38 -0
  245. package/dist/custom/sync-xll.js +130 -0
  246. package/dist/custom/tools/cron.js +413 -0
  247. package/dist/custom/tools/excel-exec.js +167 -0
  248. package/dist/custom/tools/excel-range.js +50 -0
  249. package/dist/custom/tools/llm-analysis.js +265 -0
  250. package/dist/custom/tools/render-helpers.js +38 -0
  251. package/dist/custom/tools/switch-mode.js +94 -0
  252. package/dist/custom/tools/task/agents.js +6 -0
  253. package/dist/custom/tools/task/index.js +8 -0
  254. package/dist/custom/tools/task/render.js +348 -0
  255. package/dist/custom/tools/task/subprocess.js +320 -0
  256. package/dist/custom/tools/task/task.js +205 -0
  257. package/dist/custom/tools/todo-list.js +195 -0
  258. package/dist/custom/tracing/session-upload.js +93 -0
  259. package/dist/index.js +45 -0
  260. package/dist/main.js +613 -0
  261. package/dist/migrations.js +265 -0
  262. package/dist/modes/index.js +8 -0
  263. package/dist/modes/interactive/components/armin.js +337 -0
  264. package/dist/modes/interactive/components/assistant-message.js +94 -0
  265. package/dist/modes/interactive/components/bash-execution.js +171 -0
  266. package/dist/modes/interactive/components/bordered-loader.js +51 -0
  267. package/dist/modes/interactive/components/branch-summary-message.js +45 -0
  268. package/dist/modes/interactive/components/compaction-summary-message.js +46 -0
  269. package/dist/modes/interactive/components/config-selector.js +488 -0
  270. package/dist/modes/interactive/components/countdown-timer.js +33 -0
  271. package/dist/modes/interactive/components/custom-editor.js +93 -0
  272. package/dist/modes/interactive/components/custom-message.js +81 -0
  273. package/dist/modes/interactive/components/daxnuts.js +140 -0
  274. package/dist/modes/interactive/components/diff.js +133 -0
  275. package/dist/modes/interactive/components/dynamic-border.js +21 -0
  276. package/dist/modes/interactive/components/extension-editor.js +105 -0
  277. package/dist/modes/interactive/components/extension-input.js +61 -0
  278. package/dist/modes/interactive/components/extension-selector.js +78 -0
  279. package/dist/modes/interactive/components/footer.js +309 -0
  280. package/dist/modes/interactive/components/index.js +33 -0
  281. package/dist/modes/interactive/components/keybinding-hints.js +61 -0
  282. package/dist/modes/interactive/components/layout.js +64 -0
  283. package/dist/modes/interactive/components/login-dialog.js +148 -0
  284. package/dist/modes/interactive/components/model-selector.js +237 -0
  285. package/dist/modes/interactive/components/oauth-selector.js +111 -0
  286. package/dist/modes/interactive/components/session-selector-search.js +157 -0
  287. package/dist/modes/interactive/components/session-selector.js +860 -0
  288. package/dist/modes/interactive/components/settings-selector.js +123 -0
  289. package/dist/modes/interactive/components/show-images-selector.js +35 -0
  290. package/dist/modes/interactive/components/skill-invocation-message.js +48 -0
  291. package/dist/modes/interactive/components/theme-selector.js +47 -0
  292. package/dist/modes/interactive/components/thinking-selector.js +47 -0
  293. package/dist/modes/interactive/components/tool-execution.js +789 -0
  294. package/dist/modes/interactive/components/tool-group.js +106 -0
  295. package/dist/modes/interactive/components/tree-selector.js +962 -0
  296. package/dist/modes/interactive/components/user-message-selector.js +115 -0
  297. package/dist/modes/interactive/components/user-message.js +48 -0
  298. package/dist/modes/interactive/components/visual-truncate.js +33 -0
  299. package/dist/modes/interactive/file-attachments.js +135 -0
  300. package/dist/modes/interactive/interactive-mode.js +3775 -0
  301. package/dist/modes/interactive/theme/dark.json +85 -0
  302. package/dist/modes/interactive/theme/light.json +85 -0
  303. package/dist/modes/interactive/theme/theme-schema.json +335 -0
  304. package/dist/modes/interactive/theme/theme.js +177 -0
  305. package/dist/modes/print-mode.js +101 -0
  306. package/dist/modes/rpc/rpc-client.js +387 -0
  307. package/dist/modes/rpc/rpc-mode.js +509 -0
  308. package/dist/modes/rpc/rpc-types.js +8 -0
  309. package/dist/subagent-entry.js +145 -0
  310. package/dist/tool-names.js +34 -0
  311. package/dist/tui/autocomplete.js +596 -0
  312. package/dist/tui/components/box.js +104 -0
  313. package/dist/tui/components/cancellable-loader.js +35 -0
  314. package/dist/tui/components/editor.js +1679 -0
  315. package/dist/tui/components/image.js +69 -0
  316. package/dist/tui/components/input.js +433 -0
  317. package/dist/tui/components/loader.js +49 -0
  318. package/dist/tui/components/markdown.js +629 -0
  319. package/dist/tui/components/select-list.js +152 -0
  320. package/dist/tui/components/settings-list.js +185 -0
  321. package/dist/tui/components/spacer.js +23 -0
  322. package/dist/tui/components/text.js +89 -0
  323. package/dist/tui/components/truncated-text.js +51 -0
  324. package/dist/tui/editor-component.js +2 -0
  325. package/dist/tui/fuzzy.js +107 -0
  326. package/dist/tui/get-east-asian-width/index.js +32 -0
  327. package/dist/tui/get-east-asian-width/lookup.js +404 -0
  328. package/dist/tui/index.js +32 -0
  329. package/dist/tui/keybindings.js +114 -0
  330. package/dist/tui/keys.js +959 -0
  331. package/dist/tui/kill-ring.js +44 -0
  332. package/dist/tui/stdin-buffer.js +317 -0
  333. package/dist/tui/terminal-image.js +288 -0
  334. package/dist/tui/terminal.js +249 -0
  335. package/dist/tui/tui/autocomplete.js +596 -0
  336. package/dist/tui/tui/components/box.js +106 -0
  337. package/dist/tui/tui/components/cancellable-loader.js +35 -0
  338. package/dist/tui/tui/components/editor.js +1679 -0
  339. package/dist/tui/tui/components/image.js +69 -0
  340. package/dist/tui/tui/components/input.js +433 -0
  341. package/dist/tui/tui/components/loader.js +49 -0
  342. package/dist/tui/tui/components/markdown.js +629 -0
  343. package/dist/tui/tui/components/select-list.js +152 -0
  344. package/dist/tui/tui/components/settings-list.js +185 -0
  345. package/dist/tui/tui/components/spacer.js +23 -0
  346. package/dist/tui/tui/components/text.js +91 -0
  347. package/dist/tui/tui/components/truncated-text.js +51 -0
  348. package/dist/tui/tui/editor-component.js +2 -0
  349. package/dist/tui/tui/fuzzy.js +107 -0
  350. package/dist/tui/tui/get-east-asian-width/index.js +32 -0
  351. package/dist/tui/tui/get-east-asian-width/lookup.js +404 -0
  352. package/dist/tui/tui/index.js +32 -0
  353. package/dist/tui/tui/keybindings.js +114 -0
  354. package/dist/tui/tui/keys.js +959 -0
  355. package/dist/tui/tui/kill-ring.js +44 -0
  356. package/dist/tui/tui/stdin-buffer.js +317 -0
  357. package/dist/tui/tui/terminal-image.js +288 -0
  358. package/dist/tui/tui/terminal.js +249 -0
  359. package/dist/tui/tui/tui.js +955 -0
  360. package/dist/tui/tui/undo-stack.js +25 -0
  361. package/dist/tui/tui/utils.js +800 -0
  362. package/dist/tui/tui.js +955 -0
  363. package/dist/tui/undo-stack.js +25 -0
  364. package/dist/tui/utils.js +800 -0
  365. package/dist/utils/changelog.js +87 -0
  366. package/dist/utils/clipboard-image.js +164 -0
  367. package/dist/utils/clipboard-native.js +14 -0
  368. package/dist/utils/clipboard.js +67 -0
  369. package/dist/utils/frontmatter.js +26 -0
  370. package/dist/utils/git.js +166 -0
  371. package/dist/utils/image-convert.js +35 -0
  372. package/dist/utils/image-resize.js +183 -0
  373. package/dist/utils/mime.js +26 -0
  374. package/dist/utils/photon.js +121 -0
  375. package/dist/utils/shell.js +217 -0
  376. package/dist/utils/sleep.js +17 -0
  377. package/dist/utils/tools-manager.js +259 -0
  378. package/package.json +78 -0
  379. package/skills/excel-com-api/SKILL.md +74 -0
  380. package/skills/excel-com-api/excel-type-library.py +27767 -0
  381. package/skills/excel-com-api/office-type-library.py +10867 -0
  382. package/skills/integrations/SKILL.md +138 -0
  383. package/skills/integrations/alphasense.md +457 -0
  384. package/skills/integrations/bloomberg.md +803 -0
  385. package/skills/integrations/calcbench.md +315 -0
  386. package/skills/integrations/capiq.md +848 -0
  387. package/skills/integrations/dynamics-365-finance.md +354 -0
  388. package/skills/integrations/earnings_transcripts.md +387 -0
  389. package/skills/integrations/factset.md +758 -0
  390. package/skills/integrations/ice-fixed-income.md +344 -0
  391. package/skills/integrations/moodys-analytics.md +313 -0
  392. package/skills/integrations/morningstar.md +433 -0
  393. package/skills/integrations/nasdaq-data-link.md +249 -0
  394. package/skills/integrations/pitchbook.md +413 -0
  395. package/skills/integrations/preqin.md +422 -0
  396. package/skills/integrations/quickbooks.md +289 -0
  397. package/skills/integrations/quickfs.md +314 -0
  398. package/skills/integrations/refinitiv.md +473 -0
  399. package/skills/integrations/sage-intacct.md +401 -0
  400. package/skills/integrations/visible-alpha.md +320 -0
  401. package/skills/integrations/xero.md +393 -0
  402. package/skills/integrations/ycharts.md +306 -0
  403. package/skills/pdf-creation/SKILL.md +93 -0
  404. package/skills/pdf-extraction/SKILL.md +32 -0
  405. package/skills/powerpoint-creation/SKILL.md +110 -0
  406. package/skills/sec-edgar/SKILL.md +127 -0
  407. package/skills/sec-edgar/sec_to_pdf.py +109 -0
  408. package/xll/ShortcutXL.xll +0 -0
  409. package/xll/modules/debug_render.py +272 -0
  410. package/xll/modules/gameboy.py +241 -0
  411. package/xll/modules/pong.py +188 -0
  412. package/xll/modules/shortcut_xl/__init__.py +18 -0
  413. package/xll/modules/shortcut_xl/_categorize.py +200 -0
  414. package/xll/modules/shortcut_xl/_com.py +108 -0
  415. package/xll/modules/shortcut_xl/_format.py +252 -0
  416. package/xll/modules/shortcut_xl/_log.py +12 -0
  417. package/xll/modules/shortcut_xl/_managed.py +116 -0
  418. package/xll/modules/shortcut_xl/_registry.py +44 -0
  419. package/xll/modules/shortcut_xl/_threading.py +161 -0
  420. package/xll/modules/shortcut_xl/_tracking.py +283 -0
  421. package/xll/modules/stocks.py +100 -0
  422. package/xll/python3.dll +0 -0
  423. package/xll/python312.dll +0 -0
@@ -0,0 +1,660 @@
1
+ /**
2
+ * Custom Provider Example
3
+ *
4
+ * Demonstrates registering a custom provider with:
5
+ * - Custom API identifier ("custom-anthropic-api")
6
+ * - Custom streamSimple implementation
7
+ * - OAuth support for /login
8
+ * - API key support via environment variable
9
+ * - Two model definitions
10
+ *
11
+ * Usage:
12
+ * # First install dependencies
13
+ * cd packages/coding-agent/examples/extensions/custom-provider && npm install
14
+ *
15
+ * # With OAuth (run /login custom-anthropic first)
16
+ * shortcut -e ./packages/coding-agent/examples/extensions/custom-provider
17
+ *
18
+ * # With API key
19
+ * CUSTOM_ANTHROPIC_API_KEY=sk-ant-... shortcut -e ./packages/coding-agent/examples/extensions/custom-provider
20
+ *
21
+ * Then use /model to select custom-anthropic/claude-sonnet-4-5
22
+ */
23
+
24
+ import Anthropic from '@anthropic-ai/sdk';
25
+ import type {
26
+ ContentBlockParam,
27
+ MessageCreateParamsStreaming
28
+ } from '@anthropic-ai/sdk/resources/messages.js';
29
+ import type { ExtensionAPI } from 'shortcutxl';
30
+ import {
31
+ calculateCost,
32
+ createAssistantMessageEventStream,
33
+ type Api,
34
+ type AssistantMessage,
35
+ type AssistantMessageEventStream,
36
+ type Context,
37
+ type ImageContent,
38
+ type Message,
39
+ type Model,
40
+ type OAuthCredentials,
41
+ type OAuthLoginCallbacks,
42
+ type SimpleStreamOptions,
43
+ type StopReason,
44
+ type TextContent,
45
+ type ThinkingContent,
46
+ type Tool,
47
+ type ToolCall,
48
+ type ToolResultMessage
49
+ } from 'shortcutxl';
50
+
51
+ // =============================================================================
52
+ // OAuth Implementation (copied from packages/ai/src/utils/oauth/anthropic.ts)
53
+ // =============================================================================
54
+
55
+ const decode = (s: string) => atob(s);
56
+ const CLIENT_ID = decode('OWQxYzI1MGEtZTYxYi00NGQ5LTg4ZWQtNTk0NGQxOTYyZjVl');
57
+ const AUTHORIZE_URL = 'https://claude.ai/oauth/authorize';
58
+ const TOKEN_URL = 'https://console.anthropic.com/v1/oauth/token';
59
+ const REDIRECT_URI = 'https://console.anthropic.com/oauth/code/callback';
60
+ const SCOPES = 'org:create_api_key user:profile user:inference';
61
+
62
+ async function generatePKCE(): Promise<{ verifier: string; challenge: string }> {
63
+ const array = new Uint8Array(32);
64
+ crypto.getRandomValues(array);
65
+ const verifier = btoa(String.fromCharCode(...array))
66
+ .replace(/\+/g, '-')
67
+ .replace(/\//g, '_')
68
+ .replace(/=+$/, '');
69
+
70
+ const encoder = new TextEncoder();
71
+ const data = encoder.encode(verifier);
72
+ const hash = await crypto.subtle.digest('SHA-256', data);
73
+ const challenge = btoa(String.fromCharCode(...new Uint8Array(hash)))
74
+ .replace(/\+/g, '-')
75
+ .replace(/\//g, '_')
76
+ .replace(/=+$/, '');
77
+
78
+ return { verifier, challenge };
79
+ }
80
+
81
+ async function loginAnthropic(callbacks: OAuthLoginCallbacks): Promise<OAuthCredentials> {
82
+ const { verifier, challenge } = await generatePKCE();
83
+
84
+ const authParams = new URLSearchParams({
85
+ code: 'true',
86
+ client_id: CLIENT_ID,
87
+ response_type: 'code',
88
+ redirect_uri: REDIRECT_URI,
89
+ scope: SCOPES,
90
+ code_challenge: challenge,
91
+ code_challenge_method: 'S256',
92
+ state: verifier
93
+ });
94
+
95
+ callbacks.onAuth({ url: `${AUTHORIZE_URL}?${authParams.toString()}` });
96
+
97
+ const authCode = await callbacks.onPrompt({ message: 'Paste the authorization code:' });
98
+ const [code, state] = authCode.split('#');
99
+
100
+ const tokenResponse = await fetch(TOKEN_URL, {
101
+ method: 'POST',
102
+ headers: { 'Content-Type': 'application/json' },
103
+ body: JSON.stringify({
104
+ grant_type: 'authorization_code',
105
+ client_id: CLIENT_ID,
106
+ code,
107
+ state,
108
+ redirect_uri: REDIRECT_URI,
109
+ code_verifier: verifier
110
+ })
111
+ });
112
+
113
+ if (!tokenResponse.ok) {
114
+ throw new Error(`Token exchange failed: ${await tokenResponse.text()}`);
115
+ }
116
+
117
+ const data = (await tokenResponse.json()) as {
118
+ access_token: string;
119
+ refresh_token: string;
120
+ expires_in: number;
121
+ };
122
+
123
+ return {
124
+ refresh: data.refresh_token,
125
+ access: data.access_token,
126
+ expires: Date.now() + data.expires_in * 1000 - 5 * 60 * 1000
127
+ };
128
+ }
129
+
130
+ async function refreshAnthropicToken(credentials: OAuthCredentials): Promise<OAuthCredentials> {
131
+ const response = await fetch(TOKEN_URL, {
132
+ method: 'POST',
133
+ headers: { 'Content-Type': 'application/json' },
134
+ body: JSON.stringify({
135
+ grant_type: 'refresh_token',
136
+ client_id: CLIENT_ID,
137
+ refresh_token: credentials.refresh
138
+ })
139
+ });
140
+
141
+ if (!response.ok) {
142
+ throw new Error(`Token refresh failed: ${await response.text()}`);
143
+ }
144
+
145
+ const data = (await response.json()) as {
146
+ access_token: string;
147
+ refresh_token: string;
148
+ expires_in: number;
149
+ };
150
+
151
+ return {
152
+ refresh: data.refresh_token,
153
+ access: data.access_token,
154
+ expires: Date.now() + data.expires_in * 1000 - 5 * 60 * 1000
155
+ };
156
+ }
157
+
158
+ // =============================================================================
159
+ // Streaming Implementation (simplified from packages/ai/src/providers/anthropic.ts)
160
+ // =============================================================================
161
+
162
+ // Claude Code tool names for OAuth stealth mode
163
+ const claudeCodeTools = [
164
+ 'Read',
165
+ 'Write',
166
+ 'Edit',
167
+ 'Bash',
168
+ 'Grep',
169
+ 'Glob',
170
+ 'AskUserQuestion',
171
+ 'TodoWrite',
172
+ 'WebFetch',
173
+ 'WebSearch'
174
+ ];
175
+ const ccToolLookup = new Map(claudeCodeTools.map((t) => [t.toLowerCase(), t]));
176
+ const toClaudeCodeName = (name: string) => ccToolLookup.get(name.toLowerCase()) ?? name;
177
+ const fromClaudeCodeName = (name: string, tools?: Tool[]) => {
178
+ const lowerName = name.toLowerCase();
179
+ const matched = tools?.find((t) => t.name.toLowerCase() === lowerName);
180
+ return matched?.name ?? name;
181
+ };
182
+
183
+ function isOAuthToken(apiKey: string): boolean {
184
+ return apiKey.includes('sk-ant-oat');
185
+ }
186
+
187
+ function sanitizeSurrogates(text: string): string {
188
+ return text.replace(/[\uD800-\uDFFF]/g, '\uFFFD');
189
+ }
190
+
191
+ function convertContentBlocks(
192
+ content: (TextContent | ImageContent)[]
193
+ ): string | Array<{ type: 'text'; text: string } | { type: 'image'; source: any }> {
194
+ const hasImages = content.some((c) => c.type === 'image');
195
+ if (!hasImages) {
196
+ return sanitizeSurrogates(content.map((c) => (c as TextContent).text).join('\n'));
197
+ }
198
+
199
+ const blocks = content.map((block) => {
200
+ if (block.type === 'text') {
201
+ return { type: 'text' as const, text: sanitizeSurrogates(block.text) };
202
+ }
203
+ return {
204
+ type: 'image' as const,
205
+ source: {
206
+ type: 'base64' as const,
207
+ media_type: block.mimeType,
208
+ data: block.data
209
+ }
210
+ };
211
+ });
212
+
213
+ if (!blocks.some((b) => b.type === 'text')) {
214
+ blocks.unshift({ type: 'text' as const, text: '(see attached image)' });
215
+ }
216
+
217
+ return blocks;
218
+ }
219
+
220
+ function convertMessages(messages: Message[], isOAuth: boolean, _tools?: Tool[]): any[] {
221
+ const params: any[] = [];
222
+
223
+ for (let i = 0; i < messages.length; i++) {
224
+ const msg = messages[i];
225
+
226
+ if (msg.role === 'user') {
227
+ if (typeof msg.content === 'string') {
228
+ if (msg.content.trim()) {
229
+ params.push({ role: 'user', content: sanitizeSurrogates(msg.content) });
230
+ }
231
+ } else {
232
+ const blocks: ContentBlockParam[] = msg.content.map((item) =>
233
+ item.type === 'text'
234
+ ? { type: 'text' as const, text: sanitizeSurrogates(item.text) }
235
+ : {
236
+ type: 'image' as const,
237
+ source: {
238
+ type: 'base64' as const,
239
+ media_type: item.mimeType as any,
240
+ data: item.data
241
+ }
242
+ }
243
+ );
244
+ if (blocks.length > 0) {
245
+ params.push({ role: 'user', content: blocks });
246
+ }
247
+ }
248
+ } else if (msg.role === 'assistant') {
249
+ const blocks: ContentBlockParam[] = [];
250
+ for (const block of msg.content) {
251
+ if (block.type === 'text' && block.text.trim()) {
252
+ blocks.push({ type: 'text', text: sanitizeSurrogates(block.text) });
253
+ } else if (block.type === 'thinking' && block.thinking.trim()) {
254
+ if ((block as ThinkingContent).thinkingSignature) {
255
+ blocks.push({
256
+ type: 'thinking' as any,
257
+ thinking: sanitizeSurrogates(block.thinking),
258
+ signature: (block as ThinkingContent).thinkingSignature!
259
+ });
260
+ } else {
261
+ blocks.push({ type: 'text', text: sanitizeSurrogates(block.thinking) });
262
+ }
263
+ } else if (block.type === 'toolCall') {
264
+ blocks.push({
265
+ type: 'tool_use',
266
+ id: block.id,
267
+ name: isOAuth ? toClaudeCodeName(block.name) : block.name,
268
+ input: block.arguments
269
+ });
270
+ }
271
+ }
272
+ if (blocks.length > 0) {
273
+ params.push({ role: 'assistant', content: blocks });
274
+ }
275
+ } else if (msg.role === 'toolResult') {
276
+ const toolResults: any[] = [];
277
+ toolResults.push({
278
+ type: 'tool_result',
279
+ tool_use_id: msg.toolCallId,
280
+ content: convertContentBlocks(msg.content),
281
+ is_error: msg.isError
282
+ });
283
+
284
+ let j = i + 1;
285
+ while (j < messages.length && messages[j].role === 'toolResult') {
286
+ const nextMsg = messages[j] as ToolResultMessage;
287
+ toolResults.push({
288
+ type: 'tool_result',
289
+ tool_use_id: nextMsg.toolCallId,
290
+ content: convertContentBlocks(nextMsg.content),
291
+ is_error: nextMsg.isError
292
+ });
293
+ j++;
294
+ }
295
+ i = j - 1;
296
+ params.push({ role: 'user', content: toolResults });
297
+ }
298
+ }
299
+
300
+ // Add cache control to last user message
301
+ if (params.length > 0) {
302
+ const last = params[params.length - 1];
303
+ if (last.role === 'user' && Array.isArray(last.content)) {
304
+ const lastBlock = last.content[last.content.length - 1];
305
+ if (lastBlock) {
306
+ lastBlock.cache_control = { type: 'ephemeral' };
307
+ }
308
+ }
309
+ }
310
+
311
+ return params;
312
+ }
313
+
314
+ function convertTools(tools: Tool[], isOAuth: boolean): any[] {
315
+ return tools.map((tool) => ({
316
+ name: isOAuth ? toClaudeCodeName(tool.name) : tool.name,
317
+ description: tool.description,
318
+ input_schema: {
319
+ type: 'object',
320
+ properties: (tool.parameters as any).properties || {},
321
+ required: (tool.parameters as any).required || []
322
+ }
323
+ }));
324
+ }
325
+
326
+ function mapStopReason(reason: string): StopReason {
327
+ switch (reason) {
328
+ case 'end_turn':
329
+ case 'pause_turn':
330
+ case 'stop_sequence':
331
+ return 'stop';
332
+ case 'max_tokens':
333
+ return 'length';
334
+ case 'tool_use':
335
+ return 'toolUse';
336
+ default:
337
+ return 'error';
338
+ }
339
+ }
340
+
341
+ function streamCustomAnthropic(
342
+ model: Model<Api>,
343
+ context: Context,
344
+ options?: SimpleStreamOptions
345
+ ): AssistantMessageEventStream {
346
+ const stream = createAssistantMessageEventStream();
347
+
348
+ (async () => {
349
+ const output: AssistantMessage = {
350
+ role: 'assistant',
351
+ content: [],
352
+ api: model.api,
353
+ provider: model.provider,
354
+ model: model.id,
355
+ usage: {
356
+ input: 0,
357
+ output: 0,
358
+ cacheRead: 0,
359
+ cacheWrite: 0,
360
+ totalTokens: 0,
361
+ cost: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0, total: 0 }
362
+ },
363
+ stopReason: 'stop',
364
+ timestamp: Date.now()
365
+ };
366
+
367
+ try {
368
+ const apiKey = options?.apiKey ?? '';
369
+ const isOAuth = isOAuthToken(apiKey);
370
+
371
+ // Configure client based on auth type
372
+ const betaFeatures = [
373
+ 'fine-grained-tool-streaming-2025-05-14',
374
+ 'interleaved-thinking-2025-05-14'
375
+ ];
376
+ const clientOptions: any = {
377
+ baseURL: model.baseUrl,
378
+ dangerouslyAllowBrowser: true
379
+ };
380
+
381
+ if (isOAuth) {
382
+ clientOptions.apiKey = null;
383
+ clientOptions.authToken = apiKey;
384
+ clientOptions.defaultHeaders = {
385
+ accept: 'application/json',
386
+ 'anthropic-dangerous-direct-browser-access': 'true',
387
+ 'anthropic-beta': `claude-code-20250219,oauth-2025-04-20,${betaFeatures.join(',')}`,
388
+ 'user-agent': 'claude-cli/2.1.2 (external, cli)',
389
+ 'x-app': 'cli'
390
+ };
391
+ } else {
392
+ clientOptions.apiKey = apiKey;
393
+ clientOptions.defaultHeaders = {
394
+ accept: 'application/json',
395
+ 'anthropic-dangerous-direct-browser-access': 'true',
396
+ 'anthropic-beta': betaFeatures.join(',')
397
+ };
398
+ }
399
+
400
+ const client = new Anthropic(clientOptions);
401
+
402
+ // Build request params
403
+ const params: MessageCreateParamsStreaming = {
404
+ model: model.id,
405
+ messages: convertMessages(context.messages, isOAuth, context.tools),
406
+ max_tokens: options?.maxTokens || Math.floor(model.maxTokens / 3),
407
+ stream: true
408
+ };
409
+
410
+ // System prompt with Claude Code identity for OAuth
411
+ if (isOAuth) {
412
+ params.system = [
413
+ {
414
+ type: 'text',
415
+ text: "You are Claude Code, Anthropic's official CLI for Claude.",
416
+ cache_control: { type: 'ephemeral' }
417
+ }
418
+ ];
419
+ if (context.systemPrompt) {
420
+ params.system.push({
421
+ type: 'text',
422
+ text: sanitizeSurrogates(context.systemPrompt),
423
+ cache_control: { type: 'ephemeral' }
424
+ });
425
+ }
426
+ } else if (context.systemPrompt) {
427
+ params.system = [
428
+ {
429
+ type: 'text',
430
+ text: sanitizeSurrogates(context.systemPrompt),
431
+ cache_control: { type: 'ephemeral' }
432
+ }
433
+ ];
434
+ }
435
+
436
+ if (context.tools) {
437
+ params.tools = convertTools(context.tools, isOAuth);
438
+ }
439
+
440
+ // Handle thinking/reasoning
441
+ if (options?.reasoning && model.reasoning) {
442
+ const defaultBudgets: Record<string, number> = {
443
+ minimal: 1024,
444
+ low: 4096,
445
+ medium: 10240,
446
+ high: 20480
447
+ };
448
+ const customBudget =
449
+ options.thinkingBudgets?.[options.reasoning as keyof typeof options.thinkingBudgets];
450
+ params.thinking = {
451
+ type: 'enabled',
452
+ budget_tokens: customBudget ?? defaultBudgets[options.reasoning] ?? 10240
453
+ };
454
+ }
455
+
456
+ const anthropicStream = client.messages.stream({ ...params }, { signal: options?.signal });
457
+ stream.push({ type: 'start', partial: output });
458
+
459
+ type Block = (ThinkingContent | TextContent | (ToolCall & { partialJson: string })) & {
460
+ index: number;
461
+ };
462
+ const blocks = output.content as Block[];
463
+
464
+ for await (const event of anthropicStream) {
465
+ if (event.type === 'message_start') {
466
+ output.usage.input = event.message.usage.input_tokens || 0;
467
+ output.usage.output = event.message.usage.output_tokens || 0;
468
+ output.usage.cacheRead = (event.message.usage as any).cache_read_input_tokens || 0;
469
+ output.usage.cacheWrite = (event.message.usage as any).cache_creation_input_tokens || 0;
470
+ output.usage.totalTokens =
471
+ output.usage.input +
472
+ output.usage.output +
473
+ output.usage.cacheRead +
474
+ output.usage.cacheWrite;
475
+ calculateCost(model, output.usage);
476
+ } else if (event.type === 'content_block_start') {
477
+ if (event.content_block.type === 'text') {
478
+ output.content.push({ type: 'text', text: '', index: event.index } as any);
479
+ stream.push({
480
+ type: 'text_start',
481
+ contentIndex: output.content.length - 1,
482
+ partial: output
483
+ });
484
+ } else if (event.content_block.type === 'thinking') {
485
+ output.content.push({
486
+ type: 'thinking',
487
+ thinking: '',
488
+ thinkingSignature: '',
489
+ index: event.index
490
+ } as any);
491
+ stream.push({
492
+ type: 'thinking_start',
493
+ contentIndex: output.content.length - 1,
494
+ partial: output
495
+ });
496
+ } else if (event.content_block.type === 'tool_use') {
497
+ output.content.push({
498
+ type: 'toolCall',
499
+ id: event.content_block.id,
500
+ name: isOAuth
501
+ ? fromClaudeCodeName(event.content_block.name, context.tools)
502
+ : event.content_block.name,
503
+ arguments: {},
504
+ partialJson: '',
505
+ index: event.index
506
+ } as any);
507
+ stream.push({
508
+ type: 'toolcall_start',
509
+ contentIndex: output.content.length - 1,
510
+ partial: output
511
+ });
512
+ }
513
+ } else if (event.type === 'content_block_delta') {
514
+ const index = blocks.findIndex((b) => b.index === event.index);
515
+ const block = blocks[index];
516
+ if (!block) continue;
517
+
518
+ if (event.delta.type === 'text_delta' && block.type === 'text') {
519
+ block.text += event.delta.text;
520
+ stream.push({
521
+ type: 'text_delta',
522
+ contentIndex: index,
523
+ delta: event.delta.text,
524
+ partial: output
525
+ });
526
+ } else if (event.delta.type === 'thinking_delta' && block.type === 'thinking') {
527
+ block.thinking += event.delta.thinking;
528
+ stream.push({
529
+ type: 'thinking_delta',
530
+ contentIndex: index,
531
+ delta: event.delta.thinking,
532
+ partial: output
533
+ });
534
+ } else if (event.delta.type === 'input_json_delta' && block.type === 'toolCall') {
535
+ (block as any).partialJson += event.delta.partial_json;
536
+ try {
537
+ block.arguments = JSON.parse((block as any).partialJson);
538
+ } catch {}
539
+ stream.push({
540
+ type: 'toolcall_delta',
541
+ contentIndex: index,
542
+ delta: event.delta.partial_json,
543
+ partial: output
544
+ });
545
+ } else if (event.delta.type === 'signature_delta' && block.type === 'thinking') {
546
+ block.thinkingSignature =
547
+ (block.thinkingSignature || '') + (event.delta as any).signature;
548
+ }
549
+ } else if (event.type === 'content_block_stop') {
550
+ const index = blocks.findIndex((b) => b.index === event.index);
551
+ const block = blocks[index];
552
+ if (!block) continue;
553
+
554
+ delete (block as any).index;
555
+ if (block.type === 'text') {
556
+ stream.push({
557
+ type: 'text_end',
558
+ contentIndex: index,
559
+ content: block.text,
560
+ partial: output
561
+ });
562
+ } else if (block.type === 'thinking') {
563
+ stream.push({
564
+ type: 'thinking_end',
565
+ contentIndex: index,
566
+ content: block.thinking,
567
+ partial: output
568
+ });
569
+ } else if (block.type === 'toolCall') {
570
+ try {
571
+ block.arguments = JSON.parse((block as any).partialJson);
572
+ } catch {}
573
+ delete (block as any).partialJson;
574
+ stream.push({
575
+ type: 'toolcall_end',
576
+ contentIndex: index,
577
+ toolCall: block,
578
+ partial: output
579
+ });
580
+ }
581
+ } else if (event.type === 'message_delta') {
582
+ if ((event.delta as any).stop_reason) {
583
+ output.stopReason = mapStopReason((event.delta as any).stop_reason);
584
+ }
585
+ output.usage.input = (event.usage as any).input_tokens || 0;
586
+ output.usage.output = (event.usage as any).output_tokens || 0;
587
+ output.usage.cacheRead = (event.usage as any).cache_read_input_tokens || 0;
588
+ output.usage.cacheWrite = (event.usage as any).cache_creation_input_tokens || 0;
589
+ output.usage.totalTokens =
590
+ output.usage.input +
591
+ output.usage.output +
592
+ output.usage.cacheRead +
593
+ output.usage.cacheWrite;
594
+ calculateCost(model, output.usage);
595
+ }
596
+ }
597
+
598
+ if (options?.signal?.aborted) {
599
+ throw new Error('Request was aborted');
600
+ }
601
+
602
+ stream.push({
603
+ type: 'done',
604
+ reason: output.stopReason as 'stop' | 'length' | 'toolUse',
605
+ message: output
606
+ });
607
+ stream.end();
608
+ } catch (error) {
609
+ for (const block of output.content) delete (block as any).index;
610
+ output.stopReason = options?.signal?.aborted ? 'aborted' : 'error';
611
+ output.errorMessage = error instanceof Error ? error.message : JSON.stringify(error);
612
+ stream.push({ type: 'error', reason: output.stopReason, error: output });
613
+ stream.end();
614
+ }
615
+ })();
616
+
617
+ return stream;
618
+ }
619
+
620
+ // =============================================================================
621
+ // Extension Entry Point
622
+ // =============================================================================
623
+
624
+ export default function (shortcut: ExtensionAPI) {
625
+ shortcut.registerProvider('custom-anthropic', {
626
+ baseUrl: 'https://api.anthropic.com',
627
+ apiKey: 'CUSTOM_ANTHROPIC_API_KEY',
628
+ api: 'custom-anthropic-api',
629
+
630
+ models: [
631
+ {
632
+ id: 'claude-opus-4-5',
633
+ name: 'Claude Opus 4.5 (Custom)',
634
+ reasoning: true,
635
+ input: ['text', 'image'],
636
+ cost: { input: 5, output: 25, cacheRead: 0.5, cacheWrite: 6.25 },
637
+ contextWindow: 200000,
638
+ maxTokens: 64000
639
+ },
640
+ {
641
+ id: 'claude-sonnet-4-5',
642
+ name: 'Claude Sonnet 4.5 (Custom)',
643
+ reasoning: true,
644
+ input: ['text', 'image'],
645
+ cost: { input: 3, output: 15, cacheRead: 0.3, cacheWrite: 3.75 },
646
+ contextWindow: 200000,
647
+ maxTokens: 64000
648
+ }
649
+ ],
650
+
651
+ oauth: {
652
+ name: 'Custom Anthropic (Claude Pro/Max)',
653
+ login: loginAnthropic,
654
+ refreshToken: refreshAnthropicToken,
655
+ getApiKey: (cred) => cred.access
656
+ },
657
+
658
+ streamSimple: streamCustomAnthropic
659
+ });
660
+ }
@@ -0,0 +1,24 @@
1
+ {
2
+ "name": "pi-extension-custom-provider",
3
+ "version": "1.6.4",
4
+ "lockfileVersion": 3,
5
+ "requires": true,
6
+ "packages": {
7
+ "": {
8
+ "name": "pi-extension-custom-provider",
9
+ "version": "1.6.4",
10
+ "dependencies": {
11
+ "@anthropic-ai/sdk": "^0.52.0"
12
+ }
13
+ },
14
+ "node_modules/@anthropic-ai/sdk": {
15
+ "version": "0.52.0",
16
+ "resolved": "https://registry.npmjs.org/@anthropic-ai/sdk/-/sdk-0.52.0.tgz",
17
+ "integrity": "sha512-d4c+fg+xy9e46c8+YnrrgIQR45CZlAi7PwdzIfDXDM6ACxEZli1/fxhURsq30ZpMZy6LvSkr41jGq5aF5TD7rQ==",
18
+ "license": "MIT",
19
+ "bin": {
20
+ "anthropic-ai-sdk": "bin/cli"
21
+ }
22
+ }
23
+ }
24
+ }
@@ -0,0 +1,19 @@
1
+ {
2
+ "name": "pi-extension-custom-provider-anthropic",
3
+ "private": true,
4
+ "version": "1.6.4",
5
+ "type": "module",
6
+ "scripts": {
7
+ "clean": "echo 'nothing to clean'",
8
+ "build": "echo 'nothing to build'",
9
+ "check": "echo 'nothing to check'"
10
+ },
11
+ "pi": {
12
+ "extensions": [
13
+ "./index.ts"
14
+ ]
15
+ },
16
+ "dependencies": {
17
+ "@anthropic-ai/sdk": "^0.52.0"
18
+ }
19
+ }