jonsoc 1.1.43

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 (420) hide show
  1. package/AGENTS.md +27 -0
  2. package/Dockerfile +18 -0
  3. package/PUBLISHING_GUIDE.md +151 -0
  4. package/README.md +58 -0
  5. package/bin/jonsoc +279 -0
  6. package/bunfig.toml +7 -0
  7. package/package.json +147 -0
  8. package/package.json.placeholder +11 -0
  9. package/parsers-config.ts +253 -0
  10. package/script/build.ts +115 -0
  11. package/script/publish-registries.ts +197 -0
  12. package/script/publish.ts +110 -0
  13. package/script/schema.ts +47 -0
  14. package/script/seed-e2e.ts +50 -0
  15. package/src/acp/README.md +164 -0
  16. package/src/acp/agent.ts +1437 -0
  17. package/src/acp/session.ts +105 -0
  18. package/src/acp/types.ts +22 -0
  19. package/src/agent/agent.ts +347 -0
  20. package/src/agent/generate.txt +75 -0
  21. package/src/agent/prompt/compaction.txt +12 -0
  22. package/src/agent/prompt/explore.txt +18 -0
  23. package/src/agent/prompt/summary.txt +11 -0
  24. package/src/agent/prompt/title.txt +44 -0
  25. package/src/auth/index.ts +73 -0
  26. package/src/brand/index.ts +73 -0
  27. package/src/bun/index.ts +139 -0
  28. package/src/bus/bus-event.ts +43 -0
  29. package/src/bus/global.ts +10 -0
  30. package/src/bus/index.ts +105 -0
  31. package/src/cli/bootstrap.ts +17 -0
  32. package/src/cli/cmd/acp.ts +69 -0
  33. package/src/cli/cmd/agent.ts +257 -0
  34. package/src/cli/cmd/auth.ts +405 -0
  35. package/src/cli/cmd/cmd.ts +7 -0
  36. package/src/cli/cmd/debug/agent.ts +166 -0
  37. package/src/cli/cmd/debug/config.ts +16 -0
  38. package/src/cli/cmd/debug/file.ts +97 -0
  39. package/src/cli/cmd/debug/index.ts +48 -0
  40. package/src/cli/cmd/debug/lsp.ts +52 -0
  41. package/src/cli/cmd/debug/ripgrep.ts +87 -0
  42. package/src/cli/cmd/debug/scrap.ts +16 -0
  43. package/src/cli/cmd/debug/skill.ts +16 -0
  44. package/src/cli/cmd/debug/snapshot.ts +52 -0
  45. package/src/cli/cmd/export.ts +88 -0
  46. package/src/cli/cmd/generate.ts +38 -0
  47. package/src/cli/cmd/github.ts +1548 -0
  48. package/src/cli/cmd/import.ts +99 -0
  49. package/src/cli/cmd/mcp.ts +765 -0
  50. package/src/cli/cmd/models.ts +77 -0
  51. package/src/cli/cmd/pr.ts +112 -0
  52. package/src/cli/cmd/run.ts +395 -0
  53. package/src/cli/cmd/serve.ts +20 -0
  54. package/src/cli/cmd/session.ts +135 -0
  55. package/src/cli/cmd/stats.ts +402 -0
  56. package/src/cli/cmd/tui/app.tsx +923 -0
  57. package/src/cli/cmd/tui/attach.ts +39 -0
  58. package/src/cli/cmd/tui/component/border.tsx +21 -0
  59. package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
  60. package/src/cli/cmd/tui/component/dialog-command.tsx +162 -0
  61. package/src/cli/cmd/tui/component/dialog-error-log.tsx +155 -0
  62. package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
  63. package/src/cli/cmd/tui/component/dialog-model.tsx +234 -0
  64. package/src/cli/cmd/tui/component/dialog-provider.tsx +256 -0
  65. package/src/cli/cmd/tui/component/dialog-session-list.tsx +114 -0
  66. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
  67. package/src/cli/cmd/tui/component/dialog-stash.tsx +87 -0
  68. package/src/cli/cmd/tui/component/dialog-status.tsx +164 -0
  69. package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
  70. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
  71. package/src/cli/cmd/tui/component/dynamic-layout.tsx +86 -0
  72. package/src/cli/cmd/tui/component/inspector-overlay.tsx +247 -0
  73. package/src/cli/cmd/tui/component/logo.tsx +88 -0
  74. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +653 -0
  75. package/src/cli/cmd/tui/component/prompt/frecency.tsx +89 -0
  76. package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
  77. package/src/cli/cmd/tui/component/prompt/index.tsx +1347 -0
  78. package/src/cli/cmd/tui/component/prompt/stash.tsx +101 -0
  79. package/src/cli/cmd/tui/component/textarea-keybindings.ts +73 -0
  80. package/src/cli/cmd/tui/component/tips.tsx +153 -0
  81. package/src/cli/cmd/tui/component/todo-item.tsx +32 -0
  82. package/src/cli/cmd/tui/context/args.tsx +14 -0
  83. package/src/cli/cmd/tui/context/directory.ts +13 -0
  84. package/src/cli/cmd/tui/context/error-log.tsx +56 -0
  85. package/src/cli/cmd/tui/context/exit.tsx +26 -0
  86. package/src/cli/cmd/tui/context/helper.tsx +25 -0
  87. package/src/cli/cmd/tui/context/inspector.tsx +57 -0
  88. package/src/cli/cmd/tui/context/keybind.tsx +108 -0
  89. package/src/cli/cmd/tui/context/kv.tsx +53 -0
  90. package/src/cli/cmd/tui/context/layout.tsx +240 -0
  91. package/src/cli/cmd/tui/context/local.tsx +402 -0
  92. package/src/cli/cmd/tui/context/prompt.tsx +18 -0
  93. package/src/cli/cmd/tui/context/route.tsx +51 -0
  94. package/src/cli/cmd/tui/context/sdk.tsx +94 -0
  95. package/src/cli/cmd/tui/context/sync.tsx +449 -0
  96. package/src/cli/cmd/tui/context/theme/aura.json +69 -0
  97. package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
  98. package/src/cli/cmd/tui/context/theme/carbonfox.json +248 -0
  99. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +233 -0
  100. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
  101. package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
  102. package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
  103. package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
  104. package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
  105. package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
  106. package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
  107. package/src/cli/cmd/tui/context/theme/github.json +233 -0
  108. package/src/cli/cmd/tui/context/theme/gruvbox.json +242 -0
  109. package/src/cli/cmd/tui/context/theme/jonsoc.json +245 -0
  110. package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
  111. package/src/cli/cmd/tui/context/theme/lucent-orng.json +237 -0
  112. package/src/cli/cmd/tui/context/theme/material.json +235 -0
  113. package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
  114. package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
  115. package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
  116. package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
  117. package/src/cli/cmd/tui/context/theme/nord.json +223 -0
  118. package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
  119. package/src/cli/cmd/tui/context/theme/orng.json +249 -0
  120. package/src/cli/cmd/tui/context/theme/osaka-jade.json +93 -0
  121. package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
  122. package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
  123. package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
  124. package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
  125. package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
  126. package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
  127. package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
  128. package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
  129. package/src/cli/cmd/tui/context/theme.tsx +1153 -0
  130. package/src/cli/cmd/tui/event.ts +48 -0
  131. package/src/cli/cmd/tui/hooks/use-command-registry.tsx +184 -0
  132. package/src/cli/cmd/tui/routes/home.tsx +198 -0
  133. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +64 -0
  134. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +109 -0
  135. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +26 -0
  136. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
  137. package/src/cli/cmd/tui/routes/session/footer.tsx +91 -0
  138. package/src/cli/cmd/tui/routes/session/git-commit.tsx +59 -0
  139. package/src/cli/cmd/tui/routes/session/git-history.tsx +122 -0
  140. package/src/cli/cmd/tui/routes/session/header.tsx +185 -0
  141. package/src/cli/cmd/tui/routes/session/index.tsx +2363 -0
  142. package/src/cli/cmd/tui/routes/session/navigator-ui.tsx +214 -0
  143. package/src/cli/cmd/tui/routes/session/navigator.tsx +1124 -0
  144. package/src/cli/cmd/tui/routes/session/panel-explorer.tsx +553 -0
  145. package/src/cli/cmd/tui/routes/session/panel-viewer.tsx +386 -0
  146. package/src/cli/cmd/tui/routes/session/permission.tsx +501 -0
  147. package/src/cli/cmd/tui/routes/session/question.tsx +507 -0
  148. package/src/cli/cmd/tui/routes/session/sidebar.tsx +365 -0
  149. package/src/cli/cmd/tui/routes/session/vcs-diff-viewer.tsx +37 -0
  150. package/src/cli/cmd/tui/routes/ui-settings.tsx +449 -0
  151. package/src/cli/cmd/tui/thread.ts +172 -0
  152. package/src/cli/cmd/tui/ui/dialog-alert.tsx +90 -0
  153. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +83 -0
  154. package/src/cli/cmd/tui/ui/dialog-export-options.tsx +204 -0
  155. package/src/cli/cmd/tui/ui/dialog-help.tsx +38 -0
  156. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +77 -0
  157. package/src/cli/cmd/tui/ui/dialog-select.tsx +384 -0
  158. package/src/cli/cmd/tui/ui/dialog.tsx +170 -0
  159. package/src/cli/cmd/tui/ui/link.tsx +28 -0
  160. package/src/cli/cmd/tui/ui/spinner.ts +375 -0
  161. package/src/cli/cmd/tui/ui/toast.tsx +100 -0
  162. package/src/cli/cmd/tui/util/clipboard.ts +255 -0
  163. package/src/cli/cmd/tui/util/editor.ts +32 -0
  164. package/src/cli/cmd/tui/util/signal.ts +7 -0
  165. package/src/cli/cmd/tui/util/terminal.ts +114 -0
  166. package/src/cli/cmd/tui/util/transcript.ts +98 -0
  167. package/src/cli/cmd/tui/worker.ts +152 -0
  168. package/src/cli/cmd/uninstall.ts +362 -0
  169. package/src/cli/cmd/upgrade.ts +73 -0
  170. package/src/cli/cmd/web.ts +81 -0
  171. package/src/cli/error.ts +57 -0
  172. package/src/cli/network.ts +53 -0
  173. package/src/cli/ui.ts +119 -0
  174. package/src/cli/upgrade.ts +25 -0
  175. package/src/command/index.ts +131 -0
  176. package/src/command/template/initialize.txt +10 -0
  177. package/src/command/template/review.txt +99 -0
  178. package/src/config/config.ts +1404 -0
  179. package/src/config/markdown.ts +93 -0
  180. package/src/env/index.ts +26 -0
  181. package/src/file/ignore.ts +83 -0
  182. package/src/file/index.ts +432 -0
  183. package/src/file/ripgrep.ts +407 -0
  184. package/src/file/time.ts +69 -0
  185. package/src/file/watcher.ts +127 -0
  186. package/src/flag/flag.ts +80 -0
  187. package/src/format/formatter.ts +357 -0
  188. package/src/format/index.ts +137 -0
  189. package/src/global/index.ts +58 -0
  190. package/src/id/id.ts +83 -0
  191. package/src/ide/index.ts +76 -0
  192. package/src/index.ts +208 -0
  193. package/src/installation/index.ts +258 -0
  194. package/src/lsp/client.ts +252 -0
  195. package/src/lsp/index.ts +485 -0
  196. package/src/lsp/language.ts +119 -0
  197. package/src/lsp/server.ts +2046 -0
  198. package/src/mcp/auth.ts +135 -0
  199. package/src/mcp/index.ts +934 -0
  200. package/src/mcp/oauth-callback.ts +200 -0
  201. package/src/mcp/oauth-provider.ts +155 -0
  202. package/src/patch/index.ts +680 -0
  203. package/src/permission/arity.ts +163 -0
  204. package/src/permission/index.ts +210 -0
  205. package/src/permission/next.ts +280 -0
  206. package/src/plugin/codex.ts +500 -0
  207. package/src/plugin/copilot.ts +283 -0
  208. package/src/plugin/index.ts +135 -0
  209. package/src/project/bootstrap.ts +35 -0
  210. package/src/project/instance.ts +91 -0
  211. package/src/project/project.ts +371 -0
  212. package/src/project/state.ts +66 -0
  213. package/src/project/vcs.ts +151 -0
  214. package/src/provider/auth.ts +147 -0
  215. package/src/provider/models-macro.ts +14 -0
  216. package/src/provider/models.ts +114 -0
  217. package/src/provider/provider.ts +1220 -0
  218. package/src/provider/sdk/openai-compatible/src/README.md +5 -0
  219. package/src/provider/sdk/openai-compatible/src/index.ts +2 -0
  220. package/src/provider/sdk/openai-compatible/src/openai-compatible-provider.ts +100 -0
  221. package/src/provider/sdk/openai-compatible/src/responses/convert-to-openai-responses-input.ts +303 -0
  222. package/src/provider/sdk/openai-compatible/src/responses/map-openai-responses-finish-reason.ts +22 -0
  223. package/src/provider/sdk/openai-compatible/src/responses/openai-config.ts +18 -0
  224. package/src/provider/sdk/openai-compatible/src/responses/openai-error.ts +22 -0
  225. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-api-types.ts +207 -0
  226. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-language-model.ts +1732 -0
  227. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-prepare-tools.ts +177 -0
  228. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-settings.ts +1 -0
  229. package/src/provider/sdk/openai-compatible/src/responses/tool/code-interpreter.ts +88 -0
  230. package/src/provider/sdk/openai-compatible/src/responses/tool/file-search.ts +128 -0
  231. package/src/provider/sdk/openai-compatible/src/responses/tool/image-generation.ts +115 -0
  232. package/src/provider/sdk/openai-compatible/src/responses/tool/local-shell.ts +65 -0
  233. package/src/provider/sdk/openai-compatible/src/responses/tool/web-search-preview.ts +104 -0
  234. package/src/provider/sdk/openai-compatible/src/responses/tool/web-search.ts +103 -0
  235. package/src/provider/transform.ts +742 -0
  236. package/src/pty/index.ts +241 -0
  237. package/src/question/index.ts +176 -0
  238. package/src/scheduler/index.ts +61 -0
  239. package/src/server/error.ts +36 -0
  240. package/src/server/event.ts +7 -0
  241. package/src/server/mdns.ts +59 -0
  242. package/src/server/routes/config.ts +92 -0
  243. package/src/server/routes/experimental.ts +208 -0
  244. package/src/server/routes/file.ts +227 -0
  245. package/src/server/routes/global.ts +135 -0
  246. package/src/server/routes/mcp.ts +225 -0
  247. package/src/server/routes/permission.ts +68 -0
  248. package/src/server/routes/project.ts +82 -0
  249. package/src/server/routes/provider.ts +165 -0
  250. package/src/server/routes/pty.ts +169 -0
  251. package/src/server/routes/question.ts +98 -0
  252. package/src/server/routes/session.ts +939 -0
  253. package/src/server/routes/tui.ts +379 -0
  254. package/src/server/server.ts +663 -0
  255. package/src/session/compaction.ts +225 -0
  256. package/src/session/index.ts +498 -0
  257. package/src/session/llm.ts +288 -0
  258. package/src/session/message-v2.ts +740 -0
  259. package/src/session/message.ts +189 -0
  260. package/src/session/processor.ts +406 -0
  261. package/src/session/prompt/anthropic-20250930.txt +168 -0
  262. package/src/session/prompt/anthropic.txt +172 -0
  263. package/src/session/prompt/anthropic_spoof.txt +1 -0
  264. package/src/session/prompt/beast.txt +149 -0
  265. package/src/session/prompt/build-switch.txt +5 -0
  266. package/src/session/prompt/codex_header.txt +81 -0
  267. package/src/session/prompt/copilot-gpt-5.txt +145 -0
  268. package/src/session/prompt/gemini.txt +157 -0
  269. package/src/session/prompt/max-steps.txt +16 -0
  270. package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
  271. package/src/session/prompt/plan.txt +26 -0
  272. package/src/session/prompt/qwen.txt +111 -0
  273. package/src/session/prompt.ts +1815 -0
  274. package/src/session/retry.ts +90 -0
  275. package/src/session/revert.ts +121 -0
  276. package/src/session/status.ts +76 -0
  277. package/src/session/summary.ts +150 -0
  278. package/src/session/system.ts +156 -0
  279. package/src/session/todo.ts +37 -0
  280. package/src/share/share-next.ts +204 -0
  281. package/src/share/share.ts +95 -0
  282. package/src/shell/shell.ts +67 -0
  283. package/src/skill/index.ts +1 -0
  284. package/src/skill/skill.ts +135 -0
  285. package/src/snapshot/index.ts +236 -0
  286. package/src/storage/storage.ts +227 -0
  287. package/src/tool/apply_patch.ts +279 -0
  288. package/src/tool/apply_patch.txt +33 -0
  289. package/src/tool/bash.ts +258 -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/codesearch.ts +132 -0
  294. package/src/tool/codesearch.txt +12 -0
  295. package/src/tool/edit.ts +645 -0
  296. package/src/tool/edit.txt +10 -0
  297. package/src/tool/external-directory.ts +32 -0
  298. package/src/tool/glob.ts +77 -0
  299. package/src/tool/glob.txt +6 -0
  300. package/src/tool/grep.ts +154 -0
  301. package/src/tool/grep.txt +8 -0
  302. package/src/tool/invalid.ts +17 -0
  303. package/src/tool/ls.ts +121 -0
  304. package/src/tool/ls.txt +1 -0
  305. package/src/tool/lsp.ts +96 -0
  306. package/src/tool/lsp.txt +19 -0
  307. package/src/tool/multiedit.ts +46 -0
  308. package/src/tool/multiedit.txt +41 -0
  309. package/src/tool/plan-enter.txt +14 -0
  310. package/src/tool/plan-exit.txt +13 -0
  311. package/src/tool/plan.ts +130 -0
  312. package/src/tool/question.ts +33 -0
  313. package/src/tool/question.txt +10 -0
  314. package/src/tool/read.ts +202 -0
  315. package/src/tool/read.txt +12 -0
  316. package/src/tool/registry.ts +162 -0
  317. package/src/tool/skill.ts +82 -0
  318. package/src/tool/task.ts +188 -0
  319. package/src/tool/task.txt +60 -0
  320. package/src/tool/todo.ts +53 -0
  321. package/src/tool/todoread.txt +14 -0
  322. package/src/tool/todowrite.txt +167 -0
  323. package/src/tool/tool.ts +88 -0
  324. package/src/tool/truncation.ts +106 -0
  325. package/src/tool/webfetch.ts +182 -0
  326. package/src/tool/webfetch.txt +13 -0
  327. package/src/tool/websearch.ts +150 -0
  328. package/src/tool/websearch.txt +14 -0
  329. package/src/tool/write.ts +80 -0
  330. package/src/tool/write.txt +8 -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/queue.ts +32 -0
  346. package/src/util/rpc.ts +66 -0
  347. package/src/util/scrap.ts +10 -0
  348. package/src/util/signal.ts +12 -0
  349. package/src/util/timeout.ts +14 -0
  350. package/src/util/token.ts +7 -0
  351. package/src/util/wildcard.ts +56 -0
  352. package/src/worktree/index.ts +524 -0
  353. package/sst-env.d.ts +9 -0
  354. package/test/acp/agent-interface.test.ts +51 -0
  355. package/test/acp/event-subscription.test.ts +436 -0
  356. package/test/agent/agent.test.ts +638 -0
  357. package/test/bun.test.ts +53 -0
  358. package/test/cli/cmd/tui/fileref.test.ts +30 -0
  359. package/test/cli/github-action.test.ts +129 -0
  360. package/test/cli/github-remote.test.ts +80 -0
  361. package/test/cli/tui/navigator_logic.test.ts +99 -0
  362. package/test/cli/tui/transcript.test.ts +297 -0
  363. package/test/cli/ui.test.ts +80 -0
  364. package/test/config/agent-color.test.ts +66 -0
  365. package/test/config/config.test.ts +1613 -0
  366. package/test/config/fixtures/empty-frontmatter.md +4 -0
  367. package/test/config/fixtures/frontmatter.md +28 -0
  368. package/test/config/fixtures/no-frontmatter.md +1 -0
  369. package/test/config/markdown.test.ts +192 -0
  370. package/test/file/ignore.test.ts +10 -0
  371. package/test/file/path-traversal.test.ts +198 -0
  372. package/test/fixture/fixture.ts +45 -0
  373. package/test/fixture/lsp/fake-lsp-server.js +77 -0
  374. package/test/ide/ide.test.ts +82 -0
  375. package/test/keybind.test.ts +421 -0
  376. package/test/lsp/client.test.ts +95 -0
  377. package/test/mcp/headers.test.ts +153 -0
  378. package/test/mcp/oauth-browser.test.ts +261 -0
  379. package/test/patch/patch.test.ts +348 -0
  380. package/test/permission/arity.test.ts +33 -0
  381. package/test/permission/next.test.ts +690 -0
  382. package/test/permission-task.test.ts +319 -0
  383. package/test/plugin/codex.test.ts +123 -0
  384. package/test/preload.ts +67 -0
  385. package/test/project/project.test.ts +120 -0
  386. package/test/provider/amazon-bedrock.test.ts +268 -0
  387. package/test/provider/gitlab-duo.test.ts +286 -0
  388. package/test/provider/provider.test.ts +2149 -0
  389. package/test/provider/transform.test.ts +1631 -0
  390. package/test/question/question.test.ts +300 -0
  391. package/test/scheduler.test.ts +73 -0
  392. package/test/server/session-list.test.ts +39 -0
  393. package/test/server/session-select.test.ts +78 -0
  394. package/test/session/compaction.test.ts +293 -0
  395. package/test/session/llm.test.ts +90 -0
  396. package/test/session/message-v2.test.ts +786 -0
  397. package/test/session/retry.test.ts +131 -0
  398. package/test/session/revert-compact.test.ts +285 -0
  399. package/test/session/session.test.ts +71 -0
  400. package/test/skill/skill.test.ts +185 -0
  401. package/test/snapshot/snapshot.test.ts +939 -0
  402. package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
  403. package/test/tool/apply_patch.test.ts +499 -0
  404. package/test/tool/bash.test.ts +320 -0
  405. package/test/tool/external-directory.test.ts +126 -0
  406. package/test/tool/fixtures/large-image.png +0 -0
  407. package/test/tool/fixtures/models-api.json +33453 -0
  408. package/test/tool/grep.test.ts +109 -0
  409. package/test/tool/question.test.ts +105 -0
  410. package/test/tool/read.test.ts +332 -0
  411. package/test/tool/registry.test.ts +76 -0
  412. package/test/tool/truncation.test.ts +159 -0
  413. package/test/util/filesystem.test.ts +39 -0
  414. package/test/util/format.test.ts +59 -0
  415. package/test/util/iife.test.ts +36 -0
  416. package/test/util/lazy.test.ts +50 -0
  417. package/test/util/lock.test.ts +72 -0
  418. package/test/util/timeout.test.ts +21 -0
  419. package/test/util/wildcard.test.ts +75 -0
  420. package/tsconfig.json +16 -0
