opencode-v2 1.1.53

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 (439) hide show
  1. package/AGENTS.md +27 -0
  2. package/Dockerfile +18 -0
  3. package/README.md +15 -0
  4. package/bin/opencode +84 -0
  5. package/bunfig.toml +5 -0
  6. package/package.json +126 -0
  7. package/parsers-config.ts +253 -0
  8. package/script/build.ts +193 -0
  9. package/script/postinstall.mjs +125 -0
  10. package/script/publish.ts +181 -0
  11. package/script/schema.ts +47 -0
  12. package/script/seed-e2e.ts +50 -0
  13. package/src/acp/README.md +164 -0
  14. package/src/acp/agent.ts +1676 -0
  15. package/src/acp/session.ts +117 -0
  16. package/src/acp/types.ts +23 -0
  17. package/src/agent/agent.ts +414 -0
  18. package/src/agent/generate.txt +75 -0
  19. package/src/agent/prompt/compaction.txt +12 -0
  20. package/src/agent/prompt/explore.txt +18 -0
  21. package/src/agent/prompt/summary.txt +11 -0
  22. package/src/agent/prompt/title.txt +44 -0
  23. package/src/auth/index.ts +70 -0
  24. package/src/bun/index.ts +137 -0
  25. package/src/bun/registry.ts +48 -0
  26. package/src/bus/bus-event.ts +43 -0
  27. package/src/bus/global.ts +10 -0
  28. package/src/bus/index.ts +105 -0
  29. package/src/cli/bootstrap.ts +17 -0
  30. package/src/cli/cmd/acp.ts +70 -0
  31. package/src/cli/cmd/agent.ts +257 -0
  32. package/src/cli/cmd/auth.ts +400 -0
  33. package/src/cli/cmd/cmd.ts +7 -0
  34. package/src/cli/cmd/debug/agent.ts +167 -0
  35. package/src/cli/cmd/debug/config.ts +16 -0
  36. package/src/cli/cmd/debug/file.ts +97 -0
  37. package/src/cli/cmd/debug/index.ts +48 -0
  38. package/src/cli/cmd/debug/lsp.ts +52 -0
  39. package/src/cli/cmd/debug/ripgrep.ts +87 -0
  40. package/src/cli/cmd/debug/scrap.ts +16 -0
  41. package/src/cli/cmd/debug/skill.ts +16 -0
  42. package/src/cli/cmd/debug/snapshot.ts +52 -0
  43. package/src/cli/cmd/export.ts +88 -0
  44. package/src/cli/cmd/generate.ts +38 -0
  45. package/src/cli/cmd/github.ts +1540 -0
  46. package/src/cli/cmd/import.ts +147 -0
  47. package/src/cli/cmd/mcp.ts +755 -0
  48. package/src/cli/cmd/models.ts +77 -0
  49. package/src/cli/cmd/pr.ts +112 -0
  50. package/src/cli/cmd/run.ts +617 -0
  51. package/src/cli/cmd/serve.ts +20 -0
  52. package/src/cli/cmd/session.ts +135 -0
  53. package/src/cli/cmd/stats.ts +426 -0
  54. package/src/cli/cmd/tui/app.tsx +801 -0
  55. package/src/cli/cmd/tui/attach.ts +52 -0
  56. package/src/cli/cmd/tui/component/border.tsx +21 -0
  57. package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
  58. package/src/cli/cmd/tui/component/dialog-command.tsx +148 -0
  59. package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
  60. package/src/cli/cmd/tui/component/dialog-model.tsx +234 -0
  61. package/src/cli/cmd/tui/component/dialog-provider.tsx +266 -0
  62. package/src/cli/cmd/tui/component/dialog-session-list.tsx +108 -0
  63. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
  64. package/src/cli/cmd/tui/component/dialog-skill.tsx +36 -0
  65. package/src/cli/cmd/tui/component/dialog-stash.tsx +87 -0
  66. package/src/cli/cmd/tui/component/dialog-status.tsx +177 -0
  67. package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
  68. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
  69. package/src/cli/cmd/tui/component/logo.tsx +85 -0
  70. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +666 -0
  71. package/src/cli/cmd/tui/component/prompt/frecency.tsx +89 -0
  72. package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
  73. package/src/cli/cmd/tui/component/prompt/index.tsx +1132 -0
  74. package/src/cli/cmd/tui/component/prompt/stash.tsx +101 -0
  75. package/src/cli/cmd/tui/component/spinner.tsx +24 -0
  76. package/src/cli/cmd/tui/component/textarea-keybindings.ts +73 -0
  77. package/src/cli/cmd/tui/component/tips.tsx +153 -0
  78. package/src/cli/cmd/tui/component/todo-item.tsx +32 -0
  79. package/src/cli/cmd/tui/context/args.tsx +15 -0
  80. package/src/cli/cmd/tui/context/directory.ts +13 -0
  81. package/src/cli/cmd/tui/context/exit.tsx +52 -0
  82. package/src/cli/cmd/tui/context/helper.tsx +25 -0
  83. package/src/cli/cmd/tui/context/keybind.tsx +100 -0
  84. package/src/cli/cmd/tui/context/kv.tsx +52 -0
  85. package/src/cli/cmd/tui/context/local.tsx +409 -0
  86. package/src/cli/cmd/tui/context/prompt.tsx +18 -0
  87. package/src/cli/cmd/tui/context/route.tsx +46 -0
  88. package/src/cli/cmd/tui/context/sdk.tsx +101 -0
  89. package/src/cli/cmd/tui/context/sync.tsx +470 -0
  90. package/src/cli/cmd/tui/context/theme/aura.json +69 -0
  91. package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
  92. package/src/cli/cmd/tui/context/theme/carbonfox.json +248 -0
  93. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +233 -0
  94. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
  95. package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
  96. package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
  97. package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
  98. package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
  99. package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
  100. package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
  101. package/src/cli/cmd/tui/context/theme/github.json +233 -0
  102. package/src/cli/cmd/tui/context/theme/gruvbox.json +242 -0
  103. package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
  104. package/src/cli/cmd/tui/context/theme/lucent-orng.json +237 -0
  105. package/src/cli/cmd/tui/context/theme/material.json +235 -0
  106. package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
  107. package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
  108. package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
  109. package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
  110. package/src/cli/cmd/tui/context/theme/nord.json +223 -0
  111. package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
  112. package/src/cli/cmd/tui/context/theme/orng.json +249 -0
  113. package/src/cli/cmd/tui/context/theme/osaka-jade.json +93 -0
  114. package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
  115. package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
  116. package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
  117. package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
  118. package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
  119. package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
  120. package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
  121. package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
  122. package/src/cli/cmd/tui/context/theme.tsx +1152 -0
  123. package/src/cli/cmd/tui/event.ts +48 -0
  124. package/src/cli/cmd/tui/routes/home.tsx +140 -0
  125. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +64 -0
  126. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +109 -0
  127. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +26 -0
  128. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
  129. package/src/cli/cmd/tui/routes/session/footer.tsx +91 -0
  130. package/src/cli/cmd/tui/routes/session/header.tsx +142 -0
  131. package/src/cli/cmd/tui/routes/session/index.tsx +2126 -0
  132. package/src/cli/cmd/tui/routes/session/permission.tsx +508 -0
  133. package/src/cli/cmd/tui/routes/session/question.tsx +466 -0
  134. package/src/cli/cmd/tui/routes/session/sidebar.tsx +313 -0
  135. package/src/cli/cmd/tui/thread.ts +175 -0
  136. package/src/cli/cmd/tui/ui/dialog-alert.tsx +68 -0
  137. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +93 -0
  138. package/src/cli/cmd/tui/ui/dialog-export-options.tsx +215 -0
  139. package/src/cli/cmd/tui/ui/dialog-help.tsx +49 -0
  140. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +88 -0
  141. package/src/cli/cmd/tui/ui/dialog-select.tsx +399 -0
  142. package/src/cli/cmd/tui/ui/dialog.tsx +167 -0
  143. package/src/cli/cmd/tui/ui/link.tsx +28 -0
  144. package/src/cli/cmd/tui/ui/spinner.ts +368 -0
  145. package/src/cli/cmd/tui/ui/toast.tsx +100 -0
  146. package/src/cli/cmd/tui/util/clipboard.ts +159 -0
  147. package/src/cli/cmd/tui/util/editor.ts +32 -0
  148. package/src/cli/cmd/tui/util/signal.ts +7 -0
  149. package/src/cli/cmd/tui/util/terminal.ts +114 -0
  150. package/src/cli/cmd/tui/util/transcript.ts +98 -0
  151. package/src/cli/cmd/tui/worker.ts +152 -0
  152. package/src/cli/cmd/uninstall.ts +357 -0
  153. package/src/cli/cmd/upgrade.ts +73 -0
  154. package/src/cli/cmd/web.ts +81 -0
  155. package/src/cli/error.ts +57 -0
  156. package/src/cli/logo.ts +6 -0
  157. package/src/cli/network.ts +60 -0
  158. package/src/cli/ui.ts +113 -0
  159. package/src/cli/upgrade.ts +25 -0
  160. package/src/command/index.ts +150 -0
  161. package/src/command/template/initialize.txt +10 -0
  162. package/src/command/template/review.txt +99 -0
  163. package/src/config/config.ts +1477 -0
  164. package/src/config/markdown.ts +98 -0
  165. package/src/env/index.ts +28 -0
  166. package/src/file/ignore.ts +83 -0
  167. package/src/file/index.ts +583 -0
  168. package/src/file/ripgrep.ts +375 -0
  169. package/src/file/time.ts +69 -0
  170. package/src/file/watcher.ts +127 -0
  171. package/src/flag/flag.ts +97 -0
  172. package/src/format/formatter.ts +366 -0
  173. package/src/format/index.ts +137 -0
  174. package/src/global/index.ts +55 -0
  175. package/src/id/id.ts +83 -0
  176. package/src/ide/index.ts +76 -0
  177. package/src/index.ts +159 -0
  178. package/src/installation/index.ts +246 -0
  179. package/src/lsp/client.ts +252 -0
  180. package/src/lsp/index.ts +485 -0
  181. package/src/lsp/language.ts +119 -0
  182. package/src/lsp/server.ts +2046 -0
  183. package/src/mcp/auth.ts +132 -0
  184. package/src/mcp/index.ts +934 -0
  185. package/src/mcp/oauth-callback.ts +200 -0
  186. package/src/mcp/oauth-provider.ts +154 -0
  187. package/src/patch/index.ts +680 -0
  188. package/src/permission/arity.ts +163 -0
  189. package/src/permission/index.ts +210 -0
  190. package/src/permission/next.ts +280 -0
  191. package/src/plugin/codex.ts +624 -0
  192. package/src/plugin/copilot.ts +327 -0
  193. package/src/plugin/index.ts +138 -0
  194. package/src/project/bootstrap.ts +35 -0
  195. package/src/project/instance.ts +114 -0
  196. package/src/project/project.ts +371 -0
  197. package/src/project/state.ts +70 -0
  198. package/src/project/vcs.ts +76 -0
  199. package/src/provider/auth.ts +147 -0
  200. package/src/provider/models.ts +133 -0
  201. package/src/provider/provider.ts +1262 -0
  202. package/src/provider/sdk/copilot/README.md +5 -0
  203. package/src/provider/sdk/copilot/chat/convert-to-openai-compatible-chat-messages.ts +164 -0
  204. package/src/provider/sdk/copilot/chat/get-response-metadata.ts +15 -0
  205. package/src/provider/sdk/copilot/chat/map-openai-compatible-finish-reason.ts +17 -0
  206. package/src/provider/sdk/copilot/chat/openai-compatible-api-types.ts +64 -0
  207. package/src/provider/sdk/copilot/chat/openai-compatible-chat-language-model.ts +780 -0
  208. package/src/provider/sdk/copilot/chat/openai-compatible-chat-options.ts +28 -0
  209. package/src/provider/sdk/copilot/chat/openai-compatible-metadata-extractor.ts +44 -0
  210. package/src/provider/sdk/copilot/chat/openai-compatible-prepare-tools.ts +87 -0
  211. package/src/provider/sdk/copilot/copilot-provider.ts +100 -0
  212. package/src/provider/sdk/copilot/index.ts +2 -0
  213. package/src/provider/sdk/copilot/openai-compatible-error.ts +27 -0
  214. package/src/provider/sdk/copilot/responses/convert-to-openai-responses-input.ts +303 -0
  215. package/src/provider/sdk/copilot/responses/map-openai-responses-finish-reason.ts +22 -0
  216. package/src/provider/sdk/copilot/responses/openai-config.ts +18 -0
  217. package/src/provider/sdk/copilot/responses/openai-error.ts +22 -0
  218. package/src/provider/sdk/copilot/responses/openai-responses-api-types.ts +207 -0
  219. package/src/provider/sdk/copilot/responses/openai-responses-language-model.ts +1732 -0
  220. package/src/provider/sdk/copilot/responses/openai-responses-prepare-tools.ts +177 -0
  221. package/src/provider/sdk/copilot/responses/openai-responses-settings.ts +1 -0
  222. package/src/provider/sdk/copilot/responses/tool/code-interpreter.ts +88 -0
  223. package/src/provider/sdk/copilot/responses/tool/file-search.ts +128 -0
  224. package/src/provider/sdk/copilot/responses/tool/image-generation.ts +115 -0
  225. package/src/provider/sdk/copilot/responses/tool/local-shell.ts +65 -0
  226. package/src/provider/sdk/copilot/responses/tool/web-search-preview.ts +104 -0
  227. package/src/provider/sdk/copilot/responses/tool/web-search.ts +103 -0
  228. package/src/provider/transform.ts +828 -0
  229. package/src/pty/index.ts +250 -0
  230. package/src/question/index.ts +171 -0
  231. package/src/scheduler/index.ts +61 -0
  232. package/src/server/error.ts +36 -0
  233. package/src/server/event.ts +7 -0
  234. package/src/server/mdns.ts +60 -0
  235. package/src/server/routes/config.ts +92 -0
  236. package/src/server/routes/experimental.ts +208 -0
  237. package/src/server/routes/file.ts +197 -0
  238. package/src/server/routes/global.ts +183 -0
  239. package/src/server/routes/mcp.ts +225 -0
  240. package/src/server/routes/permission.ts +68 -0
  241. package/src/server/routes/project.ts +82 -0
  242. package/src/server/routes/provider.ts +165 -0
  243. package/src/server/routes/pty.ts +169 -0
  244. package/src/server/routes/question.ts +98 -0
  245. package/src/server/routes/session.ts +939 -0
  246. package/src/server/routes/tui.ts +379 -0
  247. package/src/server/server.ts +613 -0
  248. package/src/session/compaction.ts +226 -0
  249. package/src/session/index.ts +524 -0
  250. package/src/session/instruction.ts +197 -0
  251. package/src/session/llm.ts +289 -0
  252. package/src/session/message-v2.ts +802 -0
  253. package/src/session/message.ts +189 -0
  254. package/src/session/processor.ts +407 -0
  255. package/src/session/prompt/agent.txt +43 -0
  256. package/src/session/prompt/anthropic-20250930.txt +166 -0
  257. package/src/session/prompt/anthropic.txt +105 -0
  258. package/src/session/prompt/beast.txt +147 -0
  259. package/src/session/prompt/build-switch.txt +5 -0
  260. package/src/session/prompt/codex_header.txt +79 -0
  261. package/src/session/prompt/copilot-gpt-5.txt +143 -0
  262. package/src/session/prompt/gemini.txt +155 -0
  263. package/src/session/prompt/max-steps.txt +16 -0
  264. package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
  265. package/src/session/prompt/plan.txt +26 -0
  266. package/src/session/prompt/qwen.txt +109 -0
  267. package/src/session/prompt/research.txt +81 -0
  268. package/src/session/prompt/trinity.txt +97 -0
  269. package/src/session/prompt.ts +1952 -0
  270. package/src/session/retry.ts +97 -0
  271. package/src/session/revert.ts +121 -0
  272. package/src/session/status.ts +76 -0
  273. package/src/session/summary.ts +217 -0
  274. package/src/session/system.ts +54 -0
  275. package/src/session/todo.ts +37 -0
  276. package/src/share/share-next.ts +200 -0
  277. package/src/share/share.ts +92 -0
  278. package/src/shell/shell.ts +67 -0
  279. package/src/skill/discovery.ts +97 -0
  280. package/src/skill/index.ts +1 -0
  281. package/src/skill/skill.ts +188 -0
  282. package/src/snapshot/index.ts +255 -0
  283. package/src/storage/storage.ts +227 -0
  284. package/src/tool/agent-enter.txt +1 -0
  285. package/src/tool/agent-exit.txt +1 -0
  286. package/src/tool/agent.ts +237 -0
  287. package/src/tool/apply_patch.ts +281 -0
  288. package/src/tool/apply_patch.txt +33 -0
  289. package/src/tool/bash.ts +269 -0
  290. package/src/tool/bash.txt +115 -0
  291. package/src/tool/batch.ts +175 -0
  292. package/src/tool/batch.txt +24 -0
  293. package/src/tool/chat-enter.txt +15 -0
  294. package/src/tool/chat-exit.txt +7 -0
  295. package/src/tool/chat.ts +217 -0
  296. package/src/tool/codesearch.ts +132 -0
  297. package/src/tool/codesearch.txt +12 -0
  298. package/src/tool/edit.ts +655 -0
  299. package/src/tool/edit.txt +10 -0
  300. package/src/tool/external-directory.ts +32 -0
  301. package/src/tool/glob.ts +78 -0
  302. package/src/tool/glob.txt +6 -0
  303. package/src/tool/grep.ts +147 -0
  304. package/src/tool/grep.txt +8 -0
  305. package/src/tool/invalid.ts +17 -0
  306. package/src/tool/ls.ts +121 -0
  307. package/src/tool/ls.txt +1 -0
  308. package/src/tool/lsp.ts +96 -0
  309. package/src/tool/lsp.txt +19 -0
  310. package/src/tool/multiedit.ts +46 -0
  311. package/src/tool/multiedit.txt +41 -0
  312. package/src/tool/plan-enter.txt +14 -0
  313. package/src/tool/plan-exit.txt +13 -0
  314. package/src/tool/plan.ts +130 -0
  315. package/src/tool/question.ts +33 -0
  316. package/src/tool/question.txt +10 -0
  317. package/src/tool/read.ts +211 -0
  318. package/src/tool/read.txt +12 -0
  319. package/src/tool/registry.ts +167 -0
  320. package/src/tool/research-enter.txt +1 -0
  321. package/src/tool/research-exit.txt +1 -0
  322. package/src/tool/research.ts +134 -0
  323. package/src/tool/skill.ts +123 -0
  324. package/src/tool/task.ts +165 -0
  325. package/src/tool/task.txt +60 -0
  326. package/src/tool/todo.ts +53 -0
  327. package/src/tool/todoread.txt +14 -0
  328. package/src/tool/todowrite.txt +167 -0
  329. package/src/tool/tool.ts +89 -0
  330. package/src/tool/truncation.ts +106 -0
  331. package/src/tool/webfetch.ts +186 -0
  332. package/src/tool/webfetch.txt +13 -0
  333. package/src/tool/websearch.ts +150 -0
  334. package/src/tool/websearch.txt +14 -0
  335. package/src/tool/write.ts +85 -0
  336. package/src/tool/write.txt +8 -0
  337. package/src/util/abort.ts +35 -0
  338. package/src/util/archive.ts +16 -0
  339. package/src/util/color.ts +19 -0
  340. package/src/util/context.ts +25 -0
  341. package/src/util/defer.ts +12 -0
  342. package/src/util/eventloop.ts +20 -0
  343. package/src/util/filesystem.ts +93 -0
  344. package/src/util/fn.ts +11 -0
  345. package/src/util/format.ts +20 -0
  346. package/src/util/iife.ts +3 -0
  347. package/src/util/keybind.ts +103 -0
  348. package/src/util/lazy.ts +18 -0
  349. package/src/util/locale.ts +81 -0
  350. package/src/util/lock.ts +98 -0
  351. package/src/util/log.ts +180 -0
  352. package/src/util/proxied.ts +3 -0
  353. package/src/util/queue.ts +32 -0
  354. package/src/util/rpc.ts +66 -0
  355. package/src/util/scrap.ts +10 -0
  356. package/src/util/signal.ts +12 -0
  357. package/src/util/timeout.ts +14 -0
  358. package/src/util/token.ts +7 -0
  359. package/src/util/wildcard.ts +56 -0
  360. package/src/worktree/index.ts +574 -0
  361. package/sst-env.d.ts +9 -0
  362. package/test/acp/agent-interface.test.ts +51 -0
  363. package/test/acp/event-subscription.test.ts +436 -0
  364. package/test/agent/agent.test.ts +675 -0
  365. package/test/bun.test.ts +53 -0
  366. package/test/cli/github-action.test.ts +161 -0
  367. package/test/cli/github-remote.test.ts +80 -0
  368. package/test/cli/import.test.ts +38 -0
  369. package/test/cli/tui/transcript.test.ts +322 -0
  370. package/test/config/agent-color.test.ts +71 -0
  371. package/test/config/config.test.ts +1802 -0
  372. package/test/config/fixtures/empty-frontmatter.md +4 -0
  373. package/test/config/fixtures/frontmatter.md +28 -0
  374. package/test/config/fixtures/markdown-header.md +11 -0
  375. package/test/config/fixtures/no-frontmatter.md +1 -0
  376. package/test/config/fixtures/weird-model-id.md +13 -0
  377. package/test/config/markdown.test.ts +228 -0
  378. package/test/file/ignore.test.ts +10 -0
  379. package/test/file/path-traversal.test.ts +198 -0
  380. package/test/file/ripgrep.test.ts +39 -0
  381. package/test/fixture/fixture.ts +45 -0
  382. package/test/fixture/lsp/fake-lsp-server.js +77 -0
  383. package/test/ide/ide.test.ts +82 -0
  384. package/test/keybind.test.ts +421 -0
  385. package/test/lsp/client.test.ts +95 -0
  386. package/test/mcp/headers.test.ts +153 -0
  387. package/test/mcp/oauth-browser.test.ts +249 -0
  388. package/test/memory/abort-leak.test.ts +136 -0
  389. package/test/patch/patch.test.ts +348 -0
  390. package/test/permission/arity.test.ts +33 -0
  391. package/test/permission/next.test.ts +690 -0
  392. package/test/permission-task.test.ts +319 -0
  393. package/test/plugin/auth-override.test.ts +44 -0
  394. package/test/plugin/codex.test.ts +123 -0
  395. package/test/preload.ts +63 -0
  396. package/test/project/project.test.ts +120 -0
  397. package/test/provider/amazon-bedrock.test.ts +445 -0
  398. package/test/provider/copilot/convert-to-copilot-messages.test.ts +523 -0
  399. package/test/provider/copilot/copilot-chat-model.test.ts +592 -0
  400. package/test/provider/gitlab-duo.test.ts +262 -0
  401. package/test/provider/provider.test.ts +2129 -0
  402. package/test/provider/transform.test.ts +2022 -0
  403. package/test/question/question.test.ts +300 -0
  404. package/test/scheduler.test.ts +73 -0
  405. package/test/server/session-list.test.ts +39 -0
  406. package/test/server/session-select.test.ts +78 -0
  407. package/test/session/compaction.test.ts +293 -0
  408. package/test/session/instruction.test.ts +170 -0
  409. package/test/session/llm.test.ts +691 -0
  410. package/test/session/message-v2.test.ts +786 -0
  411. package/test/session/prompt-missing-file.test.ts +53 -0
  412. package/test/session/prompt-special-chars.test.ts +56 -0
  413. package/test/session/prompt-variant.test.ts +60 -0
  414. package/test/session/retry.test.ts +179 -0
  415. package/test/session/revert-compact.test.ts +285 -0
  416. package/test/session/session.test.ts +71 -0
  417. package/test/skill/discovery.test.ts +60 -0
  418. package/test/skill/skill.test.ts +388 -0
  419. package/test/snapshot/snapshot.test.ts +1040 -0
  420. package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
  421. package/test/tool/apply_patch.test.ts +559 -0
  422. package/test/tool/bash.test.ts +399 -0
  423. package/test/tool/external-directory.test.ts +127 -0
  424. package/test/tool/fixtures/large-image.png +0 -0
  425. package/test/tool/fixtures/models-api.json +38413 -0
  426. package/test/tool/grep.test.ts +110 -0
  427. package/test/tool/question.test.ts +107 -0
  428. package/test/tool/read.test.ts +358 -0
  429. package/test/tool/registry.test.ts +122 -0
  430. package/test/tool/skill.test.ts +112 -0
  431. package/test/tool/truncation.test.ts +159 -0
  432. package/test/util/filesystem.test.ts +39 -0
  433. package/test/util/format.test.ts +59 -0
  434. package/test/util/iife.test.ts +36 -0
  435. package/test/util/lazy.test.ts +50 -0
  436. package/test/util/lock.test.ts +72 -0
  437. package/test/util/timeout.test.ts +21 -0
  438. package/test/util/wildcard.test.ts +75 -0
  439. package/tsconfig.json +16 -0
