jonsoc 1.1.50 → 1.1.51

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 +256 -256
  6. package/bunfig.toml +7 -0
  7. package/package.json +142 -8
  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 +149 -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 +345 -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 +89 -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 +1547 -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 +1152 -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 +205 -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,500 @@
1
+ import type { Hooks, PluginInput } from "@jonsoc/plugin"
2
+ import { Log } from "../util/log"
3
+ import { Installation } from "../installation"
4
+ import { Auth, OAUTH_DUMMY_KEY } from "../auth"
5
+ import os from "os"
6
+
7
+ const log = Log.create({ service: "plugin.codex" })
8
+
9
+ const CLIENT_ID = "app_EMoamEEZ73f0CkXaXp7hrann"
10
+ const ISSUER = "https://auth.openai.com"
11
+ const CODEX_API_ENDPOINT = "https://chatgpt.com/backend-api/codex/responses"
12
+ const OAUTH_PORT = 1455
13
+
14
+ interface PkceCodes {
15
+ verifier: string
16
+ challenge: string
17
+ }
18
+
19
+ async function generatePKCE(): Promise<PkceCodes> {
20
+ const verifier = generateRandomString(43)
21
+ const encoder = new TextEncoder()
22
+ const data = encoder.encode(verifier)
23
+ const hash = await crypto.subtle.digest("SHA-256", data)
24
+ const challenge = base64UrlEncode(hash)
25
+ return { verifier, challenge }
26
+ }
27
+
28
+ function generateRandomString(length: number): string {
29
+ const chars = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789-._~"
30
+ const bytes = crypto.getRandomValues(new Uint8Array(length))
31
+ return Array.from(bytes)
32
+ .map((b) => chars[b % chars.length])
33
+ .join("")
34
+ }
35
+
36
+ function base64UrlEncode(buffer: ArrayBuffer): string {
37
+ const bytes = new Uint8Array(buffer)
38
+ const binary = String.fromCharCode(...bytes)
39
+ return btoa(binary).replace(/\+/g, "-").replace(/\//g, "_").replace(/=+$/, "")
40
+ }
41
+
42
+ function generateState(): string {
43
+ return base64UrlEncode(crypto.getRandomValues(new Uint8Array(32)).buffer)
44
+ }
45
+
46
+ export interface IdTokenClaims {
47
+ chatgpt_account_id?: string
48
+ organizations?: Array<{ id: string }>
49
+ email?: string
50
+ "https://api.openai.com/auth"?: {
51
+ chatgpt_account_id?: string
52
+ }
53
+ }
54
+
55
+ export function parseJwtClaims(token: string): IdTokenClaims | undefined {
56
+ const parts = token.split(".")
57
+ if (parts.length !== 3) return undefined
58
+ try {
59
+ return JSON.parse(Buffer.from(parts[1], "base64url").toString())
60
+ } catch {
61
+ return undefined
62
+ }
63
+ }
64
+
65
+ export function extractAccountIdFromClaims(claims: IdTokenClaims): string | undefined {
66
+ return (
67
+ claims.chatgpt_account_id ||
68
+ claims["https://api.openai.com/auth"]?.chatgpt_account_id ||
69
+ claims.organizations?.[0]?.id
70
+ )
71
+ }
72
+
73
+ export function extractAccountId(tokens: TokenResponse): string | undefined {
74
+ if (tokens.id_token) {
75
+ const claims = parseJwtClaims(tokens.id_token)
76
+ const accountId = claims && extractAccountIdFromClaims(claims)
77
+ if (accountId) return accountId
78
+ }
79
+ if (tokens.access_token) {
80
+ const claims = parseJwtClaims(tokens.access_token)
81
+ return claims ? extractAccountIdFromClaims(claims) : undefined
82
+ }
83
+ return undefined
84
+ }
85
+
86
+ function buildAuthorizeUrl(redirectUri: string, pkce: PkceCodes, state: string): string {
87
+ const params = new URLSearchParams({
88
+ response_type: "code",
89
+ client_id: CLIENT_ID,
90
+ redirect_uri: redirectUri,
91
+ scope: "openid profile email offline_access",
92
+ code_challenge: pkce.challenge,
93
+ code_challenge_method: "S256",
94
+ id_token_add_organizations: "true",
95
+ codex_cli_simplified_flow: "true",
96
+ state,
97
+ originator: "jonsoc",
98
+ })
99
+ return `${ISSUER}/oauth/authorize?${params.toString()}`
100
+ }
101
+
102
+ interface TokenResponse {
103
+ id_token: string
104
+ access_token: string
105
+ refresh_token: string
106
+ expires_in?: number
107
+ }
108
+
109
+ async function exchangeCodeForTokens(code: string, redirectUri: string, pkce: PkceCodes): Promise<TokenResponse> {
110
+ const response = await fetch(`${ISSUER}/oauth/token`, {
111
+ method: "POST",
112
+ headers: { "Content-Type": "application/x-www-form-urlencoded" },
113
+ body: new URLSearchParams({
114
+ grant_type: "authorization_code",
115
+ code,
116
+ redirect_uri: redirectUri,
117
+ client_id: CLIENT_ID,
118
+ code_verifier: pkce.verifier,
119
+ }).toString(),
120
+ })
121
+ if (!response.ok) {
122
+ throw new Error(`Token exchange failed: ${response.status}`)
123
+ }
124
+ return response.json()
125
+ }
126
+
127
+ async function refreshAccessToken(refreshToken: string): Promise<TokenResponse> {
128
+ const response = await fetch(`${ISSUER}/oauth/token`, {
129
+ method: "POST",
130
+ headers: { "Content-Type": "application/x-www-form-urlencoded" },
131
+ body: new URLSearchParams({
132
+ grant_type: "refresh_token",
133
+ refresh_token: refreshToken,
134
+ client_id: CLIENT_ID,
135
+ }).toString(),
136
+ })
137
+ if (!response.ok) {
138
+ throw new Error(`Token refresh failed: ${response.status}`)
139
+ }
140
+ return response.json()
141
+ }
142
+
143
+ const HTML_SUCCESS = `<!doctype html>
144
+ <html>
145
+ <head>
146
+ <title>JonsOC - Codex Authorization Successful</title>
147
+ <style>
148
+ body {
149
+ font-family:
150
+ system-ui,
151
+ -apple-system,
152
+ sans-serif;
153
+ display: flex;
154
+ justify-content: center;
155
+ align-items: center;
156
+ height: 100vh;
157
+ margin: 0;
158
+ background: #131010;
159
+ color: #f1ecec;
160
+ }
161
+ .container {
162
+ text-align: center;
163
+ padding: 2rem;
164
+ }
165
+ h1 {
166
+ color: #f1ecec;
167
+ margin-bottom: 1rem;
168
+ }
169
+ p {
170
+ color: #b7b1b1;
171
+ }
172
+ </style>
173
+ </head>
174
+ <body>
175
+ <div class="container">
176
+ <h1>Authorization Successful</h1>
177
+ <p>You can close this window and return to JonsOC.</p>
178
+ </div>
179
+ <script>
180
+ setTimeout(() => window.close(), 2000)
181
+ </script>
182
+ </body>
183
+ </html>`
184
+
185
+ const HTML_ERROR = (error: string) => `<!doctype html>
186
+ <html>
187
+ <head>
188
+ <title>JonsOC - Codex Authorization Failed</title>
189
+ <style>
190
+ body {
191
+ font-family:
192
+ system-ui,
193
+ -apple-system,
194
+ sans-serif;
195
+ display: flex;
196
+ justify-content: center;
197
+ align-items: center;
198
+ height: 100vh;
199
+ margin: 0;
200
+ background: #131010;
201
+ color: #f1ecec;
202
+ }
203
+ .container {
204
+ text-align: center;
205
+ padding: 2rem;
206
+ }
207
+ h1 {
208
+ color: #fc533a;
209
+ margin-bottom: 1rem;
210
+ }
211
+ p {
212
+ color: #b7b1b1;
213
+ }
214
+ .error {
215
+ color: #ff917b;
216
+ font-family: monospace;
217
+ margin-top: 1rem;
218
+ padding: 1rem;
219
+ background: #3c140d;
220
+ border-radius: 0.5rem;
221
+ }
222
+ </style>
223
+ </head>
224
+ <body>
225
+ <div class="container">
226
+ <h1>Authorization Failed</h1>
227
+ <p>An error occurred during authorization.</p>
228
+ <div class="error">${error}</div>
229
+ </div>
230
+ </body>
231
+ </html>`
232
+
233
+ interface PendingOAuth {
234
+ pkce: PkceCodes
235
+ state: string
236
+ resolve: (tokens: TokenResponse) => void
237
+ reject: (error: Error) => void
238
+ }
239
+
240
+ let oauthServer: ReturnType<typeof Bun.serve> | undefined
241
+ let pendingOAuth: PendingOAuth | undefined
242
+
243
+ async function startOAuthServer(): Promise<{ port: number; redirectUri: string }> {
244
+ if (oauthServer) {
245
+ return { port: OAUTH_PORT, redirectUri: `http://localhost:${OAUTH_PORT}/auth/callback` }
246
+ }
247
+
248
+ oauthServer = Bun.serve({
249
+ port: OAUTH_PORT,
250
+ fetch(req) {
251
+ const url = new URL(req.url)
252
+
253
+ if (url.pathname === "/auth/callback") {
254
+ const code = url.searchParams.get("code")
255
+ const state = url.searchParams.get("state")
256
+ const error = url.searchParams.get("error")
257
+ const errorDescription = url.searchParams.get("error_description")
258
+
259
+ if (error) {
260
+ const errorMsg = errorDescription || error
261
+ pendingOAuth?.reject(new Error(errorMsg))
262
+ pendingOAuth = undefined
263
+ return new Response(HTML_ERROR(errorMsg), {
264
+ headers: { "Content-Type": "text/html" },
265
+ })
266
+ }
267
+
268
+ if (!code) {
269
+ const errorMsg = "Missing authorization code"
270
+ pendingOAuth?.reject(new Error(errorMsg))
271
+ pendingOAuth = undefined
272
+ return new Response(HTML_ERROR(errorMsg), {
273
+ status: 400,
274
+ headers: { "Content-Type": "text/html" },
275
+ })
276
+ }
277
+
278
+ if (!pendingOAuth || state !== pendingOAuth.state) {
279
+ const errorMsg = "Invalid state - potential CSRF attack"
280
+ pendingOAuth?.reject(new Error(errorMsg))
281
+ pendingOAuth = undefined
282
+ return new Response(HTML_ERROR(errorMsg), {
283
+ status: 400,
284
+ headers: { "Content-Type": "text/html" },
285
+ })
286
+ }
287
+
288
+ const current = pendingOAuth
289
+ pendingOAuth = undefined
290
+
291
+ exchangeCodeForTokens(code, `http://localhost:${OAUTH_PORT}/auth/callback`, current.pkce)
292
+ .then((tokens) => current.resolve(tokens))
293
+ .catch((err) => current.reject(err))
294
+
295
+ return new Response(HTML_SUCCESS, {
296
+ headers: { "Content-Type": "text/html" },
297
+ })
298
+ }
299
+
300
+ if (url.pathname === "/cancel") {
301
+ pendingOAuth?.reject(new Error("Login cancelled"))
302
+ pendingOAuth = undefined
303
+ return new Response("Login cancelled", { status: 200 })
304
+ }
305
+
306
+ return new Response("Not found", { status: 404 })
307
+ },
308
+ })
309
+
310
+ log.info("codex oauth server started", { port: OAUTH_PORT })
311
+ return { port: OAUTH_PORT, redirectUri: `http://localhost:${OAUTH_PORT}/auth/callback` }
312
+ }
313
+
314
+ function stopOAuthServer() {
315
+ if (oauthServer) {
316
+ oauthServer.stop()
317
+ oauthServer = undefined
318
+ log.info("codex oauth server stopped")
319
+ }
320
+ }
321
+
322
+ function waitForOAuthCallback(pkce: PkceCodes, state: string): Promise<TokenResponse> {
323
+ return new Promise((resolve, reject) => {
324
+ const timeout = setTimeout(
325
+ () => {
326
+ if (pendingOAuth) {
327
+ pendingOAuth = undefined
328
+ reject(new Error("OAuth callback timeout - authorization took too long"))
329
+ }
330
+ },
331
+ 5 * 60 * 1000,
332
+ ) // 5 minute timeout
333
+
334
+ pendingOAuth = {
335
+ pkce,
336
+ state,
337
+ resolve: (tokens) => {
338
+ clearTimeout(timeout)
339
+ resolve(tokens)
340
+ },
341
+ reject: (error) => {
342
+ clearTimeout(timeout)
343
+ reject(error)
344
+ },
345
+ }
346
+ })
347
+ }
348
+
349
+ export async function CodexAuthPlugin(input: PluginInput): Promise<Hooks> {
350
+ return {
351
+ auth: {
352
+ provider: "openai",
353
+ async loader(getAuth, provider) {
354
+ const auth = await getAuth()
355
+ if (auth.type !== "oauth") return {}
356
+
357
+ // Filter models to only allowed Codex models for OAuth
358
+ const allowedModels = new Set(["gpt-5.1-codex-max", "gpt-5.1-codex-mini", "gpt-5.2", "gpt-5.2-codex"])
359
+ for (const modelId of Object.keys(provider.models)) {
360
+ if (!allowedModels.has(modelId)) {
361
+ delete provider.models[modelId]
362
+ }
363
+ }
364
+
365
+ // Zero out costs for Codex (included with ChatGPT subscription)
366
+ for (const model of Object.values(provider.models)) {
367
+ model.cost = {
368
+ input: 0,
369
+ output: 0,
370
+ cache: { read: 0, write: 0 },
371
+ }
372
+ }
373
+
374
+ return {
375
+ apiKey: OAUTH_DUMMY_KEY,
376
+ async fetch(requestInput: RequestInfo | URL, init?: RequestInit) {
377
+ // Remove dummy API key authorization header
378
+ if (init?.headers) {
379
+ if (init.headers instanceof Headers) {
380
+ init.headers.delete("authorization")
381
+ init.headers.delete("Authorization")
382
+ } else if (Array.isArray(init.headers)) {
383
+ init.headers = init.headers.filter(([key]) => key.toLowerCase() !== "authorization")
384
+ } else {
385
+ delete init.headers["authorization"]
386
+ delete init.headers["Authorization"]
387
+ }
388
+ }
389
+
390
+ const currentAuth = await getAuth()
391
+ if (currentAuth.type !== "oauth") return fetch(requestInput, init)
392
+
393
+ // Cast to include accountId field
394
+ const authWithAccount = currentAuth as typeof currentAuth & { accountId?: string }
395
+
396
+ // Check if token needs refresh
397
+ if (!currentAuth.access || currentAuth.expires < Date.now()) {
398
+ log.info("refreshing codex access token")
399
+ const tokens = await refreshAccessToken(currentAuth.refresh)
400
+ const newAccountId = extractAccountId(tokens) || authWithAccount.accountId
401
+ await input.client.auth.set({
402
+ path: { id: "openai" },
403
+ body: {
404
+ type: "oauth",
405
+ refresh: tokens.refresh_token,
406
+ access: tokens.access_token,
407
+ expires: Date.now() + (tokens.expires_in ?? 3600) * 1000,
408
+ ...(newAccountId && { accountId: newAccountId }),
409
+ },
410
+ })
411
+ currentAuth.access = tokens.access_token
412
+ authWithAccount.accountId = newAccountId
413
+ }
414
+
415
+ // Build headers
416
+ const headers = new Headers()
417
+ if (init?.headers) {
418
+ if (init.headers instanceof Headers) {
419
+ init.headers.forEach((value, key) => headers.set(key, value))
420
+ } else if (Array.isArray(init.headers)) {
421
+ for (const [key, value] of init.headers) {
422
+ if (value !== undefined) headers.set(key, String(value))
423
+ }
424
+ } else {
425
+ for (const [key, value] of Object.entries(init.headers)) {
426
+ if (value !== undefined) headers.set(key, String(value))
427
+ }
428
+ }
429
+ }
430
+
431
+ // Set authorization header with access token
432
+ headers.set("authorization", `Bearer ${currentAuth.access}`)
433
+
434
+ // Set ChatGPT-Account-Id header for organization subscriptions
435
+ if (authWithAccount.accountId) {
436
+ headers.set("ChatGPT-Account-Id", authWithAccount.accountId)
437
+ }
438
+
439
+ // Rewrite URL to Codex endpoint
440
+ const parsed =
441
+ requestInput instanceof URL
442
+ ? requestInput
443
+ : new URL(typeof requestInput === "string" ? requestInput : requestInput.url)
444
+ const url =
445
+ parsed.pathname.includes("/v1/responses") || parsed.pathname.includes("/chat/completions")
446
+ ? new URL(CODEX_API_ENDPOINT)
447
+ : parsed
448
+
449
+ return fetch(url, {
450
+ ...init,
451
+ headers,
452
+ })
453
+ },
454
+ }
455
+ },
456
+ methods: [
457
+ {
458
+ label: "ChatGPT Pro/Plus",
459
+ type: "oauth",
460
+ authorize: async () => {
461
+ const { redirectUri } = await startOAuthServer()
462
+ const pkce = await generatePKCE()
463
+ const state = generateState()
464
+ const authUrl = buildAuthorizeUrl(redirectUri, pkce, state)
465
+
466
+ const callbackPromise = waitForOAuthCallback(pkce, state)
467
+
468
+ return {
469
+ url: authUrl,
470
+ instructions: "Complete authorization in your browser. This window will close automatically.",
471
+ method: "auto" as const,
472
+ callback: async () => {
473
+ const tokens = await callbackPromise
474
+ stopOAuthServer()
475
+ const accountId = extractAccountId(tokens)
476
+ return {
477
+ type: "success" as const,
478
+ refresh: tokens.refresh_token,
479
+ access: tokens.access_token,
480
+ expires: Date.now() + (tokens.expires_in ?? 3600) * 1000,
481
+ accountId,
482
+ }
483
+ },
484
+ }
485
+ },
486
+ },
487
+ {
488
+ label: "Manually enter API Key",
489
+ type: "api",
490
+ },
491
+ ],
492
+ },
493
+ "chat.headers": async (input, output) => {
494
+ if (input.model.providerID !== "openai") return
495
+ output.headers.originator = "jonsoc"
496
+ output.headers["User-Agent"] = `${Installation.USER_AGENT} (${os.platform()} ${os.release()}; ${os.arch()})`
497
+ output.headers.session_id = input.sessionID
498
+ },
499
+ }
500
+ }