@@ -0,0 +1,923 @@
1
+ import fs from "fs"
2
+ import { render, useKeyboard, useRenderer, useTerminalDimensions } from "@opentui/solid"
3
+ import { Clipboard } from "@tui/util/clipboard"
4
+ import { TextAttributes } from "@opentui/core"
5
+ import { RouteProvider, useRoute } from "@tui/context/route"
6
+ import {
7
+ Switch,
8
+ Match,
9
+ createEffect,
10
+ untrack,
11
+ ErrorBoundary,
12
+ createSignal,
13
+ onMount,
14
+ batch,
15
+ Show,
16
+ on,
17
+ For,
18
+ } from "solid-js"
19
+ import { Installation } from "@/installation"
20
+ import { Flag } from "@/flag/flag"
21
+ import { DialogProvider, useDialog } from "@tui/ui/dialog"
22
+ import { DialogProvider as DialogProviderList } from "@tui/component/dialog-provider"
23
+ import { SDKProvider, useSDK } from "@tui/context/sdk"
24
+ import { SyncProvider, useSync } from "@tui/context/sync"
25
+ import { LocalProvider, useLocal } from "@tui/context/local"
26
+ import { DialogModel, useConnected } from "@tui/component/dialog-model"
27
+ import { DialogMcp } from "@tui/component/dialog-mcp"
28
+ import { DialogStatus } from "@tui/component/dialog-status"
29
+ import { DialogThemeList } from "@tui/component/dialog-theme-list"
30
+ import { DialogHelp } from "./ui/dialog-help"
31
+ import { CommandProvider, useCommandDialog } from "@tui/component/dialog-command"
32
+ import { DialogAgent } from "@tui/component/dialog-agent"
33
+ import { DialogSessionList } from "@tui/component/dialog-session-list"
34
+ import { KeybindProvider } from "@tui/context/keybind"
35
+ import { ThemeProvider, useTheme } from "@tui/context/theme"
36
+ import { Home } from "@tui/routes/home"
37
+ import { Session } from "@tui/routes/session"
38
+ import { UISettings } from "@tui/routes/ui-settings"
39
+ import { LayoutProvider } from "@tui/context/layout"
40
+ import { PromptHistoryProvider } from "./component/prompt/history"
41
+ import { FrecencyProvider } from "./component/prompt/frecency"
42
+ import { PromptStashProvider } from "./component/prompt/stash"
43
+ import { DialogAlert } from "./ui/dialog-alert"
44
+ import { ToastProvider, useToast } from "./ui/toast"
45
+ import { ExitProvider, useExit, EXIT_CODE_RESTART } from "./context/exit"
46
+ import { ErrorLogProvider, useErrorLog, init as initErrorLog, type ErrorEntry } from "./context/error-log"
47
+ import { DialogErrorLog } from "./component/dialog-error-log"
48
+ import { Session as SessionApi } from "@/session"
49
+ import { TuiEvent } from "./event"
50
+ import { KVProvider, useKV } from "./context/kv"
51
+ import { InspectorProvider } from "./context/inspector"
52
+ import { InspectorOverlay } from "./component/inspector-overlay"
53
+ import { Provider } from "@/provider/provider"
54
+ import { ArgsProvider, useArgs, type Args } from "./context/args"
55
+ import open from "open"
56
+ import { writeHeapSnapshot } from "v8"
57
+ import { PromptRefProvider, usePromptRef } from "./context/prompt"
58
+
59
+ async function getTerminalBackgroundColor(): Promise<"dark" | "light"> {
60
+ // can't set raw mode if not a TTY
61
+ if (!process.stdin.isTTY) return "dark"
62
+
63
+ return new Promise((resolve) => {
64
+ let timeout: NodeJS.Timeout
65
+
66
+ const cleanup = () => {
67
+ process.stdin.setRawMode(false)
68
+ process.stdin.removeListener("data", handler)
69
+ clearTimeout(timeout)
70
+ }
71
+
72
+ const handler = (data: Buffer) => {
73
+ const str = data.toString()
74
+ const match = str.match(/\x1b]11;([^\x07\x1b]+)/)
75
+ if (match) {
76
+ cleanup()
77
+ const color = match[1]
78
+ // Parse RGB values from color string
79
+ // Formats: rgb:RR/GG/BB or #RRGGBB or rgb(R,G,B)
80
+ let r = 0,
81
+ g = 0,
82
+ b = 0
83
+
84
+ if (color.startsWith("rgb:")) {
85
+ const parts = color.substring(4).split("/")
86
+ r = parseInt(parts[0], 16) >> 8 // Convert 16-bit to 8-bit
87
+ g = parseInt(parts[1], 16) >> 8 // Convert 16-bit to 8-bit
88
+ b = parseInt(parts[2], 16) >> 8 // Convert 16-bit to 8-bit
89
+ } else if (color.startsWith("#")) {
90
+ r = parseInt(color.substring(1, 3), 16)
91
+ g = parseInt(color.substring(3, 5), 16)
92
+ b = parseInt(color.substring(5, 7), 16)
93
+ } else if (color.startsWith("rgb(")) {
94
+ const parts = color.substring(4, color.length - 1).split(",")
95
+ r = parseInt(parts[0])
96
+ g = parseInt(parts[1])
97
+ b = parseInt(parts[2])
98
+ }
99
+
100
+ // Calculate luminance using relative luminance formula
101
+ const luminance = (0.299 * r + 0.587 * g + 0.114 * b) / 255
102
+
103
+ // Determine if dark or light based on luminance threshold
104
+ resolve(luminance > 0.5 ? "light" : "dark")
105
+ }
106
+ }
107
+
108
+ process.stdin.setRawMode(true)
109
+ process.stdin.on("data", handler)
110
+ process.stdout.write("\x1b]11;?\x07")
111
+
112
+ timeout = setTimeout(() => {
113
+ cleanup()
114
+ resolve("dark")
115
+ }, 1000)
116
+ })
117
+ }
118
+
119
+ import type { EventSource } from "./context/sdk"
120
+
121
+ export function tui(input: {
122
+ url: string
123
+ args: Args
124
+ directory?: string
125
+ fetch?: typeof fetch
126
+ events?: EventSource
127
+ onExit?: () => Promise<void>
128
+ }) {
129
+ const errorLog = initErrorLog()
130
+
131
+ const formatConsoleArg = (arg: unknown) => {
132
+ if (typeof arg === "string") return arg
133
+ if (arg instanceof Error) return arg.message
134
+ return String(arg)
135
+ }
136
+
137
+ // Store original console functions
138
+ const originalConsoleLog = console.log
139
+ const originalConsoleError = console.error
140
+
141
+ // Keep console errors visible for debugging
142
+ console.log = (...args: unknown[]) => {
143
+ originalConsoleLog(...args)
144
+ const message = args.map(formatConsoleArg).join(" ")
145
+ errorLog.add(message, "console")
146
+ }
147
+
148
+ console.error = (...args: unknown[]) => {
149
+ originalConsoleError(...args)
150
+ const error = args.find((arg) => arg instanceof Error)
151
+ const message = args.map(formatConsoleArg).join(" ")
152
+ if (error) {
153
+ const entry = new Error(message)
154
+ if (error.stack) entry.stack = error.stack
155
+ errorLog.add(entry, "console")
156
+ return
157
+ }
158
+ errorLog.add(message, "console")
159
+ }
160
+
161
+ // Capture unhandled exceptions (suppress console output)
162
+ process.on("uncaughtException", (error) => {
163
+ errorLog.add(error, "uncaught")
164
+ // Do NOT rethrow - suppress output
165
+ })
166
+
167
+ // Capture unhandled promise rejections (suppress console output)
168
+ process.on("unhandledRejection", (reason) => {
169
+ const error = reason instanceof Error ? reason : new Error(String(reason))
170
+ errorLog.add(error, "rejection")
171
+ // Do NOT rethrow - suppress output
172
+ })
173
+
174
+ return new Promise<void>(async (resolve) => {
175
+ const mode = await getTerminalBackgroundColor()
176
+ const onExit = async () => {
177
+ await input.onExit?.()
178
+ resolve()
179
+ }
180
+
181
+ render(
182
+ () => {
183
+ return (
184
+ <ErrorLogProvider value={errorLog}>
185
+ <ErrorBoundary
186
+ fallback={(error, reset) => <ErrorComponent error={error} reset={reset} onExit={onExit} mode={mode} />}
187
+ >
188
+ <ArgsProvider {...input.args}>
189
+ <ExitProvider onExit={onExit}>
190
+ <KVProvider>
191
+ <LayoutProvider>
192
+ <ToastProvider>
193
+ <RouteProvider>
194
+ <SDKProvider
195
+ url={input.url}
196
+ directory={input.directory}
197
+ fetch={input.fetch}
198
+ events={input.events}
199
+ >
200
+ <SyncProvider>
201
+ <ThemeProvider mode={mode}>
202
+ <InspectorProvider>
203
+ <LocalProvider>
204
+ <KeybindProvider>
205
+ <PromptStashProvider>
206
+ <DialogProvider>
207
+ <CommandProvider>
208
+ <FrecencyProvider>
209
+ <PromptHistoryProvider>
210
+ <PromptRefProvider>
211
+ <App />
212
+ </PromptRefProvider>
213
+ </PromptHistoryProvider>
214
+ </FrecencyProvider>
215
+ </CommandProvider>
216
+ </DialogProvider>
217
+ </PromptStashProvider>
218
+ </KeybindProvider>
219
+ </LocalProvider>
220
+ </InspectorProvider>
221
+ </ThemeProvider>
222
+ </SyncProvider>
223
+ </SDKProvider>
224
+ </RouteProvider>
225
+ </ToastProvider>
226
+ </LayoutProvider>
227
+ </KVProvider>
228
+ </ExitProvider>
229
+ </ArgsProvider>
230
+ </ErrorBoundary>
231
+ </ErrorLogProvider>
232
+ )
233
+ },
234
+ {
235
+ targetFps: 60,
236
+ gatherStats: false,
237
+ exitOnCtrlC: false,
238
+ useKittyKeyboard: {},
239
+ // Console enabled for debugging
240
+ useConsole: true,
241
+ openConsoleOnError: true,
242
+ consoleOptions: {
243
+ keyBindings: [],
244
+ onCopySelection: () => {},
245
+ },
246
+ },
247
+ )
248
+ })
249
+ }
250
+
251
+ function App() {
252
+ const route = useRoute()
253
+ const dimensions = useTerminalDimensions()
254
+ const renderer = useRenderer()
255
+ renderer.disableStdoutInterception()
256
+ const dialog = useDialog()
257
+ const local = useLocal()
258
+ const kv = useKV()
259
+ const command = useCommandDialog()
260
+ const sdk = useSDK()
261
+ const toast = useToast()
262
+ const errorLog = useErrorLog()
263
+ const { theme, mode, setMode } = useTheme()
264
+ const sync = useSync()
265
+ const exit = useExit()
266
+ const promptRef = usePromptRef()
267
+
268
+ // Wire up console copy-to-clipboard via opentui's onCopySelection callback
269
+ renderer.console.onCopySelection = async (text: string) => {
270
+ if (!text || text.length === 0) return
271
+
272
+ await Clipboard.copy(text)
273
+ .then(() => toast.show({ message: "Copied to clipboard", variant: "info" }))
274
+ .catch(toast.error)
275
+ renderer.clearSelection()
276
+ }
277
+ const [terminalTitleEnabled, setTerminalTitleEnabled] = createSignal(kv.get("terminal_title_enabled", true))
278
+
279
+ createEffect(() => {
280
+ if (route.data.type !== "session") return
281
+ kv.set("session_last", route.data.sessionID)
282
+ })
283
+
284
+ // Update terminal window title based on current route and session
285
+ createEffect(() => {
286
+ if (!terminalTitleEnabled() || Flag.OPENCODE_DISABLE_TERMINAL_TITLE) return
287
+
288
+ if (route.data.type === "home") {
289
+ renderer.setTerminalTitle("JonsOC")
290
+ return
291
+ }
292
+
293
+ if (route.data.type === "session") {
294
+ const session = sync.session.get(route.data.sessionID)
295
+ if (!session || SessionApi.isDefaultTitle(session.title)) {
296
+ renderer.setTerminalTitle("JonsOC")
297
+ return
298
+ }
299
+
300
+ // Truncate title to 40 chars max
301
+ const title = session.title.length > 40 ? session.title.slice(0, 37) + "..." : session.title
302
+ renderer.setTerminalTitle(`JOC | ${title}`)
303
+ }
304
+ })
305
+
306
+ const args = useArgs()
307
+
308
+ createEffect(
309
+ on(
310
+ () => errorLog.count,
311
+ (count, prev) => {
312
+ if (prev === undefined) return
313
+ if (count <= prev) return
314
+ console.error("Error count changed:", count, prev)
315
+ dialog.setSize("large")
316
+ dialog.replace(() => <DialogErrorLog />)
317
+ },
318
+ ),
319
+ )
320
+
321
+ onMount(() => {
322
+ batch(() => {
323
+ if (args.agent) local.agent.set(args.agent)
324
+ if (args.model) {
325
+ const { providerID, modelID } = Provider.parseModel(args.model)
326
+ if (!providerID || !modelID)
327
+ return toast.show({
328
+ variant: "warning",
329
+ message: `Invalid model format: ${args.model}`,
330
+ duration: 3000,
331
+ })
332
+ local.model.set({ providerID, modelID }, { recent: true })
333
+ }
334
+ if (args.sessionID) {
335
+ route.navigate({
336
+ type: "session",
337
+ sessionID: args.sessionID,
338
+ })
339
+ }
340
+ })
341
+ })
342
+
343
+ let continued = false
344
+ createEffect(() => {
345
+ // When using -c, session list is loaded in blocking phase, so we can navigate at "partial"
346
+ if (continued || sync.status === "loading" || !args.continue) return
347
+ const match = sync.data.session
348
+ .toSorted((a, b) => b.time.updated - a.time.updated)
349
+ .find((x) => x.parentID === undefined)?.id
350
+ if (match) {
351
+ continued = true
352
+ route.navigate({ type: "session", sessionID: match })
353
+ }
354
+ })
355
+
356
+ createEffect(
357
+ on(
358
+ () => sync.status === "complete" && sync.data.provider.length === 0,
359
+ (isEmpty, wasEmpty) => {
360
+ // only trigger when we transition into an empty-provider state
361
+ if (!isEmpty || wasEmpty) return
362
+ dialog.replace(() => <DialogProviderList />)
363
+ },
364
+ ),
365
+ )
366
+
367
+ const connected = useConnected()
368
+ const docsUrl = process.env.JONSOC_DOCS_URL ?? process.env.OPENCODE_DOCS_URL ?? "https://YOUR_DOMAIN_HERE/docs"
369
+ const zenUrl = process.env.JONSOC_ZEN_URL ?? process.env.OPENCODE_ZEN_URL ?? "https://YOUR_DOMAIN_HERE/zen"
370
+ command.register(() => [
371
+ {
372
+ title: "Switch session",
373
+ value: "session.list",
374
+ keybind: "session_list",
375
+ category: "Session",
376
+ suggested: sync.data.session.length > 0,
377
+ slash: {
378
+ name: "sessions",
379
+ aliases: ["resume", "continue"],
380
+ },
381
+ onSelect: () => {
382
+ dialog.replace(() => <DialogSessionList />)
383
+ },
384
+ },
385
+ {
386
+ title: "New session",
387
+ suggested: route.data.type === "session",
388
+ value: "session.new",
389
+ keybind: "session_new",
390
+ category: "Session",
391
+ slash: {
392
+ name: "new",
393
+ aliases: ["clear"],
394
+ },
395
+ onSelect: () => {
396
+ const current = promptRef.current
397
+ // Don't require focus - if there's any text, preserve it
398
+ const currentPrompt = current?.current?.input ? current.current : undefined
399
+ route.navigate({
400
+ type: "home",
401
+ initialPrompt: currentPrompt,
402
+ })
403
+ dialog.clear()
404
+ },
405
+ },
406
+ {
407
+ title: "Switch model",
408
+ value: "model.list",
409
+ keybind: "model_list",
410
+ suggested: true,
411
+ category: "Agent",
412
+ slash: {
413
+ name: "models",
414
+ },
415
+ onSelect: () => {
416
+ dialog.replace(() => <DialogModel />)
417
+ },
418
+ },
419
+ {
420
+ title: "Model cycle",
421
+ value: "model.cycle_recent",
422
+ keybind: "model_cycle_recent",
423
+ category: "Agent",
424
+ hidden: true,
425
+ onSelect: () => {
426
+ local.model.cycle(1)
427
+ },
428
+ },
429
+ {
430
+ title: "Model cycle reverse",
431
+ value: "model.cycle_recent_reverse",
432
+ keybind: "model_cycle_recent_reverse",
433
+ category: "Agent",
434
+ hidden: true,
435
+ onSelect: () => {
436
+ local.model.cycle(-1)
437
+ },
438
+ },
439
+ {
440
+ title: "Favorite cycle",
441
+ value: "model.cycle_favorite",
442
+ keybind: "model_cycle_favorite",
443
+ category: "Agent",
444
+ hidden: true,
445
+ onSelect: () => {
446
+ local.model.cycleFavorite(1)
447
+ },
448
+ },
449
+ {
450
+ title: "Favorite cycle reverse",
451
+ value: "model.cycle_favorite_reverse",
452
+ keybind: "model_cycle_favorite_reverse",
453
+ category: "Agent",
454
+ hidden: true,
455
+ onSelect: () => {
456
+ local.model.cycleFavorite(-1)
457
+ },
458
+ },
459
+ {
460
+ title: "Switch agent",
461
+ value: "agent.list",
462
+ keybind: "agent_list",
463
+ category: "Agent",
464
+ slash: {
465
+ name: "agents",
466
+ },
467
+ onSelect: () => {
468
+ dialog.replace(() => <DialogAgent />)
469
+ },
470
+ },
471
+ {
472
+ title: "Toggle MCPs",
473
+ value: "mcp.list",
474
+ category: "Agent",
475
+ slash: {
476
+ name: "mcps",
477
+ },
478
+ onSelect: () => {
479
+ dialog.replace(() => <DialogMcp />)
480
+ },
481
+ },
482
+ {
483
+ title: "Agent cycle",
484
+ value: "agent.cycle",
485
+ keybind: "agent_cycle",
486
+ category: "Agent",
487
+ hidden: true,
488
+ onSelect: () => {
489
+ local.agent.move(1)
490
+ },
491
+ },
492
+ {
493
+ title: "Variant cycle",
494
+ value: "variant.cycle",
495
+ keybind: "variant_cycle",
496
+ category: "Agent",
497
+ hidden: true,
498
+ onSelect: () => {
499
+ local.model.variant.cycle()
500
+ },
501
+ },
502
+ {
503
+ title: "Agent cycle reverse",
504
+ value: "agent.cycle.reverse",
505
+ keybind: "agent_cycle_reverse",
506
+ category: "Agent",
507
+ hidden: true,
508
+ onSelect: () => {
509
+ local.agent.move(-1)
510
+ },
511
+ },
512
+ {
513
+ title: "Connect provider",
514
+ value: "provider.connect",
515
+ suggested: !connected(),
516
+ slash: {
517
+ name: "connect",
518
+ },
519
+ onSelect: () => {
520
+ dialog.replace(() => <DialogProviderList />)
521
+ },
522
+ category: "Provider",
523
+ },
524
+ {
525
+ title: "View status",
526
+ keybind: "status_view",
527
+ value: "jonsoc.status",
528
+ slash: {
529
+ name: "status",
530
+ },
531
+ onSelect: () => {
532
+ dialog.replace(() => <DialogStatus />)
533
+ },
534
+ category: "System",
535
+ },
536
+ {
537
+ title: "Switch theme",
538
+ value: "theme.switch",
539
+ keybind: "theme_list",
540
+ slash: {
541
+ name: "themes",
542
+ },
543
+ onSelect: () => {
544
+ dialog.replace(() => <DialogThemeList />)
545
+ },
546
+ category: "System",
547
+ },
548
+ {
549
+ title: "Toggle appearance",
550
+ value: "theme.switch_mode",
551
+ onSelect: (dialog) => {
552
+ setMode(mode() === "dark" ? "light" : "dark")
553
+ dialog.clear()
554
+ },
555
+ category: "System",
556
+ },
557
+ {
558
+ title: "Help",
559
+ value: "help.show",
560
+ slash: {
561
+ name: "help",
562
+ },
563
+ onSelect: () => {
564
+ dialog.replace(() => <DialogHelp />)
565
+ },
566
+ category: "System",
567
+ },
568
+ {
569
+ title: "Open docs",
570
+ value: "docs.open",
571
+ onSelect: () => {
572
+ open(docsUrl).catch(() => {})
573
+ dialog.clear()
574
+ },
575
+ category: "System",
576
+ },
577
+ {
578
+ title: "Open WebUI",
579
+ value: "webui.open",
580
+ onSelect: () => {
581
+ open(sdk.url).catch(() => {})
582
+ dialog.clear()
583
+ },
584
+ category: "System",
585
+ },
586
+ {
587
+ title: "Exit the app",
588
+ value: "app.exit",
589
+ slash: {
590
+ name: "exit",
591
+ aliases: ["quit", "q"],
592
+ },
593
+ onSelect: () => exit(),
594
+ category: "System",
595
+ },
596
+ {
597
+ title: "Restart and resume session",
598
+ value: "app.restart",
599
+ slash: {
600
+ name: "restart",
601
+ aliases: ["reload"],
602
+ },
603
+ onSelect: () => exit(undefined, EXIT_CODE_RESTART),
604
+ category: "System",
605
+ },
606
+ {
607
+ title: "Toggle debug panel",
608
+ category: "System",
609
+ value: "app.debug",
610
+ onSelect: (dialog) => {
611
+ renderer.toggleDebugOverlay()
612
+ dialog.clear()
613
+ },
614
+ },
615
+ {
616
+ title: "View error log",
617
+ category: "System",
618
+ value: "app.error_log",
619
+ onSelect: (dialog) => {
620
+ dialog.setSize("large")
621
+ dialog.replace(() => <DialogErrorLog />)
622
+ },
623
+ },
624
+ {
625
+ title: "Write heap snapshot",
626
+ category: "System",
627
+ value: "app.heap_snapshot",
628
+ onSelect: (dialog) => {
629
+ const path = writeHeapSnapshot()
630
+ toast.show({
631
+ variant: "info",
632
+ message: `Heap snapshot written to ${path}`,
633
+ duration: 5000,
634
+ })
635
+ dialog.clear()
636
+ },
637
+ },
638
+ {
639
+ title: "Suspend terminal",
640
+ value: "terminal.suspend",
641
+ keybind: "terminal_suspend",
642
+ category: "System",
643
+ hidden: true,
644
+ onSelect: () => {
645
+ process.once("SIGCONT", () => {
646
+ renderer.resume()
647
+ })
648
+
649
+ renderer.suspend()
650
+ // pid=0 means send the signal to all processes in the process group
651
+ process.kill(0, "SIGTSTP")
652
+ },
653
+ },
654
+ {
655
+ title: terminalTitleEnabled() ? "Disable terminal title" : "Enable terminal title",
656
+ value: "terminal.title.toggle",
657
+ keybind: "terminal_title_toggle",
658
+ category: "System",
659
+ onSelect: (dialog) => {
660
+ setTerminalTitleEnabled((prev) => {
661
+ const next = !prev
662
+ kv.set("terminal_title_enabled", next)
663
+ if (!next) renderer.setTerminalTitle("")
664
+ return next
665
+ })
666
+ dialog.clear()
667
+ },
668
+ },
669
+ ])
670
+
671
+ createEffect(() => {
672
+ const currentModel = local.model.current()
673
+ if (!currentModel) return
674
+ if (currentModel.providerID === "openrouter" && !kv.get("openrouter_warning", false)) {
675
+ untrack(() => {
676
+ DialogAlert.show(
677
+ dialog,
678
+ "Warning",
679
+ `While openrouter is a convenient way to access LLMs your request will often be routed to subpar providers that do not work well in our testing.\n\nFor reliable access to models check out JonsOC Zen\n${zenUrl}`,
680
+ ).then(() => kv.set("openrouter_warning", true))
681
+ })
682
+ }
683
+ })
684
+
685
+ sdk.event.on(TuiEvent.CommandExecute.type, (evt) => {
686
+ command.trigger(evt.properties.command)
687
+ })
688
+
689
+ sdk.event.on(TuiEvent.ToastShow.type, (evt) => {
690
+ toast.show({
691
+ title: evt.properties.title,
692
+ message: evt.properties.message,
693
+ variant: evt.properties.variant,
694
+ duration: evt.properties.duration,
695
+ })
696
+ })
697
+
698
+ sdk.event.on(TuiEvent.SessionSelect.type, (evt) => {
699
+ route.navigate({
700
+ type: "session",
701
+ sessionID: evt.properties.sessionID,
702
+ })
703
+ })
704
+
705
+ sdk.event.on(SessionApi.Event.Deleted.type, (evt) => {
706
+ if (route.data.type === "session" && route.data.sessionID === evt.properties.info.id) {
707
+ route.navigate({ type: "home" })
708
+ toast.show({
709
+ variant: "info",
710
+ message: "The current session was deleted",
711
+ })
712
+ }
713
+ })
714
+
715
+ sdk.event.on(SessionApi.Event.Error.type, (evt) => {
716
+ const error = evt.properties.error
717
+ if (error && typeof error === "object" && error.name === "MessageAbortedError") return
718
+ const message = (() => {
719
+ if (!error) return "An error occurred"
720
+
721
+ if (typeof error === "object") {
722
+ const data = error.data
723
+ if ("message" in data && typeof data.message === "string") {
724
+ return data.message
725
+ }
726
+ }
727
+ return String(error)
728
+ })()
729
+
730
+ toast.show({
731
+ variant: "error",
732
+ message,
733
+ duration: 5000,
734
+ })
735
+ })
736
+
737
+ sdk.event.on(Installation.Event.UpdateAvailable.type, (evt) => {
738
+ toast.show({
739
+ variant: "info",
740
+ title: "Update Available",
741
+ message: `JonsOC v${evt.properties.version} is available. Run 'jonsoc upgrade' to update manually.`,
742
+ duration: 10000,
743
+ })
744
+ })
745
+
746
+ return (
747
+ <box
748
+ width={dimensions().width}
749
+ height={dimensions().height}
750
+ backgroundColor={theme.background}
751
+ onMouseUp={async () => {
752
+ if (Flag.OPENCODE_EXPERIMENTAL_DISABLE_COPY_ON_SELECT) {
753
+ renderer.clearSelection()
754
+ return
755
+ }
756
+ const text = renderer.getSelection()?.getSelectedText()
757
+ if (text && text.length > 0) {
758
+ await Clipboard.copy(text)
759
+ .then(() => toast.show({ message: "Copied to clipboard", variant: "info" }))
760
+ .catch(toast.error)
761
+ renderer.clearSelection()
762
+ }
763
+ }}
764
+ >
765
+ <Switch>
766
+ <Match when={route.data.type === "home"}>
767
+ <Home />
768
+ </Match>
769
+ <Match when={route.data.type === "session"}>
770
+ <Session />
771
+ </Match>
772
+ <Match when={route.data.type === "ui-settings"}>
773
+ <UISettings />
774
+ </Match>
775
+ </Switch>
776
+ <InspectorOverlay />
777
+ </box>
778
+ )
779
+ }
780
+
781
+ function ErrorComponent(props: {
782
+ error: Error
783
+ reset: () => void
784
+ onExit: () => Promise<void>
785
+ mode?: "dark" | "light"
786
+ }) {
787
+ const errorLog = useErrorLog()
788
+
789
+ let term: ReturnType<typeof useTerminalDimensions> | null = null
790
+ let renderer: ReturnType<typeof useRenderer> | null = null
791
+ let hasRendererContext = false
792
+
793
+ try {
794
+ term = useTerminalDimensions()
795
+ renderer = useRenderer()
796
+ hasRendererContext = true
797
+ } catch {
798
+ // Renderer context not available - will use console fallback
799
+ }
800
+
801
+ // If no renderer context, log to file and exit
802
+ if (!hasRendererContext) {
803
+ const logEntry = `[${new Date().toISOString()}] Fatal error (renderer unavailable):\nMessage: ${props.error.message}\nStack: ${props.error.stack}\n\n`
804
+ fs.appendFileSync("jonsoc-fatal.log", logEntry)
805
+ console.error("Fatal error (renderer unavailable):")
806
+ console.error(props.error.message)
807
+ console.error(props.error.stack)
808
+ return null
809
+ }
810
+
811
+ const handleExit = async () => {
812
+ if (renderer) {
813
+ renderer.setTerminalTitle("")
814
+ renderer.destroy()
815
+ }
816
+ props.onExit()
817
+ }
818
+
819
+ try {
820
+ useKeyboard((evt) => {
821
+ if (evt.ctrl && evt.name === "c") {
822
+ handleExit()
823
+ }
824
+ })
825
+ } catch {
826
+ // Keyboard context not available - will exit without handler
827
+ }
828
+ const [copied, setCopied] = createSignal(false)
829
+ const [showLog, setShowLog] = createSignal(false)
830
+
831
+ const repo = (process.env.JONSOC_REPO ?? process.env.OPENCODE_REPO ?? "Noisemaker111/Jonsoc").replace(
832
+ /^https?:\/\/github\.com\//,
833
+ "",
834
+ )
835
+ const issueURL = new URL(`https://github.com/${repo}/issues/new?template=bug-report.yml`)
836
+
837
+ // Choose safe fallback colors per mode since theme context may not be available
838
+ const isLight = props.mode === "light"
839
+ const colors = {
840
+ bg: isLight ? "#ffffff" : "#0a0a0a",
841
+ text: isLight ? "#1a1a1a" : "#eeeeee",
842
+ muted: isLight ? "#8a8a8a" : "#808080",
843
+ primary: isLight ? "#3b7dd8" : "#fab283",
844
+ error: isLight ? "#d32f2f" : "#f44336",
845
+ }
846
+
847
+ if (props.error.message) {
848
+ issueURL.searchParams.set("title", `opentui: fatal: ${props.error.message}`)
849
+ }
850
+
851
+ if (props.error.stack) {
852
+ issueURL.searchParams.set(
853
+ "description",
854
+ "```\n" + props.error.stack.substring(0, 6000 - issueURL.toString().length) + "...\n```",
855
+ )
856
+ }
857
+
858
+ issueURL.searchParams.set("jonsoc-version", Installation.VERSION)
859
+
860
+ const copyIssueURL = () => {
861
+ Clipboard.copy(issueURL.toString()).then(() => {
862
+ setCopied(true)
863
+ })
864
+ }
865
+
866
+ const termWidth = term?.().width ?? 80
867
+ const termHeight = term?.().height ?? 24
868
+
869
+ return (
870
+ <box flexDirection="column" gap={1} backgroundColor={colors.bg} width={termWidth} height={termHeight}>
871
+ <box flexDirection="row" gap={1} alignItems="center">
872
+ <text attributes={TextAttributes.BOLD} fg={colors.text}>
873
+ Please report an issue.
874
+ </text>
875
+ <box onMouseUp={copyIssueURL} backgroundColor={colors.primary} padding={1}>
876
+ <text attributes={TextAttributes.BOLD} fg={colors.bg}>
877
+ Copy issue URL (exception info pre-filled)
878
+ </text>
879
+ </box>
880
+ {copied() && <text fg={colors.muted}>Successfully copied</text>}
881
+ </box>
882
+ <box flexDirection="row" gap={2} alignItems="center">
883
+ <text fg={colors.text}>A fatal error occurred!</text>
884
+ <box onMouseUp={props.reset} backgroundColor={colors.primary} padding={1}>
885
+ <text fg={colors.bg}>Reset TUI</text>
886
+ </box>
887
+ <box onMouseUp={handleExit} backgroundColor={colors.primary} padding={1}>
888
+ <text fg={colors.bg}>Exit</text>
889
+ </box>
890
+ <box onMouseUp={() => setShowLog(!showLog())} backgroundColor={colors.primary} padding={1}>
891
+ <text fg={colors.bg}>{showLog() ? "Show Fatal Error" : `Show Error Log (${errorLog.count})`}</text>
892
+ </box>
893
+ </box>
894
+ <Show
895
+ when={showLog()}
896
+ fallback={
897
+ <>
898
+ <scrollbox height={Math.floor(termHeight * 0.7)}>
899
+ <text fg={colors.muted}>{props.error.stack}</text>
900
+ </scrollbox>
901
+ <text fg={colors.text}>{props.error.message}</text>
902
+ </>
903
+ }
904
+ >
905
+ <scrollbox height={Math.floor(termHeight * 0.7)}>
906
+ <For each={errorLog.errors}>
907
+ {(error: ErrorEntry) => (
908
+ <box flexDirection="column" border={["bottom"]} borderColor={colors.muted} paddingBottom={1} gap={1}>
909
+ <text fg={colors.primary}>
910
+ [{error.source ?? "unknown"}] {new Date(error.timestamp).toLocaleTimeString()}
911
+ </text>
912
+ <text fg={colors.error}>{error.message}</text>
913
+ <Show when={error.stack}>
914
+ <text fg={colors.muted}>{error.stack}</text>
915
+ </Show>
916
+ </box>
917
+ )}
918
+ </For>
919
+ </scrollbox>
920
+ </Show>
921
+ </box>
922
+ )
923
+ }