@@ -0,0 +1,1152 @@
1
+ import { SyntaxStyle, RGBA, type TerminalColors } from "@opentui/core"
2
+ import path from "path"
3
+ import { createEffect, createMemo, onMount } from "solid-js"
4
+ import { useSync } from "@tui/context/sync"
5
+ import { createSimpleContext } from "./helper"
6
+ import aura from "./theme/aura.json" with { type: "json" }
7
+ import ayu from "./theme/ayu.json" with { type: "json" }
8
+ import catppuccin from "./theme/catppuccin.json" with { type: "json" }
9
+ import catppuccinFrappe from "./theme/catppuccin-frappe.json" with { type: "json" }
10
+ import catppuccinMacchiato from "./theme/catppuccin-macchiato.json" with { type: "json" }
11
+ import cobalt2 from "./theme/cobalt2.json" with { type: "json" }
12
+ import cursor from "./theme/cursor.json" with { type: "json" }
13
+ import dracula from "./theme/dracula.json" with { type: "json" }
14
+ import everforest from "./theme/everforest.json" with { type: "json" }
15
+ import flexoki from "./theme/flexoki.json" with { type: "json" }
16
+ import github from "./theme/github.json" with { type: "json" }
17
+ import gruvbox from "./theme/gruvbox.json" with { type: "json" }
18
+ import kanagawa from "./theme/kanagawa.json" with { type: "json" }
19
+ import material from "./theme/material.json" with { type: "json" }
20
+ import matrix from "./theme/matrix.json" with { type: "json" }
21
+ import mercury from "./theme/mercury.json" with { type: "json" }
22
+ import monokai from "./theme/monokai.json" with { type: "json" }
23
+ import nightowl from "./theme/nightowl.json" with { type: "json" }
24
+ import nord from "./theme/nord.json" with { type: "json" }
25
+ import osakaJade from "./theme/osaka-jade.json" with { type: "json" }
26
+ import onedark from "./theme/one-dark.json" with { type: "json" }
27
+ import opencode from "./theme/opencode.json" with { type: "json" }
28
+ import orng from "./theme/orng.json" with { type: "json" }
29
+ import lucentOrng from "./theme/lucent-orng.json" with { type: "json" }
30
+ import palenight from "./theme/palenight.json" with { type: "json" }
31
+ import rosepine from "./theme/rosepine.json" with { type: "json" }
32
+ import solarized from "./theme/solarized.json" with { type: "json" }
33
+ import synthwave84 from "./theme/synthwave84.json" with { type: "json" }
34
+ import tokyonight from "./theme/tokyonight.json" with { type: "json" }
35
+ import vercel from "./theme/vercel.json" with { type: "json" }
36
+ import vesper from "./theme/vesper.json" with { type: "json" }
37
+ import zenburn from "./theme/zenburn.json" with { type: "json" }
38
+ import carbonfox from "./theme/carbonfox.json" with { type: "json" }
39
+ import { useKV } from "./kv"
40
+ import { useRenderer } from "@opentui/solid"
41
+ import { createStore, produce } from "solid-js/store"
42
+ import { Global } from "@/global"
43
+ import { Filesystem } from "@/util/filesystem"
44
+
45
+ type ThemeColors = {
46
+ primary: RGBA
47
+ secondary: RGBA
48
+ accent: RGBA
49
+ error: RGBA
50
+ warning: RGBA
51
+ success: RGBA
52
+ info: RGBA
53
+ text: RGBA
54
+ textMuted: RGBA
55
+ selectedListItemText: RGBA
56
+ background: RGBA
57
+ backgroundPanel: RGBA
58
+ backgroundElement: RGBA
59
+ backgroundMenu: RGBA
60
+ border: RGBA
61
+ borderActive: RGBA
62
+ borderSubtle: RGBA
63
+ diffAdded: RGBA
64
+ diffRemoved: RGBA
65
+ diffContext: RGBA
66
+ diffHunkHeader: RGBA
67
+ diffHighlightAdded: RGBA
68
+ diffHighlightRemoved: RGBA
69
+ diffAddedBg: RGBA
70
+ diffRemovedBg: RGBA
71
+ diffContextBg: RGBA
72
+ diffLineNumber: RGBA
73
+ diffAddedLineNumberBg: RGBA
74
+ diffRemovedLineNumberBg: RGBA
75
+ markdownText: RGBA
76
+ markdownHeading: RGBA
77
+ markdownLink: RGBA
78
+ markdownLinkText: RGBA
79
+ markdownCode: RGBA
80
+ markdownBlockQuote: RGBA
81
+ markdownEmph: RGBA
82
+ markdownStrong: RGBA
83
+ markdownHorizontalRule: RGBA
84
+ markdownListItem: RGBA
85
+ markdownListEnumeration: RGBA
86
+ markdownImage: RGBA
87
+ markdownImageText: RGBA
88
+ markdownCodeBlock: RGBA
89
+ syntaxComment: RGBA
90
+ syntaxKeyword: RGBA
91
+ syntaxFunction: RGBA
92
+ syntaxVariable: RGBA
93
+ syntaxString: RGBA
94
+ syntaxNumber: RGBA
95
+ syntaxType: RGBA
96
+ syntaxOperator: RGBA
97
+ syntaxPunctuation: RGBA
98
+ }
99
+
100
+ type Theme = ThemeColors & {
101
+ _hasSelectedListItemText: boolean
102
+ thinkingOpacity: number
103
+ }
104
+
105
+ export function selectedForeground(theme: Theme, bg?: RGBA): RGBA {
106
+ // If theme explicitly defines selectedListItemText, use it
107
+ if (theme._hasSelectedListItemText) {
108
+ return theme.selectedListItemText
109
+ }
110
+
111
+ // For transparent backgrounds, calculate contrast based on the actual bg (or fallback to primary)
112
+ if (theme.background.a === 0) {
113
+ const targetColor = bg ?? theme.primary
114
+ const { r, g, b } = targetColor
115
+ const luminance = 0.299 * r + 0.587 * g + 0.114 * b
116
+ return luminance > 0.5 ? RGBA.fromInts(0, 0, 0) : RGBA.fromInts(255, 255, 255)
117
+ }
118
+
119
+ // Fall back to background color
120
+ return theme.background
121
+ }
122
+
123
+ type HexColor = `#${string}`
124
+ type RefName = string
125
+ type Variant = {
126
+ dark: HexColor | RefName
127
+ light: HexColor | RefName
128
+ }
129
+ type ColorValue = HexColor | RefName | Variant | RGBA
130
+ type ThemeJson = {
131
+ $schema?: string
132
+ defs?: Record<string, HexColor | RefName>
133
+ theme: Omit<Record<keyof ThemeColors, ColorValue>, "selectedListItemText" | "backgroundMenu"> & {
134
+ selectedListItemText?: ColorValue
135
+ backgroundMenu?: ColorValue
136
+ thinkingOpacity?: number
137
+ }
138
+ }
139
+
140
+ export const DEFAULT_THEMES: Record<string, ThemeJson> = {
141
+ aura,
142
+ ayu,
143
+ catppuccin,
144
+ ["catppuccin-frappe"]: catppuccinFrappe,
145
+ ["catppuccin-macchiato"]: catppuccinMacchiato,
146
+ cobalt2,
147
+ cursor,
148
+ dracula,
149
+ everforest,
150
+ flexoki,
151
+ github,
152
+ gruvbox,
153
+ kanagawa,
154
+ material,
155
+ matrix,
156
+ mercury,
157
+ monokai,
158
+ nightowl,
159
+ nord,
160
+ ["one-dark"]: onedark,
161
+ ["osaka-jade"]: osakaJade,
162
+ opencode,
163
+ orng,
164
+ ["lucent-orng"]: lucentOrng,
165
+ palenight,
166
+ rosepine,
167
+ solarized,
168
+ synthwave84,
169
+ tokyonight,
170
+ vesper,
171
+ vercel,
172
+ zenburn,
173
+ carbonfox,
174
+ }
175
+
176
+ function resolveTheme(theme: ThemeJson, mode: "dark" | "light") {
177
+ const defs = theme.defs ?? {}
178
+ function resolveColor(c: ColorValue): RGBA {
179
+ if (c instanceof RGBA) return c
180
+ if (typeof c === "string") {
181
+ if (c === "transparent" || c === "none") return RGBA.fromInts(0, 0, 0, 0)
182
+
183
+ if (c.startsWith("#")) return RGBA.fromHex(c)
184
+
185
+ if (defs[c] != null) {
186
+ return resolveColor(defs[c])
187
+ } else if (theme.theme[c as keyof ThemeColors] !== undefined) {
188
+ return resolveColor(theme.theme[c as keyof ThemeColors]!)
189
+ } else {
190
+ throw new Error(`Color reference "${c}" not found in defs or theme`)
191
+ }
192
+ }
193
+ if (typeof c === "number") {
194
+ return ansiToRgba(c)
195
+ }
196
+ return resolveColor(c[mode])
197
+ }
198
+
199
+ const resolved = Object.fromEntries(
200
+ Object.entries(theme.theme)
201
+ .filter(([key]) => key !== "selectedListItemText" && key !== "backgroundMenu" && key !== "thinkingOpacity")
202
+ .map(([key, value]) => {
203
+ return [key, resolveColor(value as ColorValue)]
204
+ }),
205
+ ) as Partial<ThemeColors>
206
+
207
+ // Handle selectedListItemText separately since it's optional
208
+ const hasSelectedListItemText = theme.theme.selectedListItemText !== undefined
209
+ if (hasSelectedListItemText) {
210
+ resolved.selectedListItemText = resolveColor(theme.theme.selectedListItemText!)
211
+ } else {
212
+ // Backward compatibility: if selectedListItemText is not defined, use background color
213
+ // This preserves the current behavior for all existing themes
214
+ resolved.selectedListItemText = resolved.background
215
+ }
216
+
217
+ // Handle backgroundMenu - optional with fallback to backgroundElement
218
+ if (theme.theme.backgroundMenu !== undefined) {
219
+ resolved.backgroundMenu = resolveColor(theme.theme.backgroundMenu)
220
+ } else {
221
+ resolved.backgroundMenu = resolved.backgroundElement
222
+ }
223
+
224
+ // Handle thinkingOpacity - optional with default of 0.6
225
+ const thinkingOpacity = theme.theme.thinkingOpacity ?? 0.6
226
+
227
+ return {
228
+ ...resolved,
229
+ _hasSelectedListItemText: hasSelectedListItemText,
230
+ thinkingOpacity,
231
+ } as Theme
232
+ }
233
+
234
+ function ansiToRgba(code: number): RGBA {
235
+ // Standard ANSI colors (0-15)
236
+ if (code < 16) {
237
+ const ansiColors = [
238
+ "#000000", // Black
239
+ "#800000", // Red
240
+ "#008000", // Green
241
+ "#808000", // Yellow
242
+ "#000080", // Blue
243
+ "#800080", // Magenta
244
+ "#008080", // Cyan
245
+ "#c0c0c0", // White
246
+ "#808080", // Bright Black
247
+ "#ff0000", // Bright Red
248
+ "#00ff00", // Bright Green
249
+ "#ffff00", // Bright Yellow
250
+ "#0000ff", // Bright Blue
251
+ "#ff00ff", // Bright Magenta
252
+ "#00ffff", // Bright Cyan
253
+ "#ffffff", // Bright White
254
+ ]
255
+ return RGBA.fromHex(ansiColors[code] ?? "#000000")
256
+ }
257
+
258
+ // 6x6x6 Color Cube (16-231)
259
+ if (code < 232) {
260
+ const index = code - 16
261
+ const b = index % 6
262
+ const g = Math.floor(index / 6) % 6
263
+ const r = Math.floor(index / 36)
264
+
265
+ const val = (x: number) => (x === 0 ? 0 : x * 40 + 55)
266
+ return RGBA.fromInts(val(r), val(g), val(b))
267
+ }
268
+
269
+ // Grayscale Ramp (232-255)
270
+ if (code < 256) {
271
+ const gray = (code - 232) * 10 + 8
272
+ return RGBA.fromInts(gray, gray, gray)
273
+ }
274
+
275
+ // Fallback for invalid codes
276
+ return RGBA.fromInts(0, 0, 0)
277
+ }
278
+
279
+ export const { use: useTheme, provider: ThemeProvider } = createSimpleContext({
280
+ name: "Theme",
281
+ init: (props: { mode: "dark" | "light" }) => {
282
+ const sync = useSync()
283
+ const kv = useKV()
284
+ const [store, setStore] = createStore({
285
+ themes: DEFAULT_THEMES,
286
+ mode: kv.get("theme_mode", props.mode),
287
+ active: (sync.data.config.theme ?? kv.get("theme", "opencode")) as string,
288
+ ready: false,
289
+ })
290
+
291
+ createEffect(() => {
292
+ const theme = sync.data.config.theme
293
+ if (theme) setStore("active", theme)
294
+ })
295
+
296
+ function init() {
297
+ resolveSystemTheme()
298
+ getCustomThemes()
299
+ .then((custom) => {
300
+ setStore(
301
+ produce((draft) => {
302
+ Object.assign(draft.themes, custom)
303
+ }),
304
+ )
305
+ })
306
+ .catch(() => {
307
+ setStore("active", "opencode")
308
+ })
309
+ .finally(() => {
310
+ if (store.active !== "system") {
311
+ setStore("ready", true)
312
+ }
313
+ })
314
+ }
315
+
316
+ onMount(init)
317
+
318
+ function resolveSystemTheme() {
319
+ console.log("resolveSystemTheme")
320
+ renderer
321
+ .getPalette({
322
+ size: 16,
323
+ })
324
+ .then((colors) => {
325
+ console.log(colors.palette)
326
+ if (!colors.palette[0]) {
327
+ if (store.active === "system") {
328
+ setStore(
329
+ produce((draft) => {
330
+ draft.active = "opencode"
331
+ draft.ready = true
332
+ }),
333
+ )
334
+ }
335
+ return
336
+ }
337
+ setStore(
338
+ produce((draft) => {
339
+ draft.themes.system = generateSystem(colors, store.mode)
340
+ if (store.active === "system") {
341
+ draft.ready = true
342
+ }
343
+ }),
344
+ )
345
+ })
346
+ }
347
+
348
+ const renderer = useRenderer()
349
+ process.on("SIGUSR2", async () => {
350
+ renderer.clearPaletteCache()
351
+ init()
352
+ })
353
+
354
+ const values = createMemo(() => {
355
+ return resolveTheme(store.themes[store.active] ?? store.themes.opencode, store.mode)
356
+ })
357
+
358
+ const syntax = createMemo(() => generateSyntax(values()))
359
+ const subtleSyntax = createMemo(() => generateSubtleSyntax(values()))
360
+
361
+ return {
362
+ theme: new Proxy(values(), {
363
+ get(_target, prop) {
364
+ // @ts-expect-error
365
+ return values()[prop]
366
+ },
367
+ }),
368
+ get selected() {
369
+ return store.active
370
+ },
371
+ all() {
372
+ return store.themes
373
+ },
374
+ syntax,
375
+ subtleSyntax,
376
+ mode() {
377
+ return store.mode
378
+ },
379
+ setMode(mode: "dark" | "light") {
380
+ setStore("mode", mode)
381
+ kv.set("theme_mode", mode)
382
+ },
383
+ set(theme: string) {
384
+ setStore("active", theme)
385
+ kv.set("theme", theme)
386
+ },
387
+ get ready() {
388
+ return store.ready
389
+ },
390
+ }
391
+ },
392
+ })
393
+
394
+ const CUSTOM_THEME_GLOB = new Bun.Glob("themes/*.json")
395
+ async function getCustomThemes() {
396
+ const directories = [
397
+ Global.Path.config,
398
+ ...(await Array.fromAsync(
399
+ Filesystem.up({
400
+ targets: [".opencode"],
401
+ start: process.cwd(),
402
+ }),
403
+ )),
404
+ ]
405
+
406
+ const result: Record<string, ThemeJson> = {}
407
+ for (const dir of directories) {
408
+ for await (const item of CUSTOM_THEME_GLOB.scan({
409
+ absolute: true,
410
+ followSymlinks: true,
411
+ dot: true,
412
+ cwd: dir,
413
+ })) {
414
+ const name = path.basename(item, ".json")
415
+ result[name] = await Bun.file(item).json()
416
+ }
417
+ }
418
+ return result
419
+ }
420
+
421
+ export function tint(base: RGBA, overlay: RGBA, alpha: number): RGBA {
422
+ const r = base.r + (overlay.r - base.r) * alpha
423
+ const g = base.g + (overlay.g - base.g) * alpha
424
+ const b = base.b + (overlay.b - base.b) * alpha
425
+ return RGBA.fromInts(Math.round(r * 255), Math.round(g * 255), Math.round(b * 255))
426
+ }
427
+
428
+ function generateSystem(colors: TerminalColors, mode: "dark" | "light"): ThemeJson {
429
+ const bg = RGBA.fromHex(colors.defaultBackground ?? colors.palette[0]!)
430
+ const fg = RGBA.fromHex(colors.defaultForeground ?? colors.palette[7]!)
431
+ const transparent = RGBA.fromInts(0, 0, 0, 0)
432
+ const isDark = mode == "dark"
433
+
434
+ const col = (i: number) => {
435
+ const value = colors.palette[i]
436
+ if (value) return RGBA.fromHex(value)
437
+ return ansiToRgba(i)
438
+ }
439
+
440
+ // Generate gray scale based on terminal background
441
+ const grays = generateGrayScale(bg, isDark)
442
+ const textMuted = generateMutedTextColor(bg, isDark)
443
+
444
+ // ANSI color references
445
+ const ansiColors = {
446
+ black: col(0),
447
+ red: col(1),
448
+ green: col(2),
449
+ yellow: col(3),
450
+ blue: col(4),
451
+ magenta: col(5),
452
+ cyan: col(6),
453
+ white: col(7),
454
+ redBright: col(9),
455
+ greenBright: col(10),
456
+ }
457
+
458
+ const diffAlpha = isDark ? 0.22 : 0.14
459
+ const diffAddedBg = tint(bg, ansiColors.green, diffAlpha)
460
+ const diffRemovedBg = tint(bg, ansiColors.red, diffAlpha)
461
+ const diffAddedLineNumberBg = tint(grays[3], ansiColors.green, diffAlpha)
462
+ const diffRemovedLineNumberBg = tint(grays[3], ansiColors.red, diffAlpha)
463
+
464
+ return {
465
+ theme: {
466
+ // Primary colors using ANSI
467
+ primary: ansiColors.cyan,
468
+ secondary: ansiColors.magenta,
469
+ accent: ansiColors.cyan,
470
+
471
+ // Status colors using ANSI
472
+ error: ansiColors.red,
473
+ warning: ansiColors.yellow,
474
+ success: ansiColors.green,
475
+ info: ansiColors.cyan,
476
+
477
+ // Text colors
478
+ text: fg,
479
+ textMuted,
480
+ selectedListItemText: bg,
481
+
482
+ // Background colors - use transparent to respect terminal transparency
483
+ background: transparent,
484
+ backgroundPanel: grays[2],
485
+ backgroundElement: grays[3],
486
+ backgroundMenu: grays[3],
487
+
488
+ // Border colors
489
+ borderSubtle: grays[6],
490
+ border: grays[7],
491
+ borderActive: grays[8],
492
+
493
+ // Diff colors
494
+ diffAdded: ansiColors.green,
495
+ diffRemoved: ansiColors.red,
496
+ diffContext: grays[7],
497
+ diffHunkHeader: grays[7],
498
+ diffHighlightAdded: ansiColors.greenBright,
499
+ diffHighlightRemoved: ansiColors.redBright,
500
+ diffAddedBg,
501
+ diffRemovedBg,
502
+ diffContextBg: grays[1],
503
+ diffLineNumber: grays[6],
504
+ diffAddedLineNumberBg,
505
+ diffRemovedLineNumberBg,
506
+
507
+ // Markdown colors
508
+ markdownText: fg,
509
+ markdownHeading: fg,
510
+ markdownLink: ansiColors.blue,
511
+ markdownLinkText: ansiColors.cyan,
512
+ markdownCode: ansiColors.green,
513
+ markdownBlockQuote: ansiColors.yellow,
514
+ markdownEmph: ansiColors.yellow,
515
+ markdownStrong: fg,
516
+ markdownHorizontalRule: grays[7],
517
+ markdownListItem: ansiColors.blue,
518
+ markdownListEnumeration: ansiColors.cyan,
519
+ markdownImage: ansiColors.blue,
520
+ markdownImageText: ansiColors.cyan,
521
+ markdownCodeBlock: fg,
522
+
523
+ // Syntax colors
524
+ syntaxComment: textMuted,
525
+ syntaxKeyword: ansiColors.magenta,
526
+ syntaxFunction: ansiColors.blue,
527
+ syntaxVariable: fg,
528
+ syntaxString: ansiColors.green,
529
+ syntaxNumber: ansiColors.yellow,
530
+ syntaxType: ansiColors.cyan,
531
+ syntaxOperator: ansiColors.cyan,
532
+ syntaxPunctuation: fg,
533
+ },
534
+ }
535
+ }
536
+
537
+ function generateGrayScale(bg: RGBA, isDark: boolean): Record<number, RGBA> {
538
+ const grays: Record<number, RGBA> = {}
539
+
540
+ // RGBA stores floats in range 0-1, convert to 0-255
541
+ const bgR = bg.r * 255
542
+ const bgG = bg.g * 255
543
+ const bgB = bg.b * 255
544
+
545
+ const luminance = 0.299 * bgR + 0.587 * bgG + 0.114 * bgB
546
+
547
+ for (let i = 1; i <= 12; i++) {
548
+ const factor = i / 12.0
549
+
550
+ let grayValue: number
551
+ let newR: number
552
+ let newG: number
553
+ let newB: number
554
+
555
+ if (isDark) {
556
+ if (luminance < 10) {
557
+ grayValue = Math.floor(factor * 0.4 * 255)
558
+ newR = grayValue
559
+ newG = grayValue
560
+ newB = grayValue
561
+ } else {
562
+ const newLum = luminance + (255 - luminance) * factor * 0.4
563
+
564
+ const ratio = newLum / luminance
565
+ newR = Math.min(bgR * ratio, 255)
566
+ newG = Math.min(bgG * ratio, 255)
567
+ newB = Math.min(bgB * ratio, 255)
568
+ }
569
+ } else {
570
+ if (luminance > 245) {
571
+ grayValue = Math.floor(255 - factor * 0.4 * 255)
572
+ newR = grayValue
573
+ newG = grayValue
574
+ newB = grayValue
575
+ } else {
576
+ const newLum = luminance * (1 - factor * 0.4)
577
+
578
+ const ratio = newLum / luminance
579
+ newR = Math.max(bgR * ratio, 0)
580
+ newG = Math.max(bgG * ratio, 0)
581
+ newB = Math.max(bgB * ratio, 0)
582
+ }
583
+ }
584
+
585
+ grays[i] = RGBA.fromInts(Math.floor(newR), Math.floor(newG), Math.floor(newB))
586
+ }
587
+
588
+ return grays
589
+ }
590
+
591
+ function generateMutedTextColor(bg: RGBA, isDark: boolean): RGBA {
592
+ // RGBA stores floats in range 0-1, convert to 0-255
593
+ const bgR = bg.r * 255
594
+ const bgG = bg.g * 255
595
+ const bgB = bg.b * 255
596
+
597
+ const bgLum = 0.299 * bgR + 0.587 * bgG + 0.114 * bgB
598
+
599
+ let grayValue: number
600
+
601
+ if (isDark) {
602
+ if (bgLum < 10) {
603
+ // Very dark/black background
604
+ grayValue = 180 // #b4b4b4
605
+ } else {
606
+ // Scale up for lighter dark backgrounds
607
+ grayValue = Math.min(Math.floor(160 + bgLum * 0.3), 200)
608
+ }
609
+ } else {
610
+ if (bgLum > 245) {
611
+ // Very light/white background
612
+ grayValue = 75 // #4b4b4b
613
+ } else {
614
+ // Scale down for darker light backgrounds
615
+ grayValue = Math.max(Math.floor(100 - (255 - bgLum) * 0.2), 60)
616
+ }
617
+ }
618
+
619
+ return RGBA.fromInts(grayValue, grayValue, grayValue)
620
+ }
621
+
622
+ function generateSyntax(theme: Theme) {
623
+ return SyntaxStyle.fromTheme(getSyntaxRules(theme))
624
+ }
625
+
626
+ function generateSubtleSyntax(theme: Theme) {
627
+ const rules = getSyntaxRules(theme)
628
+ return SyntaxStyle.fromTheme(
629
+ rules.map((rule) => {
630
+ if (rule.style.foreground) {
631
+ const fg = rule.style.foreground
632
+ return {
633
+ ...rule,
634
+ style: {
635
+ ...rule.style,
636
+ foreground: RGBA.fromInts(
637
+ Math.round(fg.r * 255),
638
+ Math.round(fg.g * 255),
639
+ Math.round(fg.b * 255),
640
+ Math.round(theme.thinkingOpacity * 255),
641
+ ),
642
+ },
643
+ }
644
+ }
645
+ return rule
646
+ }),
647
+ )
648
+ }
649
+
650
+ function getSyntaxRules(theme: Theme) {
651
+ return [
652
+ {
653
+ scope: ["default"],
654
+ style: {
655
+ foreground: theme.text,
656
+ },
657
+ },
658
+ {
659
+ scope: ["prompt"],
660
+ style: {
661
+ foreground: theme.accent,
662
+ },
663
+ },
664
+ {
665
+ scope: ["extmark.file"],
666
+ style: {
667
+ foreground: theme.warning,
668
+ bold: true,
669
+ },
670
+ },
671
+ {
672
+ scope: ["extmark.agent"],
673
+ style: {
674
+ foreground: theme.secondary,
675
+ bold: true,
676
+ },
677
+ },
678
+ {
679
+ scope: ["extmark.paste"],
680
+ style: {
681
+ foreground: theme.background,
682
+ background: theme.warning,
683
+ bold: true,
684
+ },
685
+ },
686
+ {
687
+ scope: ["comment"],
688
+ style: {
689
+ foreground: theme.syntaxComment,
690
+ italic: true,
691
+ },
692
+ },
693
+ {
694
+ scope: ["comment.documentation"],
695
+ style: {
696
+ foreground: theme.syntaxComment,
697
+ italic: true,
698
+ },
699
+ },
700
+ {
701
+ scope: ["string", "symbol"],
702
+ style: {
703
+ foreground: theme.syntaxString,
704
+ },
705
+ },
706
+ {
707
+ scope: ["number", "boolean"],
708
+ style: {
709
+ foreground: theme.syntaxNumber,
710
+ },
711
+ },
712
+ {
713
+ scope: ["character.special"],
714
+ style: {
715
+ foreground: theme.syntaxString,
716
+ },
717
+ },
718
+ {
719
+ scope: ["keyword.return", "keyword.conditional", "keyword.repeat", "keyword.coroutine"],
720
+ style: {
721
+ foreground: theme.syntaxKeyword,
722
+ italic: true,
723
+ },
724
+ },
725
+ {
726
+ scope: ["keyword.type"],
727
+ style: {
728
+ foreground: theme.syntaxType,
729
+ bold: true,
730
+ italic: true,
731
+ },
732
+ },
733
+ {
734
+ scope: ["keyword.function", "function.method"],
735
+ style: {
736
+ foreground: theme.syntaxFunction,
737
+ },
738
+ },
739
+ {
740
+ scope: ["keyword"],
741
+ style: {
742
+ foreground: theme.syntaxKeyword,
743
+ italic: true,
744
+ },
745
+ },
746
+ {
747
+ scope: ["keyword.import"],
748
+ style: {
749
+ foreground: theme.syntaxKeyword,
750
+ },
751
+ },
752
+ {
753
+ scope: ["operator", "keyword.operator", "punctuation.delimiter"],
754
+ style: {
755
+ foreground: theme.syntaxOperator,
756
+ },
757
+ },
758
+ {
759
+ scope: ["keyword.conditional.ternary"],
760
+ style: {
761
+ foreground: theme.syntaxOperator,
762
+ },
763
+ },
764
+ {
765
+ scope: ["variable", "variable.parameter", "function.method.call", "function.call"],
766
+ style: {
767
+ foreground: theme.syntaxVariable,
768
+ },
769
+ },
770
+ {
771
+ scope: ["variable.member", "function", "constructor"],
772
+ style: {
773
+ foreground: theme.syntaxFunction,
774
+ },
775
+ },
776
+ {
777
+ scope: ["type", "module"],
778
+ style: {
779
+ foreground: theme.syntaxType,
780
+ },
781
+ },
782
+ {
783
+ scope: ["constant"],
784
+ style: {
785
+ foreground: theme.syntaxNumber,
786
+ },
787
+ },
788
+ {
789
+ scope: ["property"],
790
+ style: {
791
+ foreground: theme.syntaxVariable,
792
+ },
793
+ },
794
+ {
795
+ scope: ["class"],
796
+ style: {
797
+ foreground: theme.syntaxType,
798
+ },
799
+ },
800
+ {
801
+ scope: ["parameter"],
802
+ style: {
803
+ foreground: theme.syntaxVariable,
804
+ },
805
+ },
806
+ {
807
+ scope: ["punctuation", "punctuation.bracket"],
808
+ style: {
809
+ foreground: theme.syntaxPunctuation,
810
+ },
811
+ },
812
+ {
813
+ scope: ["variable.builtin", "type.builtin", "function.builtin", "module.builtin", "constant.builtin"],
814
+ style: {
815
+ foreground: theme.error,
816
+ },
817
+ },
818
+ {
819
+ scope: ["variable.super"],
820
+ style: {
821
+ foreground: theme.error,
822
+ },
823
+ },
824
+ {
825
+ scope: ["string.escape", "string.regexp"],
826
+ style: {
827
+ foreground: theme.syntaxKeyword,
828
+ },
829
+ },
830
+ {
831
+ scope: ["keyword.directive"],
832
+ style: {
833
+ foreground: theme.syntaxKeyword,
834
+ italic: true,
835
+ },
836
+ },
837
+ {
838
+ scope: ["punctuation.special"],
839
+ style: {
840
+ foreground: theme.syntaxOperator,
841
+ },
842
+ },
843
+ {
844
+ scope: ["keyword.modifier"],
845
+ style: {
846
+ foreground: theme.syntaxKeyword,
847
+ italic: true,
848
+ },
849
+ },
850
+ {
851
+ scope: ["keyword.exception"],
852
+ style: {
853
+ foreground: theme.syntaxKeyword,
854
+ italic: true,
855
+ },
856
+ },
857
+ // Markdown specific styles
858
+ {
859
+ scope: ["markup.heading"],
860
+ style: {
861
+ foreground: theme.markdownHeading,
862
+ bold: true,
863
+ },
864
+ },
865
+ {
866
+ scope: ["markup.heading.1"],
867
+ style: {
868
+ foreground: theme.markdownHeading,
869
+ bold: true,
870
+ },
871
+ },
872
+ {
873
+ scope: ["markup.heading.2"],
874
+ style: {
875
+ foreground: theme.markdownHeading,
876
+ bold: true,
877
+ },
878
+ },
879
+ {
880
+ scope: ["markup.heading.3"],
881
+ style: {
882
+ foreground: theme.markdownHeading,
883
+ bold: true,
884
+ },
885
+ },
886
+ {
887
+ scope: ["markup.heading.4"],
888
+ style: {
889
+ foreground: theme.markdownHeading,
890
+ bold: true,
891
+ },
892
+ },
893
+ {
894
+ scope: ["markup.heading.5"],
895
+ style: {
896
+ foreground: theme.markdownHeading,
897
+ bold: true,
898
+ },
899
+ },
900
+ {
901
+ scope: ["markup.heading.6"],
902
+ style: {
903
+ foreground: theme.markdownHeading,
904
+ bold: true,
905
+ },
906
+ },
907
+ {
908
+ scope: ["markup.bold", "markup.strong"],
909
+ style: {
910
+ foreground: theme.markdownStrong,
911
+ bold: true,
912
+ },
913
+ },
914
+ {
915
+ scope: ["markup.italic"],
916
+ style: {
917
+ foreground: theme.markdownEmph,
918
+ italic: true,
919
+ },
920
+ },
921
+ {
922
+ scope: ["markup.list"],
923
+ style: {
924
+ foreground: theme.markdownListItem,
925
+ },
926
+ },
927
+ {
928
+ scope: ["markup.quote"],
929
+ style: {
930
+ foreground: theme.markdownBlockQuote,
931
+ italic: true,
932
+ },
933
+ },
934
+ {
935
+ scope: ["markup.raw", "markup.raw.block"],
936
+ style: {
937
+ foreground: theme.markdownCode,
938
+ },
939
+ },
940
+ {
941
+ scope: ["markup.raw.inline"],
942
+ style: {
943
+ foreground: theme.markdownCode,
944
+ background: theme.background,
945
+ },
946
+ },
947
+ {
948
+ scope: ["markup.link"],
949
+ style: {
950
+ foreground: theme.markdownLink,
951
+ underline: true,
952
+ },
953
+ },
954
+ {
955
+ scope: ["markup.link.label"],
956
+ style: {
957
+ foreground: theme.markdownLinkText,
958
+ underline: true,
959
+ },
960
+ },
961
+ {
962
+ scope: ["markup.link.url"],
963
+ style: {
964
+ foreground: theme.markdownLink,
965
+ underline: true,
966
+ },
967
+ },
968
+ {
969
+ scope: ["label"],
970
+ style: {
971
+ foreground: theme.markdownLinkText,
972
+ },
973
+ },
974
+ {
975
+ scope: ["spell", "nospell"],
976
+ style: {
977
+ foreground: theme.text,
978
+ },
979
+ },
980
+ {
981
+ scope: ["conceal"],
982
+ style: {
983
+ foreground: theme.textMuted,
984
+ },
985
+ },
986
+ // Additional common highlight groups
987
+ {
988
+ scope: ["string.special", "string.special.url"],
989
+ style: {
990
+ foreground: theme.markdownLink,
991
+ underline: true,
992
+ },
993
+ },
994
+ {
995
+ scope: ["character"],
996
+ style: {
997
+ foreground: theme.syntaxString,
998
+ },
999
+ },
1000
+ {
1001
+ scope: ["float"],
1002
+ style: {
1003
+ foreground: theme.syntaxNumber,
1004
+ },
1005
+ },
1006
+ {
1007
+ scope: ["comment.error"],
1008
+ style: {
1009
+ foreground: theme.error,
1010
+ italic: true,
1011
+ bold: true,
1012
+ },
1013
+ },
1014
+ {
1015
+ scope: ["comment.warning"],
1016
+ style: {
1017
+ foreground: theme.warning,
1018
+ italic: true,
1019
+ bold: true,
1020
+ },
1021
+ },
1022
+ {
1023
+ scope: ["comment.todo", "comment.note"],
1024
+ style: {
1025
+ foreground: theme.info,
1026
+ italic: true,
1027
+ bold: true,
1028
+ },
1029
+ },
1030
+ {
1031
+ scope: ["namespace"],
1032
+ style: {
1033
+ foreground: theme.syntaxType,
1034
+ },
1035
+ },
1036
+ {
1037
+ scope: ["field"],
1038
+ style: {
1039
+ foreground: theme.syntaxVariable,
1040
+ },
1041
+ },
1042
+ {
1043
+ scope: ["type.definition"],
1044
+ style: {
1045
+ foreground: theme.syntaxType,
1046
+ bold: true,
1047
+ },
1048
+ },
1049
+ {
1050
+ scope: ["keyword.export"],
1051
+ style: {
1052
+ foreground: theme.syntaxKeyword,
1053
+ },
1054
+ },
1055
+ {
1056
+ scope: ["attribute", "annotation"],
1057
+ style: {
1058
+ foreground: theme.warning,
1059
+ },
1060
+ },
1061
+ {
1062
+ scope: ["tag"],
1063
+ style: {
1064
+ foreground: theme.error,
1065
+ },
1066
+ },
1067
+ {
1068
+ scope: ["tag.attribute"],
1069
+ style: {
1070
+ foreground: theme.syntaxKeyword,
1071
+ },
1072
+ },
1073
+ {
1074
+ scope: ["tag.delimiter"],
1075
+ style: {
1076
+ foreground: theme.syntaxOperator,
1077
+ },
1078
+ },
1079
+ {
1080
+ scope: ["markup.strikethrough"],
1081
+ style: {
1082
+ foreground: theme.textMuted,
1083
+ },
1084
+ },
1085
+ {
1086
+ scope: ["markup.underline"],
1087
+ style: {
1088
+ foreground: theme.text,
1089
+ underline: true,
1090
+ },
1091
+ },
1092
+ {
1093
+ scope: ["markup.list.checked"],
1094
+ style: {
1095
+ foreground: theme.success,
1096
+ },
1097
+ },
1098
+ {
1099
+ scope: ["markup.list.unchecked"],
1100
+ style: {
1101
+ foreground: theme.textMuted,
1102
+ },
1103
+ },
1104
+ {
1105
+ scope: ["diff.plus"],
1106
+ style: {
1107
+ foreground: theme.diffAdded,
1108
+ background: theme.diffAddedBg,
1109
+ },
1110
+ },
1111
+ {
1112
+ scope: ["diff.minus"],
1113
+ style: {
1114
+ foreground: theme.diffRemoved,
1115
+ background: theme.diffRemovedBg,
1116
+ },
1117
+ },
1118
+ {
1119
+ scope: ["diff.delta"],
1120
+ style: {
1121
+ foreground: theme.diffContext,
1122
+ background: theme.diffContextBg,
1123
+ },
1124
+ },
1125
+ {
1126
+ scope: ["error"],
1127
+ style: {
1128
+ foreground: theme.error,
1129
+ bold: true,
1130
+ },
1131
+ },
1132
+ {
1133
+ scope: ["warning"],
1134
+ style: {
1135
+ foreground: theme.warning,
1136
+ bold: true,
1137
+ },
1138
+ },
1139
+ {
1140
+ scope: ["info"],
1141
+ style: {
1142
+ foreground: theme.info,
1143
+ },
1144
+ },
1145
+ {
1146
+ scope: ["debug"],
1147
+ style: {
1148
+ foreground: theme.textMuted,
1149
+ },
1150
+ },
1151
+ ]
1152
+ }