innocode 1.0.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 (434) hide show
  1. package/AGENTS.md +27 -0
  2. package/Dockerfile +18 -0
  3. package/README.md +15 -0
  4. package/bin/innocode +84 -0
  5. package/bin/opencode +84 -0
  6. package/bunfig.toml +5 -0
  7. package/package.json +126 -0
  8. package/parsers-config.ts +253 -0
  9. package/script/build.ts +198 -0
  10. package/script/postinstall.mjs +125 -0
  11. package/script/publish.ts +186 -0
  12. package/script/schema.ts +47 -0
  13. package/script/seed-e2e.ts +50 -0
  14. package/src/acp/README.md +164 -0
  15. package/src/acp/agent.ts +1676 -0
  16. package/src/acp/session.ts +117 -0
  17. package/src/acp/types.ts +23 -0
  18. package/src/agent/agent.ts +338 -0
  19. package/src/agent/generate.txt +75 -0
  20. package/src/agent/prompt/compaction.txt +14 -0
  21. package/src/agent/prompt/explore.txt +18 -0
  22. package/src/agent/prompt/summary.txt +11 -0
  23. package/src/agent/prompt/title.txt +44 -0
  24. package/src/auth/index.ts +70 -0
  25. package/src/bun/index.ts +137 -0
  26. package/src/bun/registry.ts +48 -0
  27. package/src/bus/bus-event.ts +43 -0
  28. package/src/bus/global.ts +10 -0
  29. package/src/bus/index.ts +105 -0
  30. package/src/cli/bootstrap.ts +17 -0
  31. package/src/cli/cmd/acp.ts +70 -0
  32. package/src/cli/cmd/agent.ts +257 -0
  33. package/src/cli/cmd/auth.ts +400 -0
  34. package/src/cli/cmd/cmd.ts +7 -0
  35. package/src/cli/cmd/debug/agent.ts +167 -0
  36. package/src/cli/cmd/debug/config.ts +16 -0
  37. package/src/cli/cmd/debug/file.ts +97 -0
  38. package/src/cli/cmd/debug/index.ts +48 -0
  39. package/src/cli/cmd/debug/lsp.ts +52 -0
  40. package/src/cli/cmd/debug/ripgrep.ts +87 -0
  41. package/src/cli/cmd/debug/scrap.ts +16 -0
  42. package/src/cli/cmd/debug/skill.ts +16 -0
  43. package/src/cli/cmd/debug/snapshot.ts +52 -0
  44. package/src/cli/cmd/export.ts +88 -0
  45. package/src/cli/cmd/generate.ts +38 -0
  46. package/src/cli/cmd/github.ts +1540 -0
  47. package/src/cli/cmd/import.ts +147 -0
  48. package/src/cli/cmd/mcp.ts +765 -0
  49. package/src/cli/cmd/models.ts +77 -0
  50. package/src/cli/cmd/pr.ts +113 -0
  51. package/src/cli/cmd/run.ts +598 -0
  52. package/src/cli/cmd/serve.ts +20 -0
  53. package/src/cli/cmd/session.ts +135 -0
  54. package/src/cli/cmd/stats.ts +426 -0
  55. package/src/cli/cmd/tui/app.tsx +812 -0
  56. package/src/cli/cmd/tui/attach.ts +60 -0
  57. package/src/cli/cmd/tui/component/border.tsx +21 -0
  58. package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
  59. package/src/cli/cmd/tui/component/dialog-command.tsx +148 -0
  60. package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
  61. package/src/cli/cmd/tui/component/dialog-model.tsx +165 -0
  62. package/src/cli/cmd/tui/component/dialog-provider.tsx +243 -0
  63. package/src/cli/cmd/tui/component/dialog-session-list.tsx +108 -0
  64. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
  65. package/src/cli/cmd/tui/component/dialog-skill.tsx +36 -0
  66. package/src/cli/cmd/tui/component/dialog-stash.tsx +87 -0
  67. package/src/cli/cmd/tui/component/dialog-status.tsx +167 -0
  68. package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
  69. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
  70. package/src/cli/cmd/tui/component/logo.tsx +85 -0
  71. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +666 -0
  72. package/src/cli/cmd/tui/component/prompt/frecency.tsx +89 -0
  73. package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
  74. package/src/cli/cmd/tui/component/prompt/index.tsx +1153 -0
  75. package/src/cli/cmd/tui/component/prompt/stash.tsx +101 -0
  76. package/src/cli/cmd/tui/component/spinner.tsx +24 -0
  77. package/src/cli/cmd/tui/component/textarea-keybindings.ts +73 -0
  78. package/src/cli/cmd/tui/component/tips.tsx +153 -0
  79. package/src/cli/cmd/tui/component/todo-item.tsx +32 -0
  80. package/src/cli/cmd/tui/context/args.tsx +15 -0
  81. package/src/cli/cmd/tui/context/directory.ts +13 -0
  82. package/src/cli/cmd/tui/context/exit.tsx +54 -0
  83. package/src/cli/cmd/tui/context/helper.tsx +25 -0
  84. package/src/cli/cmd/tui/context/keybind.tsx +100 -0
  85. package/src/cli/cmd/tui/context/kv.tsx +52 -0
  86. package/src/cli/cmd/tui/context/local.tsx +409 -0
  87. package/src/cli/cmd/tui/context/prompt.tsx +18 -0
  88. package/src/cli/cmd/tui/context/route.tsx +46 -0
  89. package/src/cli/cmd/tui/context/sdk.tsx +101 -0
  90. package/src/cli/cmd/tui/context/sync.tsx +470 -0
  91. package/src/cli/cmd/tui/context/theme/aura.json +69 -0
  92. package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
  93. package/src/cli/cmd/tui/context/theme/carbonfox.json +248 -0
  94. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +233 -0
  95. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
  96. package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
  97. package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
  98. package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
  99. package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
  100. package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
  101. package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
  102. package/src/cli/cmd/tui/context/theme/github.json +233 -0
  103. package/src/cli/cmd/tui/context/theme/gruvbox.json +242 -0
  104. package/src/cli/cmd/tui/context/theme/innocode.json +245 -0
  105. package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
  106. package/src/cli/cmd/tui/context/theme/lucent-orng.json +237 -0
  107. package/src/cli/cmd/tui/context/theme/material.json +235 -0
  108. package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
  109. package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
  110. package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
  111. package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
  112. package/src/cli/cmd/tui/context/theme/nord.json +223 -0
  113. package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
  114. package/src/cli/cmd/tui/context/theme/orng.json +249 -0
  115. package/src/cli/cmd/tui/context/theme/osaka-jade.json +93 -0
  116. package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
  117. package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
  118. package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
  119. package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
  120. package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
  121. package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
  122. package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
  123. package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
  124. package/src/cli/cmd/tui/context/theme.tsx +1154 -0
  125. package/src/cli/cmd/tui/event.ts +48 -0
  126. package/src/cli/cmd/tui/routes/home.tsx +145 -0
  127. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +64 -0
  128. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +109 -0
  129. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +26 -0
  130. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
  131. package/src/cli/cmd/tui/routes/session/footer.tsx +91 -0
  132. package/src/cli/cmd/tui/routes/session/header.tsx +135 -0
  133. package/src/cli/cmd/tui/routes/session/index.tsx +2139 -0
  134. package/src/cli/cmd/tui/routes/session/permission.tsx +508 -0
  135. package/src/cli/cmd/tui/routes/session/question.tsx +466 -0
  136. package/src/cli/cmd/tui/routes/session/sidebar.tsx +313 -0
  137. package/src/cli/cmd/tui/thread.ts +188 -0
  138. package/src/cli/cmd/tui/ui/dialog-alert.tsx +59 -0
  139. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +85 -0
  140. package/src/cli/cmd/tui/ui/dialog-export-options.tsx +207 -0
  141. package/src/cli/cmd/tui/ui/dialog-help.tsx +40 -0
  142. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +80 -0
  143. package/src/cli/cmd/tui/ui/dialog-select.tsx +401 -0
  144. package/src/cli/cmd/tui/ui/dialog.tsx +167 -0
  145. package/src/cli/cmd/tui/ui/link.tsx +28 -0
  146. package/src/cli/cmd/tui/ui/spinner.ts +368 -0
  147. package/src/cli/cmd/tui/ui/toast.tsx +100 -0
  148. package/src/cli/cmd/tui/util/clipboard.ts +159 -0
  149. package/src/cli/cmd/tui/util/editor.ts +32 -0
  150. package/src/cli/cmd/tui/util/signal.ts +7 -0
  151. package/src/cli/cmd/tui/util/terminal.ts +114 -0
  152. package/src/cli/cmd/tui/util/transcript.ts +98 -0
  153. package/src/cli/cmd/tui/win32.ts +129 -0
  154. package/src/cli/cmd/tui/worker.ts +152 -0
  155. package/src/cli/cmd/uninstall.ts +363 -0
  156. package/src/cli/cmd/upgrade.ts +73 -0
  157. package/src/cli/cmd/web.ts +81 -0
  158. package/src/cli/error.ts +57 -0
  159. package/src/cli/logo.ts +6 -0
  160. package/src/cli/network.ts +60 -0
  161. package/src/cli/ui.ts +113 -0
  162. package/src/cli/upgrade.ts +25 -0
  163. package/src/command/index.ts +150 -0
  164. package/src/command/template/initialize.txt +10 -0
  165. package/src/command/template/review.txt +101 -0
  166. package/src/config/config.ts +1517 -0
  167. package/src/config/markdown.ts +98 -0
  168. package/src/env/index.ts +28 -0
  169. package/src/file/ignore.ts +83 -0
  170. package/src/file/index.ts +583 -0
  171. package/src/file/ripgrep.ts +375 -0
  172. package/src/file/time.ts +69 -0
  173. package/src/file/watcher.ts +127 -0
  174. package/src/flag/flag.ts +148 -0
  175. package/src/format/formatter.ts +366 -0
  176. package/src/format/index.ts +137 -0
  177. package/src/global/index.ts +80 -0
  178. package/src/id/id.ts +83 -0
  179. package/src/ide/index.ts +76 -0
  180. package/src/index.ts +160 -0
  181. package/src/installation/index.ts +268 -0
  182. package/src/lsp/client.ts +252 -0
  183. package/src/lsp/index.ts +485 -0
  184. package/src/lsp/language.ts +119 -0
  185. package/src/lsp/server.ts +2046 -0
  186. package/src/mcp/auth.ts +132 -0
  187. package/src/mcp/index.ts +937 -0
  188. package/src/mcp/oauth-callback.ts +200 -0
  189. package/src/mcp/oauth-provider.ts +154 -0
  190. package/src/patch/index.ts +680 -0
  191. package/src/permission/arity.ts +163 -0
  192. package/src/permission/index.ts +210 -0
  193. package/src/permission/next.ts +280 -0
  194. package/src/plugin/codex.ts +624 -0
  195. package/src/plugin/copilot.ts +327 -0
  196. package/src/plugin/index.ts +138 -0
  197. package/src/project/bootstrap.ts +35 -0
  198. package/src/project/instance.ts +114 -0
  199. package/src/project/project.ts +371 -0
  200. package/src/project/state.ts +70 -0
  201. package/src/project/vcs.ts +76 -0
  202. package/src/provider/auth.ts +147 -0
  203. package/src/provider/error.ts +189 -0
  204. package/src/provider/models.ts +133 -0
  205. package/src/provider/provider.ts +1370 -0
  206. package/src/provider/sdk/copilot/README.md +5 -0
  207. package/src/provider/sdk/copilot/chat/convert-to-openai-compatible-chat-messages.ts +164 -0
  208. package/src/provider/sdk/copilot/chat/get-response-metadata.ts +15 -0
  209. package/src/provider/sdk/copilot/chat/map-openai-compatible-finish-reason.ts +17 -0
  210. package/src/provider/sdk/copilot/chat/openai-compatible-api-types.ts +64 -0
  211. package/src/provider/sdk/copilot/chat/openai-compatible-chat-language-model.ts +780 -0
  212. package/src/provider/sdk/copilot/chat/openai-compatible-chat-options.ts +28 -0
  213. package/src/provider/sdk/copilot/chat/openai-compatible-metadata-extractor.ts +44 -0
  214. package/src/provider/sdk/copilot/chat/openai-compatible-prepare-tools.ts +87 -0
  215. package/src/provider/sdk/copilot/copilot-provider.ts +100 -0
  216. package/src/provider/sdk/copilot/index.ts +2 -0
  217. package/src/provider/sdk/copilot/openai-compatible-error.ts +27 -0
  218. package/src/provider/sdk/copilot/responses/convert-to-openai-responses-input.ts +303 -0
  219. package/src/provider/sdk/copilot/responses/map-openai-responses-finish-reason.ts +22 -0
  220. package/src/provider/sdk/copilot/responses/openai-config.ts +18 -0
  221. package/src/provider/sdk/copilot/responses/openai-error.ts +22 -0
  222. package/src/provider/sdk/copilot/responses/openai-responses-api-types.ts +207 -0
  223. package/src/provider/sdk/copilot/responses/openai-responses-language-model.ts +1732 -0
  224. package/src/provider/sdk/copilot/responses/openai-responses-prepare-tools.ts +177 -0
  225. package/src/provider/sdk/copilot/responses/openai-responses-settings.ts +1 -0
  226. package/src/provider/sdk/copilot/responses/tool/code-interpreter.ts +88 -0
  227. package/src/provider/sdk/copilot/responses/tool/file-search.ts +128 -0
  228. package/src/provider/sdk/copilot/responses/tool/image-generation.ts +115 -0
  229. package/src/provider/sdk/copilot/responses/tool/local-shell.ts +65 -0
  230. package/src/provider/sdk/copilot/responses/tool/web-search-preview.ts +104 -0
  231. package/src/provider/sdk/copilot/responses/tool/web-search.ts +103 -0
  232. package/src/provider/transform.ts +806 -0
  233. package/src/pty/index.ts +286 -0
  234. package/src/question/index.ts +171 -0
  235. package/src/scheduler/index.ts +61 -0
  236. package/src/server/error.ts +36 -0
  237. package/src/server/event.ts +7 -0
  238. package/src/server/mdns.ts +60 -0
  239. package/src/server/routes/config.ts +92 -0
  240. package/src/server/routes/experimental.ts +208 -0
  241. package/src/server/routes/file.ts +197 -0
  242. package/src/server/routes/global.ts +183 -0
  243. package/src/server/routes/mcp.ts +225 -0
  244. package/src/server/routes/permission.ts +68 -0
  245. package/src/server/routes/project.ts +82 -0
  246. package/src/server/routes/provider.ts +179 -0
  247. package/src/server/routes/pty.ts +176 -0
  248. package/src/server/routes/question.ts +98 -0
  249. package/src/server/routes/session.ts +939 -0
  250. package/src/server/routes/tui.ts +379 -0
  251. package/src/server/server.ts +621 -0
  252. package/src/session/compaction.ts +261 -0
  253. package/src/session/index.ts +543 -0
  254. package/src/session/instruction.ts +197 -0
  255. package/src/session/llm.ts +283 -0
  256. package/src/session/message-v2.ts +841 -0
  257. package/src/session/message.ts +189 -0
  258. package/src/session/processor.ts +410 -0
  259. package/src/session/prompt/anthropic-20250930.txt +166 -0
  260. package/src/session/prompt/anthropic.txt +105 -0
  261. package/src/session/prompt/beast.txt +147 -0
  262. package/src/session/prompt/build-switch.txt +5 -0
  263. package/src/session/prompt/codex_header.txt +79 -0
  264. package/src/session/prompt/copilot-gpt-5.txt +143 -0
  265. package/src/session/prompt/gemini.txt +155 -0
  266. package/src/session/prompt/max-steps.txt +16 -0
  267. package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
  268. package/src/session/prompt/plan.txt +26 -0
  269. package/src/session/prompt/qwen.txt +109 -0
  270. package/src/session/prompt/trinity.txt +97 -0
  271. package/src/session/prompt.ts +1964 -0
  272. package/src/session/retry.ts +101 -0
  273. package/src/session/revert.ts +121 -0
  274. package/src/session/status.ts +76 -0
  275. package/src/session/summary.ts +203 -0
  276. package/src/session/system.ts +54 -0
  277. package/src/session/todo.ts +37 -0
  278. package/src/share/share-next.ts +200 -0
  279. package/src/share/share.ts +92 -0
  280. package/src/shell/shell.ts +67 -0
  281. package/src/skill/discovery.ts +97 -0
  282. package/src/skill/index.ts +1 -0
  283. package/src/skill/skill.ts +188 -0
  284. package/src/snapshot/index.ts +255 -0
  285. package/src/storage/storage.ts +227 -0
  286. package/src/tool/apply_patch.ts +281 -0
  287. package/src/tool/apply_patch.txt +33 -0
  288. package/src/tool/bash.ts +269 -0
  289. package/src/tool/bash.txt +115 -0
  290. package/src/tool/batch.ts +175 -0
  291. package/src/tool/batch.txt +24 -0
  292. package/src/tool/codesearch.ts +132 -0
  293. package/src/tool/codesearch.txt +12 -0
  294. package/src/tool/edit.ts +655 -0
  295. package/src/tool/edit.txt +10 -0
  296. package/src/tool/external-directory.ts +32 -0
  297. package/src/tool/glob.ts +80 -0
  298. package/src/tool/glob.txt +6 -0
  299. package/src/tool/grep.ts +150 -0
  300. package/src/tool/grep.txt +8 -0
  301. package/src/tool/invalid.ts +17 -0
  302. package/src/tool/ls.ts +121 -0
  303. package/src/tool/ls.txt +1 -0
  304. package/src/tool/lsp.ts +96 -0
  305. package/src/tool/lsp.txt +19 -0
  306. package/src/tool/multiedit.ts +46 -0
  307. package/src/tool/multiedit.txt +41 -0
  308. package/src/tool/plan-enter.txt +14 -0
  309. package/src/tool/plan-exit.txt +13 -0
  310. package/src/tool/plan.ts +130 -0
  311. package/src/tool/question.ts +33 -0
  312. package/src/tool/question.txt +10 -0
  313. package/src/tool/read.ts +261 -0
  314. package/src/tool/read.txt +14 -0
  315. package/src/tool/registry.ts +160 -0
  316. package/src/tool/skill.ts +123 -0
  317. package/src/tool/task.ts +165 -0
  318. package/src/tool/task.txt +60 -0
  319. package/src/tool/todo.ts +53 -0
  320. package/src/tool/todoread.txt +14 -0
  321. package/src/tool/todowrite.txt +167 -0
  322. package/src/tool/tool.ts +89 -0
  323. package/src/tool/truncation.ts +106 -0
  324. package/src/tool/webfetch.ts +186 -0
  325. package/src/tool/webfetch.txt +13 -0
  326. package/src/tool/websearch.ts +150 -0
  327. package/src/tool/websearch.txt +14 -0
  328. package/src/tool/write.ts +85 -0
  329. package/src/tool/write.txt +8 -0
  330. package/src/util/abort.ts +35 -0
  331. package/src/util/archive.ts +16 -0
  332. package/src/util/color.ts +19 -0
  333. package/src/util/context.ts +25 -0
  334. package/src/util/defer.ts +12 -0
  335. package/src/util/eventloop.ts +20 -0
  336. package/src/util/filesystem.ts +93 -0
  337. package/src/util/fn.ts +11 -0
  338. package/src/util/format.ts +20 -0
  339. package/src/util/iife.ts +3 -0
  340. package/src/util/keybind.ts +103 -0
  341. package/src/util/lazy.ts +18 -0
  342. package/src/util/locale.ts +81 -0
  343. package/src/util/lock.ts +98 -0
  344. package/src/util/log.ts +180 -0
  345. package/src/util/proxied.ts +3 -0
  346. package/src/util/queue.ts +32 -0
  347. package/src/util/rpc.ts +66 -0
  348. package/src/util/scrap.ts +10 -0
  349. package/src/util/signal.ts +12 -0
  350. package/src/util/timeout.ts +14 -0
  351. package/src/util/token.ts +7 -0
  352. package/src/util/wildcard.ts +56 -0
  353. package/src/worktree/index.ts +612 -0
  354. package/sst-env.d.ts +9 -0
  355. package/test/acp/agent-interface.test.ts +51 -0
  356. package/test/acp/event-subscription.test.ts +436 -0
  357. package/test/agent/agent.test.ts +675 -0
  358. package/test/bun.test.ts +53 -0
  359. package/test/cli/github-action.test.ts +161 -0
  360. package/test/cli/github-remote.test.ts +80 -0
  361. package/test/cli/import.test.ts +38 -0
  362. package/test/cli/tui/transcript.test.ts +322 -0
  363. package/test/config/agent-color.test.ts +71 -0
  364. package/test/config/config.test.ts +1802 -0
  365. package/test/config/fixtures/empty-frontmatter.md +4 -0
  366. package/test/config/fixtures/frontmatter.md +28 -0
  367. package/test/config/fixtures/markdown-header.md +11 -0
  368. package/test/config/fixtures/no-frontmatter.md +1 -0
  369. package/test/config/fixtures/weird-model-id.md +13 -0
  370. package/test/config/markdown.test.ts +228 -0
  371. package/test/file/ignore.test.ts +10 -0
  372. package/test/file/path-traversal.test.ts +198 -0
  373. package/test/file/ripgrep.test.ts +39 -0
  374. package/test/fixture/fixture.ts +45 -0
  375. package/test/fixture/lsp/fake-lsp-server.js +77 -0
  376. package/test/ide/ide.test.ts +82 -0
  377. package/test/keybind.test.ts +421 -0
  378. package/test/lsp/client.test.ts +95 -0
  379. package/test/mcp/headers.test.ts +153 -0
  380. package/test/mcp/oauth-browser.test.ts +249 -0
  381. package/test/memory/abort-leak.test.ts +136 -0
  382. package/test/patch/patch.test.ts +348 -0
  383. package/test/permission/arity.test.ts +33 -0
  384. package/test/permission/next.test.ts +690 -0
  385. package/test/permission-task.test.ts +319 -0
  386. package/test/plugin/auth-override.test.ts +44 -0
  387. package/test/plugin/codex.test.ts +123 -0
  388. package/test/preload.ts +63 -0
  389. package/test/project/project.test.ts +120 -0
  390. package/test/provider/amazon-bedrock.test.ts +445 -0
  391. package/test/provider/copilot/convert-to-copilot-messages.test.ts +523 -0
  392. package/test/provider/copilot/copilot-chat-model.test.ts +592 -0
  393. package/test/provider/gitlab-duo.test.ts +262 -0
  394. package/test/provider/provider.test.ts +2129 -0
  395. package/test/provider/transform.test.ts +1928 -0
  396. package/test/question/question.test.ts +300 -0
  397. package/test/scheduler.test.ts +73 -0
  398. package/test/server/session-list.test.ts +39 -0
  399. package/test/server/session-select.test.ts +78 -0
  400. package/test/session/compaction.test.ts +423 -0
  401. package/test/session/instruction.test.ts +170 -0
  402. package/test/session/llm.test.ts +667 -0
  403. package/test/session/message-v2.test.ts +924 -0
  404. package/test/session/prompt-missing-file.test.ts +53 -0
  405. package/test/session/prompt-special-chars.test.ts +56 -0
  406. package/test/session/prompt-variant.test.ts +68 -0
  407. package/test/session/retry.test.ts +188 -0
  408. package/test/session/revert-compact.test.ts +285 -0
  409. package/test/session/session.test.ts +71 -0
  410. package/test/session/structured-output-integration.test.ts +233 -0
  411. package/test/session/structured-output.test.ts +385 -0
  412. package/test/skill/discovery.test.ts +60 -0
  413. package/test/skill/skill.test.ts +388 -0
  414. package/test/snapshot/snapshot.test.ts +1040 -0
  415. package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
  416. package/test/tool/apply_patch.test.ts +559 -0
  417. package/test/tool/bash.test.ts +399 -0
  418. package/test/tool/external-directory.test.ts +127 -0
  419. package/test/tool/fixtures/large-image.png +0 -0
  420. package/test/tool/fixtures/models-api.json +38413 -0
  421. package/test/tool/grep.test.ts +110 -0
  422. package/test/tool/question.test.ts +107 -0
  423. package/test/tool/read.test.ts +421 -0
  424. package/test/tool/registry.test.ts +122 -0
  425. package/test/tool/skill.test.ts +112 -0
  426. package/test/tool/truncation.test.ts +159 -0
  427. package/test/util/filesystem.test.ts +39 -0
  428. package/test/util/format.test.ts +59 -0
  429. package/test/util/iife.test.ts +36 -0
  430. package/test/util/lazy.test.ts +50 -0
  431. package/test/util/lock.test.ts +72 -0
  432. package/test/util/timeout.test.ts +21 -0
  433. package/test/util/wildcard.test.ts +75 -0
  434. package/tsconfig.json +16 -0
@@ -0,0 +1,543 @@
1
+ import { Slug } from "@opencode-ai/util/slug"
2
+ import path from "path"
3
+ import { BusEvent } from "@/bus/bus-event"
4
+ import { Bus } from "@/bus"
5
+ import { Decimal } from "decimal.js"
6
+ import z from "zod"
7
+ import { type ProviderMetadata } from "ai"
8
+ import { Config } from "../config/config"
9
+ import { Flag } from "../flag/flag"
10
+ import { Identifier } from "../id/id"
11
+ import { Installation } from "../installation"
12
+
13
+ import { Storage } from "../storage/storage"
14
+ import { Log } from "../util/log"
15
+ import { MessageV2 } from "./message-v2"
16
+ import { Instance } from "../project/instance"
17
+ import { SessionPrompt } from "./prompt"
18
+ import { fn } from "@/util/fn"
19
+ import { Command } from "../command"
20
+ import { Snapshot } from "@/snapshot"
21
+
22
+ import type { Provider } from "@/provider/provider"
23
+ import { PermissionNext } from "@/permission/next"
24
+ import { Global } from "@/global"
25
+ import type { LanguageModelV2Usage } from "@ai-sdk/provider"
26
+ import { iife } from "@/util/iife"
27
+
28
+ export namespace Session {
29
+ const log = Log.create({ service: "session" })
30
+
31
+ const parentTitlePrefix = "New session - "
32
+ const childTitlePrefix = "Child session - "
33
+
34
+ function createDefaultTitle(isChild = false) {
35
+ return (isChild ? childTitlePrefix : parentTitlePrefix) + new Date().toISOString()
36
+ }
37
+
38
+ export function isDefaultTitle(title: string) {
39
+ return new RegExp(
40
+ `^(${parentTitlePrefix}|${childTitlePrefix})\\d{4}-\\d{2}-\\d{2}T\\d{2}:\\d{2}:\\d{2}\\.\\d{3}Z$`,
41
+ ).test(title)
42
+ }
43
+
44
+ function getForkedTitle(title: string): string {
45
+ const match = title.match(/^(.+) \(fork #(\d+)\)$/)
46
+ if (match) {
47
+ const base = match[1]
48
+ const num = parseInt(match[2], 10)
49
+ return `${base} (fork #${num + 1})`
50
+ }
51
+ return `${title} (fork #1)`
52
+ }
53
+
54
+ export const Info = z
55
+ .object({
56
+ id: Identifier.schema("session"),
57
+ slug: z.string(),
58
+ projectID: z.string(),
59
+ directory: z.string(),
60
+ parentID: Identifier.schema("session").optional(),
61
+ summary: z
62
+ .object({
63
+ additions: z.number(),
64
+ deletions: z.number(),
65
+ files: z.number(),
66
+ diffs: Snapshot.FileDiff.array().optional(),
67
+ })
68
+ .optional(),
69
+ share: z
70
+ .object({
71
+ url: z.string(),
72
+ })
73
+ .optional(),
74
+ title: z.string(),
75
+ version: z.string(),
76
+ time: z.object({
77
+ created: z.number(),
78
+ updated: z.number(),
79
+ compacting: z.number().optional(),
80
+ archived: z.number().optional(),
81
+ }),
82
+ permission: PermissionNext.Ruleset.optional(),
83
+ revert: z
84
+ .object({
85
+ messageID: z.string(),
86
+ partID: z.string().optional(),
87
+ snapshot: z.string().optional(),
88
+ diff: z.string().optional(),
89
+ })
90
+ .optional(),
91
+ })
92
+ .meta({
93
+ ref: "Session",
94
+ })
95
+ export type Info = z.output<typeof Info>
96
+
97
+ export const ShareInfo = z
98
+ .object({
99
+ secret: z.string(),
100
+ url: z.string(),
101
+ })
102
+ .meta({
103
+ ref: "SessionShare",
104
+ })
105
+ export type ShareInfo = z.output<typeof ShareInfo>
106
+
107
+ export const Event = {
108
+ Created: BusEvent.define(
109
+ "session.created",
110
+ z.object({
111
+ info: Info,
112
+ }),
113
+ ),
114
+ Updated: BusEvent.define(
115
+ "session.updated",
116
+ z.object({
117
+ info: Info,
118
+ }),
119
+ ),
120
+ Deleted: BusEvent.define(
121
+ "session.deleted",
122
+ z.object({
123
+ info: Info,
124
+ }),
125
+ ),
126
+ Diff: BusEvent.define(
127
+ "session.diff",
128
+ z.object({
129
+ sessionID: z.string(),
130
+ diff: Snapshot.FileDiff.array(),
131
+ }),
132
+ ),
133
+ Error: BusEvent.define(
134
+ "session.error",
135
+ z.object({
136
+ sessionID: z.string().optional(),
137
+ error: MessageV2.Assistant.shape.error,
138
+ }),
139
+ ),
140
+ }
141
+
142
+ export const create = fn(
143
+ z
144
+ .object({
145
+ parentID: Identifier.schema("session").optional(),
146
+ title: z.string().optional(),
147
+ permission: Info.shape.permission,
148
+ })
149
+ .optional(),
150
+ async (input) => {
151
+ return createNext({
152
+ parentID: input?.parentID,
153
+ directory: Instance.directory,
154
+ title: input?.title,
155
+ permission: input?.permission,
156
+ })
157
+ },
158
+ )
159
+
160
+ export const fork = fn(
161
+ z.object({
162
+ sessionID: Identifier.schema("session"),
163
+ messageID: Identifier.schema("message").optional(),
164
+ }),
165
+ async (input) => {
166
+ const original = await get(input.sessionID)
167
+ if (!original) throw new Error("session not found")
168
+ const title = getForkedTitle(original.title)
169
+ const session = await createNext({
170
+ directory: Instance.directory,
171
+ title,
172
+ })
173
+ const msgs = await messages({ sessionID: input.sessionID })
174
+ const idMap = new Map<string, string>()
175
+
176
+ for (const msg of msgs) {
177
+ if (input.messageID && msg.info.id >= input.messageID) break
178
+ const newID = Identifier.ascending("message")
179
+ idMap.set(msg.info.id, newID)
180
+
181
+ const parentID = msg.info.role === "assistant" && msg.info.parentID ? idMap.get(msg.info.parentID) : undefined
182
+ const cloned = await updateMessage({
183
+ ...msg.info,
184
+ sessionID: session.id,
185
+ id: newID,
186
+ ...(parentID && { parentID }),
187
+ })
188
+
189
+ for (const part of msg.parts) {
190
+ await updatePart({
191
+ ...part,
192
+ id: Identifier.ascending("part"),
193
+ messageID: cloned.id,
194
+ sessionID: session.id,
195
+ })
196
+ }
197
+ }
198
+ return session
199
+ },
200
+ )
201
+
202
+ export const touch = fn(Identifier.schema("session"), async (sessionID) => {
203
+ await update(sessionID, (draft) => {
204
+ draft.time.updated = Date.now()
205
+ })
206
+ })
207
+
208
+ export async function createNext(input: {
209
+ id?: string
210
+ title?: string
211
+ parentID?: string
212
+ directory: string
213
+ permission?: PermissionNext.Ruleset
214
+ }) {
215
+ const result: Info = {
216
+ id: Identifier.descending("session", input.id),
217
+ slug: Slug.create(),
218
+ version: Installation.VERSION,
219
+ projectID: Instance.project.id,
220
+ directory: input.directory,
221
+ parentID: input.parentID,
222
+ title: input.title ?? createDefaultTitle(!!input.parentID),
223
+ permission: input.permission,
224
+ time: {
225
+ created: Date.now(),
226
+ updated: Date.now(),
227
+ },
228
+ }
229
+ log.info("created", result)
230
+ await Storage.write(["session", Instance.project.id, result.id], result)
231
+ Bus.publish(Event.Created, {
232
+ info: result,
233
+ })
234
+ const cfg = await Config.get()
235
+ if (!result.parentID && (Flag.OPENCODE_AUTO_SHARE || cfg.share === "auto"))
236
+ share(result.id)
237
+ .then((share) => {
238
+ update(result.id, (draft) => {
239
+ draft.share = share
240
+ })
241
+ })
242
+ .catch(() => {
243
+ // Silently ignore sharing errors during session creation
244
+ })
245
+ Bus.publish(Event.Updated, {
246
+ info: result,
247
+ })
248
+ return result
249
+ }
250
+
251
+ export function plan(input: { slug: string; time: { created: number } }) {
252
+ const base = Instance.project.vcs
253
+ ? path.join(Instance.worktree, ".innocode", "plans")
254
+ : path.join(Global.Path.data, "plans")
255
+ return path.join(base, [input.time.created, input.slug].join("-") + ".md")
256
+ }
257
+
258
+ export const get = fn(Identifier.schema("session"), async (id) => {
259
+ const read = await Storage.read<Info>(["session", Instance.project.id, id])
260
+ return read as Info
261
+ })
262
+
263
+ export const getShare = fn(Identifier.schema("session"), async (id) => {
264
+ return Storage.read<ShareInfo>(["share", id])
265
+ })
266
+
267
+ export const share = fn(Identifier.schema("session"), async (id) => {
268
+ const cfg = await Config.get()
269
+ if (cfg.share === "disabled") {
270
+ throw new Error("Sharing is disabled in configuration")
271
+ }
272
+ const { ShareNext } = await import("@/share/share-next")
273
+ const share = await ShareNext.create(id)
274
+ await update(
275
+ id,
276
+ (draft) => {
277
+ draft.share = {
278
+ url: share.url,
279
+ }
280
+ },
281
+ { touch: false },
282
+ )
283
+ return share
284
+ })
285
+
286
+ export const unshare = fn(Identifier.schema("session"), async (id) => {
287
+ // Use ShareNext to remove the share (same as share function uses ShareNext to create)
288
+ const { ShareNext } = await import("@/share/share-next")
289
+ await ShareNext.remove(id)
290
+ await update(
291
+ id,
292
+ (draft) => {
293
+ draft.share = undefined
294
+ },
295
+ { touch: false },
296
+ )
297
+ })
298
+
299
+ export async function update(id: string, editor: (session: Info) => void, options?: { touch?: boolean }) {
300
+ const project = Instance.project
301
+ const result = await Storage.update<Info>(["session", project.id, id], (draft) => {
302
+ editor(draft)
303
+ if (options?.touch !== false) {
304
+ draft.time.updated = Date.now()
305
+ }
306
+ })
307
+ Bus.publish(Event.Updated, {
308
+ info: result,
309
+ })
310
+ return result
311
+ }
312
+
313
+ export const diff = fn(Identifier.schema("session"), async (sessionID) => {
314
+ const diffs = await Storage.read<Snapshot.FileDiff[]>(["session_diff", sessionID])
315
+ return diffs ?? []
316
+ })
317
+
318
+ export const messages = fn(
319
+ z.object({
320
+ sessionID: Identifier.schema("session"),
321
+ limit: z.number().optional(),
322
+ }),
323
+ async (input) => {
324
+ const result = [] as MessageV2.WithParts[]
325
+ for await (const msg of MessageV2.stream(input.sessionID)) {
326
+ if (input.limit && result.length >= input.limit) break
327
+ result.push(msg)
328
+ }
329
+ result.reverse()
330
+ return result
331
+ },
332
+ )
333
+
334
+ export async function* list() {
335
+ const project = Instance.project
336
+ for (const item of await Storage.list(["session", project.id])) {
337
+ const session = await Storage.read<Info>(item).catch(() => undefined)
338
+ if (!session) continue
339
+ yield session
340
+ }
341
+ }
342
+
343
+ export const children = fn(Identifier.schema("session"), async (parentID) => {
344
+ const project = Instance.project
345
+ const result = [] as Session.Info[]
346
+ for (const item of await Storage.list(["session", project.id])) {
347
+ const session = await Storage.read<Info>(item).catch(() => undefined)
348
+ if (!session) continue
349
+ if (session.parentID !== parentID) continue
350
+ result.push(session)
351
+ }
352
+ return result
353
+ })
354
+
355
+ export const remove = fn(Identifier.schema("session"), async (sessionID) => {
356
+ const project = Instance.project
357
+ try {
358
+ const session = await get(sessionID)
359
+ for (const child of await children(sessionID)) {
360
+ await remove(child.id)
361
+ }
362
+ await unshare(sessionID).catch(() => {})
363
+ for (const msg of await Storage.list(["message", sessionID])) {
364
+ for (const part of await Storage.list(["part", msg.at(-1)!])) {
365
+ await Storage.remove(part)
366
+ }
367
+ await Storage.remove(msg)
368
+ }
369
+ await Storage.remove(["session", project.id, sessionID])
370
+ Bus.publish(Event.Deleted, {
371
+ info: session,
372
+ })
373
+ } catch (e) {
374
+ log.error(e)
375
+ }
376
+ })
377
+
378
+ export const updateMessage = fn(MessageV2.Info, async (msg) => {
379
+ await Storage.write(["message", msg.sessionID, msg.id], msg)
380
+ Bus.publish(MessageV2.Event.Updated, {
381
+ info: msg,
382
+ })
383
+ return msg
384
+ })
385
+
386
+ export const removeMessage = fn(
387
+ z.object({
388
+ sessionID: Identifier.schema("session"),
389
+ messageID: Identifier.schema("message"),
390
+ }),
391
+ async (input) => {
392
+ await Storage.remove(["message", input.sessionID, input.messageID])
393
+ Bus.publish(MessageV2.Event.Removed, {
394
+ sessionID: input.sessionID,
395
+ messageID: input.messageID,
396
+ })
397
+ return input.messageID
398
+ },
399
+ )
400
+
401
+ export const removePart = fn(
402
+ z.object({
403
+ sessionID: Identifier.schema("session"),
404
+ messageID: Identifier.schema("message"),
405
+ partID: Identifier.schema("part"),
406
+ }),
407
+ async (input) => {
408
+ await Storage.remove(["part", input.messageID, input.partID])
409
+ Bus.publish(MessageV2.Event.PartRemoved, {
410
+ sessionID: input.sessionID,
411
+ messageID: input.messageID,
412
+ partID: input.partID,
413
+ })
414
+ return input.partID
415
+ },
416
+ )
417
+
418
+ const UpdatePartInput = z.union([
419
+ MessageV2.Part,
420
+ z.object({
421
+ part: MessageV2.TextPart,
422
+ delta: z.string(),
423
+ }),
424
+ z.object({
425
+ part: MessageV2.ReasoningPart,
426
+ delta: z.string(),
427
+ }),
428
+ ])
429
+
430
+ export const updatePart = fn(UpdatePartInput, async (input) => {
431
+ const part = "delta" in input ? input.part : input
432
+ const delta = "delta" in input ? input.delta : undefined
433
+ await Storage.write(["part", part.messageID, part.id], part)
434
+ Bus.publish(MessageV2.Event.PartUpdated, {
435
+ part,
436
+ delta,
437
+ })
438
+ return part
439
+ })
440
+
441
+ export const getUsage = fn(
442
+ z.object({
443
+ model: z.custom<Provider.Model>(),
444
+ usage: z.custom<LanguageModelV2Usage>(),
445
+ metadata: z.custom<ProviderMetadata>().optional(),
446
+ }),
447
+ (input) => {
448
+ const safe = (value: number) => {
449
+ if (!Number.isFinite(value)) return 0
450
+ return value
451
+ }
452
+ const inputTokens = safe(input.usage.inputTokens ?? 0)
453
+ const outputTokens = safe(input.usage.outputTokens ?? 0)
454
+ const reasoningTokens = safe(input.usage.reasoningTokens ?? 0)
455
+
456
+ const cacheReadInputTokens = safe(input.usage.cachedInputTokens ?? 0)
457
+ const cacheWriteInputTokens = safe(
458
+ (input.metadata?.["anthropic"]?.["cacheCreationInputTokens"] ??
459
+ // @ts-expect-error
460
+ input.metadata?.["bedrock"]?.["usage"]?.["cacheWriteInputTokens"] ??
461
+ // @ts-expect-error
462
+ input.metadata?.["venice"]?.["usage"]?.["cacheCreationInputTokens"] ??
463
+ 0) as number,
464
+ )
465
+
466
+ // OpenRouter provides inputTokens as the total count of input tokens (including cached).
467
+ // AFAIK other providers (OpenRouter/OpenAI/Gemini etc.) do it the same way e.g. vercel/ai#8794 (comment)
468
+ // Anthropic does it differently though - inputTokens doesn't include cached tokens.
469
+ // It looks like InnoCode's cost calculation assumes all providers return inputTokens the same way Anthropic does (I'm guessing getUsage logic was originally implemented with anthropic), so it's causing incorrect cost calculation for OpenRouter and others.
470
+ const excludesCachedTokens = !!(input.metadata?.["anthropic"] || input.metadata?.["bedrock"])
471
+ const adjustedInputTokens = safe(
472
+ excludesCachedTokens ? inputTokens : inputTokens - cacheReadInputTokens - cacheWriteInputTokens,
473
+ )
474
+
475
+ const total = iife(() => {
476
+ // Anthropic doesn't provide total_tokens, also ai sdk will vastly undercount if we
477
+ // don't compute from components
478
+ if (
479
+ input.model.api.npm === "@ai-sdk/anthropic" ||
480
+ input.model.api.npm === "@ai-sdk/amazon-bedrock" ||
481
+ input.model.api.npm === "@ai-sdk/google-vertex/anthropic"
482
+ ) {
483
+ return adjustedInputTokens + outputTokens + cacheReadInputTokens + cacheWriteInputTokens
484
+ }
485
+ return input.usage.totalTokens
486
+ })
487
+
488
+ const tokens = {
489
+ total,
490
+ input: adjustedInputTokens,
491
+ output: outputTokens,
492
+ reasoning: reasoningTokens,
493
+ cache: {
494
+ write: cacheWriteInputTokens,
495
+ read: cacheReadInputTokens,
496
+ },
497
+ }
498
+
499
+ const costInfo =
500
+ input.model.cost?.experimentalOver200K && tokens.input + tokens.cache.read > 200_000
501
+ ? input.model.cost.experimentalOver200K
502
+ : input.model.cost
503
+ return {
504
+ cost: safe(
505
+ new Decimal(0)
506
+ .add(new Decimal(tokens.input).mul(costInfo?.input ?? 0).div(1_000_000))
507
+ .add(new Decimal(tokens.output).mul(costInfo?.output ?? 0).div(1_000_000))
508
+ .add(new Decimal(tokens.cache.read).mul(costInfo?.cache?.read ?? 0).div(1_000_000))
509
+ .add(new Decimal(tokens.cache.write).mul(costInfo?.cache?.write ?? 0).div(1_000_000))
510
+ // TODO: update models.dev to have better pricing model, for now:
511
+ // charge reasoning tokens at the same rate as output tokens
512
+ .add(new Decimal(tokens.reasoning).mul(costInfo?.output ?? 0).div(1_000_000))
513
+ .toNumber(),
514
+ ),
515
+ tokens,
516
+ }
517
+ },
518
+ )
519
+
520
+ export class BusyError extends Error {
521
+ constructor(public readonly sessionID: string) {
522
+ super(`Session ${sessionID} is busy`)
523
+ }
524
+ }
525
+
526
+ export const initialize = fn(
527
+ z.object({
528
+ sessionID: Identifier.schema("session"),
529
+ modelID: z.string(),
530
+ providerID: z.string(),
531
+ messageID: Identifier.schema("message"),
532
+ }),
533
+ async (input) => {
534
+ await SessionPrompt.command({
535
+ sessionID: input.sessionID,
536
+ messageID: input.messageID,
537
+ model: input.providerID + "/" + input.modelID,
538
+ command: Command.Default.INIT,
539
+ arguments: "",
540
+ })
541
+ },
542
+ )
543
+ }