easc-cli 1.1.37 → 1.1.40

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 (412) hide show
  1. package/bin/{opencode → opencode.cjs} +4 -6
  2. package/package.json +15 -120
  3. package/AGENTS.md +0 -27
  4. package/Dockerfile +0 -18
  5. package/README.md +0 -15
  6. package/bunfig.toml +0 -7
  7. package/charity-website/README.md +0 -252
  8. package/charity-website/admin/index.html +0 -299
  9. package/charity-website/package.json +0 -40
  10. package/charity-website/public/index.html +0 -265
  11. package/charity-website/src/css/admin.css +0 -710
  12. package/charity-website/src/css/style.css +0 -741
  13. package/charity-website/src/js/admin.js +0 -743
  14. package/charity-website/src/js/app.js +0 -444
  15. package/parsers-config.ts +0 -253
  16. package/script/build.ts +0 -172
  17. package/script/deploy.ts +0 -96
  18. package/script/publish-registries.ts +0 -187
  19. package/script/publish.ts +0 -70
  20. package/script/schema.ts +0 -47
  21. package/script/seed-e2e.ts +0 -50
  22. package/src/acp/README.md +0 -164
  23. package/src/acp/agent.ts +0 -1285
  24. package/src/acp/session.ts +0 -105
  25. package/src/acp/types.ts +0 -22
  26. package/src/agent/agent.ts +0 -332
  27. package/src/agent/generate.txt +0 -75
  28. package/src/agent/prompt/compaction.txt +0 -12
  29. package/src/agent/prompt/explore.txt +0 -18
  30. package/src/agent/prompt/summary.txt +0 -11
  31. package/src/agent/prompt/title.txt +0 -43
  32. package/src/auth/eliseart.ts +0 -76
  33. package/src/auth/index.ts +0 -73
  34. package/src/bun/index.ts +0 -134
  35. package/src/bus/bus-event.ts +0 -43
  36. package/src/bus/global.ts +0 -10
  37. package/src/bus/index.ts +0 -105
  38. package/src/cli/bootstrap.ts +0 -17
  39. package/src/cli/cmd/account.ts +0 -81
  40. package/src/cli/cmd/acp.ts +0 -69
  41. package/src/cli/cmd/agent.ts +0 -257
  42. package/src/cli/cmd/auth.ts +0 -427
  43. package/src/cli/cmd/cmd.ts +0 -7
  44. package/src/cli/cmd/debug/agent.ts +0 -166
  45. package/src/cli/cmd/debug/config.ts +0 -16
  46. package/src/cli/cmd/debug/file.ts +0 -97
  47. package/src/cli/cmd/debug/index.ts +0 -48
  48. package/src/cli/cmd/debug/lsp.ts +0 -52
  49. package/src/cli/cmd/debug/ripgrep.ts +0 -87
  50. package/src/cli/cmd/debug/scrap.ts +0 -16
  51. package/src/cli/cmd/debug/skill.ts +0 -16
  52. package/src/cli/cmd/debug/snapshot.ts +0 -52
  53. package/src/cli/cmd/export.ts +0 -88
  54. package/src/cli/cmd/generate.ts +0 -38
  55. package/src/cli/cmd/github.ts +0 -1548
  56. package/src/cli/cmd/import.ts +0 -98
  57. package/src/cli/cmd/mcp.ts +0 -827
  58. package/src/cli/cmd/models.ts +0 -77
  59. package/src/cli/cmd/pr.ts +0 -112
  60. package/src/cli/cmd/run.ts +0 -407
  61. package/src/cli/cmd/serve.ts +0 -20
  62. package/src/cli/cmd/session.ts +0 -135
  63. package/src/cli/cmd/stats.ts +0 -402
  64. package/src/cli/cmd/tui/app.tsx +0 -774
  65. package/src/cli/cmd/tui/attach.ts +0 -31
  66. package/src/cli/cmd/tui/component/border.tsx +0 -21
  67. package/src/cli/cmd/tui/component/dialog-agent.tsx +0 -31
  68. package/src/cli/cmd/tui/component/dialog-command.tsx +0 -148
  69. package/src/cli/cmd/tui/component/dialog-mcp.tsx +0 -86
  70. package/src/cli/cmd/tui/component/dialog-model.tsx +0 -234
  71. package/src/cli/cmd/tui/component/dialog-provider.tsx +0 -256
  72. package/src/cli/cmd/tui/component/dialog-session-list.tsx +0 -114
  73. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +0 -31
  74. package/src/cli/cmd/tui/component/dialog-stash.tsx +0 -87
  75. package/src/cli/cmd/tui/component/dialog-status.tsx +0 -164
  76. package/src/cli/cmd/tui/component/dialog-supabase.tsx +0 -102
  77. package/src/cli/cmd/tui/component/dialog-tag.tsx +0 -44
  78. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +0 -50
  79. package/src/cli/cmd/tui/component/logo.tsx +0 -88
  80. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +0 -653
  81. package/src/cli/cmd/tui/component/prompt/frecency.tsx +0 -89
  82. package/src/cli/cmd/tui/component/prompt/history.tsx +0 -108
  83. package/src/cli/cmd/tui/component/prompt/index.tsx +0 -1182
  84. package/src/cli/cmd/tui/component/prompt/stash.tsx +0 -101
  85. package/src/cli/cmd/tui/component/spinner.tsx +0 -16
  86. package/src/cli/cmd/tui/component/textarea-keybindings.ts +0 -73
  87. package/src/cli/cmd/tui/component/tips.tsx +0 -153
  88. package/src/cli/cmd/tui/component/todo-item.tsx +0 -32
  89. package/src/cli/cmd/tui/context/args.tsx +0 -14
  90. package/src/cli/cmd/tui/context/directory.ts +0 -13
  91. package/src/cli/cmd/tui/context/exit.tsx +0 -23
  92. package/src/cli/cmd/tui/context/helper.tsx +0 -25
  93. package/src/cli/cmd/tui/context/keybind.tsx +0 -101
  94. package/src/cli/cmd/tui/context/kv.tsx +0 -52
  95. package/src/cli/cmd/tui/context/local.tsx +0 -402
  96. package/src/cli/cmd/tui/context/prompt.tsx +0 -18
  97. package/src/cli/cmd/tui/context/route.tsx +0 -46
  98. package/src/cli/cmd/tui/context/sdk.tsx +0 -94
  99. package/src/cli/cmd/tui/context/sync.tsx +0 -445
  100. package/src/cli/cmd/tui/context/theme/aura.json +0 -69
  101. package/src/cli/cmd/tui/context/theme/ayu.json +0 -80
  102. package/src/cli/cmd/tui/context/theme/carbonfox.json +0 -248
  103. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +0 -233
  104. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +0 -233
  105. package/src/cli/cmd/tui/context/theme/catppuccin.json +0 -112
  106. package/src/cli/cmd/tui/context/theme/cobalt2.json +0 -228
  107. package/src/cli/cmd/tui/context/theme/cursor.json +0 -249
  108. package/src/cli/cmd/tui/context/theme/dracula.json +0 -219
  109. package/src/cli/cmd/tui/context/theme/everforest.json +0 -241
  110. package/src/cli/cmd/tui/context/theme/flexoki.json +0 -237
  111. package/src/cli/cmd/tui/context/theme/github.json +0 -233
  112. package/src/cli/cmd/tui/context/theme/gruvbox.json +0 -95
  113. package/src/cli/cmd/tui/context/theme/kanagawa.json +0 -77
  114. package/src/cli/cmd/tui/context/theme/lucent-orng.json +0 -237
  115. package/src/cli/cmd/tui/context/theme/material.json +0 -235
  116. package/src/cli/cmd/tui/context/theme/matrix.json +0 -77
  117. package/src/cli/cmd/tui/context/theme/mercury.json +0 -252
  118. package/src/cli/cmd/tui/context/theme/monokai.json +0 -221
  119. package/src/cli/cmd/tui/context/theme/nightowl.json +0 -221
  120. package/src/cli/cmd/tui/context/theme/nord.json +0 -223
  121. package/src/cli/cmd/tui/context/theme/one-dark.json +0 -84
  122. package/src/cli/cmd/tui/context/theme/orng.json +0 -249
  123. package/src/cli/cmd/tui/context/theme/osaka-jade.json +0 -93
  124. package/src/cli/cmd/tui/context/theme/palenight.json +0 -222
  125. package/src/cli/cmd/tui/context/theme/rosepine.json +0 -234
  126. package/src/cli/cmd/tui/context/theme/solarized.json +0 -223
  127. package/src/cli/cmd/tui/context/theme/synthwave84.json +0 -226
  128. package/src/cli/cmd/tui/context/theme/tokyonight.json +0 -243
  129. package/src/cli/cmd/tui/context/theme/vercel.json +0 -245
  130. package/src/cli/cmd/tui/context/theme/vesper.json +0 -218
  131. package/src/cli/cmd/tui/context/theme/zenburn.json +0 -223
  132. package/src/cli/cmd/tui/context/theme.tsx +0 -1152
  133. package/src/cli/cmd/tui/event.ts +0 -48
  134. package/src/cli/cmd/tui/routes/home.tsx +0 -140
  135. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +0 -64
  136. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +0 -109
  137. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +0 -26
  138. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +0 -47
  139. package/src/cli/cmd/tui/routes/session/dialog-tool.tsx +0 -63
  140. package/src/cli/cmd/tui/routes/session/footer.tsx +0 -129
  141. package/src/cli/cmd/tui/routes/session/header.tsx +0 -136
  142. package/src/cli/cmd/tui/routes/session/index.tsx +0 -2132
  143. package/src/cli/cmd/tui/routes/session/permission.tsx +0 -495
  144. package/src/cli/cmd/tui/routes/session/question.tsx +0 -435
  145. package/src/cli/cmd/tui/routes/session/sidebar.tsx +0 -313
  146. package/src/cli/cmd/tui/thread.ts +0 -165
  147. package/src/cli/cmd/tui/ui/dialog-alert.tsx +0 -57
  148. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +0 -83
  149. package/src/cli/cmd/tui/ui/dialog-export-options.tsx +0 -204
  150. package/src/cli/cmd/tui/ui/dialog-help.tsx +0 -38
  151. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +0 -77
  152. package/src/cli/cmd/tui/ui/dialog-select.tsx +0 -376
  153. package/src/cli/cmd/tui/ui/dialog.tsx +0 -167
  154. package/src/cli/cmd/tui/ui/link.tsx +0 -28
  155. package/src/cli/cmd/tui/ui/spinner.ts +0 -368
  156. package/src/cli/cmd/tui/ui/toast.tsx +0 -100
  157. package/src/cli/cmd/tui/util/clipboard.ts +0 -160
  158. package/src/cli/cmd/tui/util/editor.ts +0 -32
  159. package/src/cli/cmd/tui/util/signal.ts +0 -7
  160. package/src/cli/cmd/tui/util/terminal.ts +0 -114
  161. package/src/cli/cmd/tui/util/transcript.ts +0 -98
  162. package/src/cli/cmd/tui/worker.ts +0 -152
  163. package/src/cli/cmd/uninstall.ts +0 -357
  164. package/src/cli/cmd/upgrade.ts +0 -73
  165. package/src/cli/cmd/web.ts +0 -81
  166. package/src/cli/error.ts +0 -57
  167. package/src/cli/network.ts +0 -53
  168. package/src/cli/ui.ts +0 -84
  169. package/src/cli/upgrade.ts +0 -25
  170. package/src/command/index.ts +0 -131
  171. package/src/command/template/initialize.txt +0 -10
  172. package/src/command/template/review.txt +0 -99
  173. package/src/config/config.ts +0 -1361
  174. package/src/config/markdown.ts +0 -93
  175. package/src/env/index.ts +0 -26
  176. package/src/file/ignore.ts +0 -83
  177. package/src/file/index.ts +0 -411
  178. package/src/file/ripgrep.ts +0 -407
  179. package/src/file/time.ts +0 -64
  180. package/src/file/watcher.ts +0 -127
  181. package/src/flag/flag.ts +0 -54
  182. package/src/format/formatter.ts +0 -342
  183. package/src/format/index.ts +0 -137
  184. package/src/global/index.ts +0 -55
  185. package/src/id/id.ts +0 -83
  186. package/src/ide/index.ts +0 -76
  187. package/src/index.ts +0 -162
  188. package/src/installation/index.ts +0 -246
  189. package/src/lsp/client.ts +0 -252
  190. package/src/lsp/index.ts +0 -485
  191. package/src/lsp/language.ts +0 -119
  192. package/src/lsp/server.ts +0 -2046
  193. package/src/mcp/auth.ts +0 -135
  194. package/src/mcp/index.ts +0 -931
  195. package/src/mcp/oauth-callback.ts +0 -200
  196. package/src/mcp/oauth-provider.ts +0 -154
  197. package/src/patch/index.ts +0 -680
  198. package/src/permission/arity.ts +0 -163
  199. package/src/permission/index.ts +0 -210
  200. package/src/permission/next.ts +0 -269
  201. package/src/plugin/codex.ts +0 -493
  202. package/src/plugin/copilot.ts +0 -269
  203. package/src/plugin/index.ts +0 -136
  204. package/src/project/bootstrap.ts +0 -35
  205. package/src/project/instance.ts +0 -91
  206. package/src/project/project.ts +0 -339
  207. package/src/project/state.ts +0 -66
  208. package/src/project/vcs.ts +0 -76
  209. package/src/provider/auth.ts +0 -147
  210. package/src/provider/models-macro.ts +0 -11
  211. package/src/provider/models.ts +0 -112
  212. package/src/provider/provider.ts +0 -1435
  213. package/src/provider/sdk/openai-compatible/src/README.md +0 -5
  214. package/src/provider/sdk/openai-compatible/src/index.ts +0 -2
  215. package/src/provider/sdk/openai-compatible/src/openai-compatible-provider.ts +0 -100
  216. package/src/provider/sdk/openai-compatible/src/responses/convert-to-openai-responses-input.ts +0 -303
  217. package/src/provider/sdk/openai-compatible/src/responses/map-openai-responses-finish-reason.ts +0 -22
  218. package/src/provider/sdk/openai-compatible/src/responses/openai-config.ts +0 -18
  219. package/src/provider/sdk/openai-compatible/src/responses/openai-error.ts +0 -22
  220. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-api-types.ts +0 -207
  221. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-language-model.ts +0 -1732
  222. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-prepare-tools.ts +0 -177
  223. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-settings.ts +0 -1
  224. package/src/provider/sdk/openai-compatible/src/responses/tool/code-interpreter.ts +0 -88
  225. package/src/provider/sdk/openai-compatible/src/responses/tool/file-search.ts +0 -128
  226. package/src/provider/sdk/openai-compatible/src/responses/tool/image-generation.ts +0 -115
  227. package/src/provider/sdk/openai-compatible/src/responses/tool/local-shell.ts +0 -65
  228. package/src/provider/sdk/openai-compatible/src/responses/tool/web-search-preview.ts +0 -104
  229. package/src/provider/sdk/openai-compatible/src/responses/tool/web-search.ts +0 -103
  230. package/src/provider/transform.ts +0 -733
  231. package/src/pty/index.ts +0 -232
  232. package/src/question/index.ts +0 -171
  233. package/src/scheduler/index.ts +0 -61
  234. package/src/server/error.ts +0 -36
  235. package/src/server/event.ts +0 -7
  236. package/src/server/mdns.ts +0 -59
  237. package/src/server/routes/config.ts +0 -92
  238. package/src/server/routes/experimental.ts +0 -208
  239. package/src/server/routes/file.ts +0 -197
  240. package/src/server/routes/global.ts +0 -135
  241. package/src/server/routes/mcp.ts +0 -361
  242. package/src/server/routes/permission.ts +0 -68
  243. package/src/server/routes/project.ts +0 -82
  244. package/src/server/routes/provider.ts +0 -165
  245. package/src/server/routes/pty.ts +0 -169
  246. package/src/server/routes/question.ts +0 -98
  247. package/src/server/routes/session.ts +0 -935
  248. package/src/server/routes/tui.ts +0 -379
  249. package/src/server/server.ts +0 -573
  250. package/src/session/compaction.ts +0 -225
  251. package/src/session/index.ts +0 -488
  252. package/src/session/llm.ts +0 -279
  253. package/src/session/message-v2.ts +0 -702
  254. package/src/session/message.ts +0 -189
  255. package/src/session/processor.ts +0 -406
  256. package/src/session/prompt/anthropic-20250930.txt +0 -166
  257. package/src/session/prompt/anthropic.txt +0 -105
  258. package/src/session/prompt/anthropic_spoof.txt +0 -1
  259. package/src/session/prompt/beast.txt +0 -147
  260. package/src/session/prompt/build-switch.txt +0 -5
  261. package/src/session/prompt/codex_header.txt +0 -79
  262. package/src/session/prompt/copilot-gpt-5.txt +0 -143
  263. package/src/session/prompt/gemini.txt +0 -155
  264. package/src/session/prompt/max-steps.txt +0 -16
  265. package/src/session/prompt/plan-reminder-anthropic.txt +0 -67
  266. package/src/session/prompt/plan.txt +0 -26
  267. package/src/session/prompt/qwen.txt +0 -109
  268. package/src/session/prompt.ts +0 -1820
  269. package/src/session/retry.ts +0 -90
  270. package/src/session/revert.ts +0 -108
  271. package/src/session/status.ts +0 -76
  272. package/src/session/summary.ts +0 -150
  273. package/src/session/system.ts +0 -152
  274. package/src/session/todo.ts +0 -37
  275. package/src/share/share-next.ts +0 -200
  276. package/src/share/share.ts +0 -92
  277. package/src/shell/shell.ts +0 -67
  278. package/src/skill/index.ts +0 -1
  279. package/src/skill/skill.ts +0 -136
  280. package/src/snapshot/index.ts +0 -236
  281. package/src/storage/storage.ts +0 -227
  282. package/src/tool/apply_patch.ts +0 -269
  283. package/src/tool/apply_patch.txt +0 -33
  284. package/src/tool/bash.ts +0 -259
  285. package/src/tool/bash.txt +0 -115
  286. package/src/tool/batch.ts +0 -175
  287. package/src/tool/batch.txt +0 -24
  288. package/src/tool/codesearch.ts +0 -132
  289. package/src/tool/codesearch.txt +0 -12
  290. package/src/tool/edit.ts +0 -645
  291. package/src/tool/edit.txt +0 -10
  292. package/src/tool/external-directory.ts +0 -32
  293. package/src/tool/glob.ts +0 -77
  294. package/src/tool/glob.txt +0 -6
  295. package/src/tool/grep.ts +0 -154
  296. package/src/tool/grep.txt +0 -8
  297. package/src/tool/invalid.ts +0 -17
  298. package/src/tool/ls.ts +0 -121
  299. package/src/tool/ls.txt +0 -1
  300. package/src/tool/lsp.ts +0 -96
  301. package/src/tool/lsp.txt +0 -19
  302. package/src/tool/multiedit.ts +0 -46
  303. package/src/tool/multiedit.txt +0 -41
  304. package/src/tool/plan-enter.txt +0 -14
  305. package/src/tool/plan-exit.txt +0 -13
  306. package/src/tool/plan.ts +0 -130
  307. package/src/tool/question.ts +0 -33
  308. package/src/tool/question.txt +0 -10
  309. package/src/tool/read.ts +0 -202
  310. package/src/tool/read.txt +0 -12
  311. package/src/tool/registry.ts +0 -164
  312. package/src/tool/skill.ts +0 -75
  313. package/src/tool/task.ts +0 -188
  314. package/src/tool/task.txt +0 -60
  315. package/src/tool/todo.ts +0 -53
  316. package/src/tool/todoread.txt +0 -14
  317. package/src/tool/todowrite.txt +0 -167
  318. package/src/tool/tool.ts +0 -88
  319. package/src/tool/truncation.ts +0 -106
  320. package/src/tool/webfetch.ts +0 -182
  321. package/src/tool/webfetch.txt +0 -13
  322. package/src/tool/websearch.ts +0 -150
  323. package/src/tool/websearch.txt +0 -14
  324. package/src/tool/write.ts +0 -80
  325. package/src/tool/write.txt +0 -8
  326. package/src/util/archive.ts +0 -16
  327. package/src/util/color.ts +0 -19
  328. package/src/util/context.ts +0 -25
  329. package/src/util/defer.ts +0 -12
  330. package/src/util/eventloop.ts +0 -20
  331. package/src/util/filesystem.ts +0 -93
  332. package/src/util/fn.ts +0 -11
  333. package/src/util/format.ts +0 -20
  334. package/src/util/iife.ts +0 -3
  335. package/src/util/keybind.ts +0 -103
  336. package/src/util/lazy.ts +0 -18
  337. package/src/util/locale.ts +0 -81
  338. package/src/util/lock.ts +0 -98
  339. package/src/util/log.ts +0 -180
  340. package/src/util/queue.ts +0 -32
  341. package/src/util/rpc.ts +0 -66
  342. package/src/util/scrap.ts +0 -10
  343. package/src/util/signal.ts +0 -12
  344. package/src/util/timeout.ts +0 -14
  345. package/src/util/token.ts +0 -7
  346. package/src/util/wildcard.ts +0 -56
  347. package/src/worktree/index.ts +0 -424
  348. package/sst-env.d.ts +0 -9
  349. package/test/acp/event-subscription.test.ts +0 -436
  350. package/test/agent/agent.test.ts +0 -638
  351. package/test/bun.test.ts +0 -53
  352. package/test/cli/github-action.test.ts +0 -129
  353. package/test/cli/github-remote.test.ts +0 -80
  354. package/test/cli/tui/transcript.test.ts +0 -297
  355. package/test/config/agent-color.test.ts +0 -66
  356. package/test/config/config.test.ts +0 -1414
  357. package/test/config/fixtures/empty-frontmatter.md +0 -4
  358. package/test/config/fixtures/frontmatter.md +0 -28
  359. package/test/config/fixtures/no-frontmatter.md +0 -1
  360. package/test/config/markdown.test.ts +0 -192
  361. package/test/file/ignore.test.ts +0 -10
  362. package/test/file/path-traversal.test.ts +0 -198
  363. package/test/fixture/fixture.ts +0 -45
  364. package/test/fixture/lsp/fake-lsp-server.js +0 -77
  365. package/test/ide/ide.test.ts +0 -82
  366. package/test/keybind.test.ts +0 -421
  367. package/test/lsp/client.test.ts +0 -95
  368. package/test/mcp/headers.test.ts +0 -153
  369. package/test/mcp/oauth-browser.test.ts +0 -261
  370. package/test/patch/patch.test.ts +0 -348
  371. package/test/permission/arity.test.ts +0 -33
  372. package/test/permission/next.test.ts +0 -652
  373. package/test/permission-task.test.ts +0 -319
  374. package/test/plugin/codex.test.ts +0 -123
  375. package/test/preload.ts +0 -65
  376. package/test/project/project.test.ts +0 -120
  377. package/test/provider/amazon-bedrock.test.ts +0 -268
  378. package/test/provider/gitlab-duo.test.ts +0 -286
  379. package/test/provider/provider.test.ts +0 -2149
  380. package/test/provider/transform.test.ts +0 -1596
  381. package/test/question/question.test.ts +0 -300
  382. package/test/scheduler.test.ts +0 -73
  383. package/test/server/session-list.test.ts +0 -39
  384. package/test/server/session-select.test.ts +0 -78
  385. package/test/session/compaction.test.ts +0 -293
  386. package/test/session/llm.test.ts +0 -90
  387. package/test/session/message-v2.test.ts +0 -662
  388. package/test/session/retry.test.ts +0 -131
  389. package/test/session/revert-compact.test.ts +0 -285
  390. package/test/session/session.test.ts +0 -71
  391. package/test/skill/skill.test.ts +0 -185
  392. package/test/snapshot/snapshot.test.ts +0 -939
  393. package/test/tool/__snapshots__/tool.test.ts.snap +0 -9
  394. package/test/tool/apply_patch.test.ts +0 -499
  395. package/test/tool/bash.test.ts +0 -320
  396. package/test/tool/external-directory.test.ts +0 -126
  397. package/test/tool/fixtures/large-image.png +0 -0
  398. package/test/tool/fixtures/models-api.json +0 -33453
  399. package/test/tool/grep.test.ts +0 -109
  400. package/test/tool/question.test.ts +0 -105
  401. package/test/tool/read.test.ts +0 -332
  402. package/test/tool/registry.test.ts +0 -76
  403. package/test/tool/truncation.test.ts +0 -159
  404. package/test/util/filesystem.test.ts +0 -39
  405. package/test/util/format.test.ts +0 -59
  406. package/test/util/iife.test.ts +0 -36
  407. package/test/util/lazy.test.ts +0 -50
  408. package/test/util/lock.test.ts +0 -72
  409. package/test/util/timeout.test.ts +0 -21
  410. package/test/util/wildcard.test.ts +0 -75
  411. package/tsconfig.json +0 -16
  412. /package/{script/postinstall.mjs → postinstall.mjs} +0 -0
@@ -1,4 +0,0 @@
1
- ---
2
- ---
3
-
4
- Content
@@ -1,28 +0,0 @@
1
- ---
2
- description: "This is a description wrapped in quotes"
3
- # field: this is a commented out field that should be ignored
4
- occupation: This man has the following occupation: Software Engineer
5
- title: 'Hello World'
6
- name: John "Doe"
7
-
8
- family: He has no 'family'
9
- summary: >
10
- This is a summary
11
- url: https://example.com:8080/path?query=value
12
- time: The time is 12:30:00 PM
13
- nested: First: Second: Third: Fourth
14
- quoted_colon: "Already quoted: no change needed"
15
- single_quoted_colon: 'Single quoted: also fine'
16
- mixed: He said "hello: world" and then left
17
- empty:
18
- dollar: Use $' and $& for special patterns
19
- ---
20
-
21
- Content that should not be parsed:
22
-
23
- fake_field: this is not yaml
24
- another: neither is this
25
- time: 10:30:00 AM
26
- url: https://should-not-be-parsed.com:3000
27
-
28
- The above lines look like YAML but are just content.
@@ -1 +0,0 @@
1
- Content
@@ -1,192 +0,0 @@
1
- import { expect, test, describe } from "bun:test"
2
- import { ConfigMarkdown } from "../../src/config/markdown"
3
-
4
- describe("ConfigMarkdown: normal template", () => {
5
- const template = `This is a @valid/path/to/a/file and it should also match at
6
- the beginning of a line:
7
-
8
- @another-valid/path/to/a/file
9
-
10
- but this is not:
11
-
12
- - Adds a "Co-authored-by:" footer which clarifies which AI agent
13
- helped create this commit, using an appropriate \`noreply@...\`
14
- or \`noreply@anthropic.com\` email address.
15
-
16
- We also need to deal with files followed by @commas, ones
17
- with @file-extensions.md, even @multiple.extensions.bak,
18
- hidden directories like @.config/ or files like @.bashrc
19
- and ones at the end of a sentence like @foo.md.
20
-
21
- Also shouldn't forget @/absolute/paths.txt with and @/without/extensions,
22
- as well as @~/home-files and @~/paths/under/home.txt.
23
-
24
- If the reference is \`@quoted/in/backticks\` then it shouldn't match at all.`
25
-
26
- const matches = ConfigMarkdown.files(template)
27
-
28
- test("should extract exactly 12 file references", () => {
29
- expect(matches.length).toBe(12)
30
- })
31
-
32
- test("should extract valid/path/to/a/file", () => {
33
- expect(matches[0][1]).toBe("valid/path/to/a/file")
34
- })
35
-
36
- test("should extract another-valid/path/to/a/file", () => {
37
- expect(matches[1][1]).toBe("another-valid/path/to/a/file")
38
- })
39
-
40
- test("should extract paths ignoring comma after", () => {
41
- expect(matches[2][1]).toBe("commas")
42
- })
43
-
44
- test("should extract a path with a file extension and comma after", () => {
45
- expect(matches[3][1]).toBe("file-extensions.md")
46
- })
47
-
48
- test("should extract a path with multiple dots and comma after", () => {
49
- expect(matches[4][1]).toBe("multiple.extensions.bak")
50
- })
51
-
52
- test("should extract hidden directory", () => {
53
- expect(matches[5][1]).toBe(".config/")
54
- })
55
-
56
- test("should extract hidden file", () => {
57
- expect(matches[6][1]).toBe(".bashrc")
58
- })
59
-
60
- test("should extract a file ignoring period at end of sentence", () => {
61
- expect(matches[7][1]).toBe("foo.md")
62
- })
63
-
64
- test("should extract an absolute path with an extension", () => {
65
- expect(matches[8][1]).toBe("/absolute/paths.txt")
66
- })
67
-
68
- test("should extract an absolute path without an extension", () => {
69
- expect(matches[9][1]).toBe("/without/extensions")
70
- })
71
-
72
- test("should extract an absolute path in home directory", () => {
73
- expect(matches[10][1]).toBe("~/home-files")
74
- })
75
-
76
- test("should extract an absolute path under home directory", () => {
77
- expect(matches[11][1]).toBe("~/paths/under/home.txt")
78
- })
79
-
80
- test("should not match when preceded by backtick", () => {
81
- const backtickTest = "This `@should/not/match` should be ignored"
82
- const backtickMatches = ConfigMarkdown.files(backtickTest)
83
- expect(backtickMatches.length).toBe(0)
84
- })
85
-
86
- test("should not match email addresses", () => {
87
- const emailTest = "Contact user@example.com for help"
88
- const emailMatches = ConfigMarkdown.files(emailTest)
89
- expect(emailMatches.length).toBe(0)
90
- })
91
- })
92
-
93
- describe("ConfigMarkdown: frontmatter parsing", async () => {
94
- const parsed = await ConfigMarkdown.parse(import.meta.dir + "/fixtures/frontmatter.md")
95
-
96
- test("should parse without throwing", () => {
97
- expect(parsed).toBeDefined()
98
- expect(parsed.data).toBeDefined()
99
- expect(parsed.content).toBeDefined()
100
- })
101
-
102
- test("should extract description field", () => {
103
- expect(parsed.data.description).toBe("This is a description wrapped in quotes")
104
- })
105
-
106
- test("should extract occupation field with colon in value", () => {
107
- expect(parsed.data.occupation).toBe("This man has the following occupation: Software Engineer\n")
108
- })
109
-
110
- test("should extract title field with single quotes", () => {
111
- expect(parsed.data.title).toBe("Hello World")
112
- })
113
-
114
- test("should extract name field with embedded quotes", () => {
115
- expect(parsed.data.name).toBe('John "Doe"')
116
- })
117
-
118
- test("should extract family field with embedded single quotes", () => {
119
- expect(parsed.data.family).toBe("He has no 'family'")
120
- })
121
-
122
- test("should extract multiline summary field", () => {
123
- expect(parsed.data.summary).toBe("This is a summary\n")
124
- })
125
-
126
- test("should not include commented fields in data", () => {
127
- expect(parsed.data.field).toBeUndefined()
128
- })
129
-
130
- test("should extract URL with port", () => {
131
- expect(parsed.data.url).toBe("https://example.com:8080/path?query=value\n")
132
- })
133
-
134
- test("should extract time with colons", () => {
135
- expect(parsed.data.time).toBe("The time is 12:30:00 PM\n")
136
- })
137
-
138
- test("should extract value with multiple colons", () => {
139
- expect(parsed.data.nested).toBe("First: Second: Third: Fourth\n")
140
- })
141
-
142
- test("should preserve already double-quoted values with colons", () => {
143
- expect(parsed.data.quoted_colon).toBe("Already quoted: no change needed")
144
- })
145
-
146
- test("should preserve already single-quoted values with colons", () => {
147
- expect(parsed.data.single_quoted_colon).toBe("Single quoted: also fine")
148
- })
149
-
150
- test("should extract value with quotes and colons mixed", () => {
151
- expect(parsed.data.mixed).toBe('He said "hello: world" and then left\n')
152
- })
153
-
154
- test("should handle empty values", () => {
155
- expect(parsed.data.empty).toBeNull()
156
- })
157
-
158
- test("should handle dollar sign replacement patterns literally", () => {
159
- expect(parsed.data.dollar).toBe("Use $' and $& for special patterns")
160
- })
161
-
162
- test("should not parse fake yaml from content", () => {
163
- expect(parsed.data.fake_field).toBeUndefined()
164
- expect(parsed.data.another).toBeUndefined()
165
- })
166
-
167
- test("should extract content after frontmatter without modification", () => {
168
- expect(parsed.content).toContain("Content that should not be parsed:")
169
- expect(parsed.content).toContain("fake_field: this is not yaml")
170
- expect(parsed.content).toContain("url: https://should-not-be-parsed.com:3000")
171
- })
172
- })
173
-
174
- describe("ConfigMarkdown: frontmatter parsing w/ empty frontmatter", async () => {
175
- const result = await ConfigMarkdown.parse(import.meta.dir + "/fixtures/empty-frontmatter.md")
176
-
177
- test("should parse without throwing", () => {
178
- expect(result).toBeDefined()
179
- expect(result.data).toEqual({})
180
- expect(result.content.trim()).toBe("Content")
181
- })
182
- })
183
-
184
- describe("ConfigMarkdown: frontmatter parsing w/ no frontmatter", async () => {
185
- const result = await ConfigMarkdown.parse(import.meta.dir + "/fixtures/no-frontmatter.md")
186
-
187
- test("should parse without throwing", () => {
188
- expect(result).toBeDefined()
189
- expect(result.data).toEqual({})
190
- expect(result.content.trim()).toBe("Content")
191
- })
192
- })
@@ -1,10 +0,0 @@
1
- import { test, expect } from "bun:test"
2
- import { FileIgnore } from "../../src/file/ignore"
3
-
4
- test("match nested and non-nested", () => {
5
- expect(FileIgnore.match("node_modules/index.js")).toBe(true)
6
- expect(FileIgnore.match("node_modules")).toBe(true)
7
- expect(FileIgnore.match("node_modules/")).toBe(true)
8
- expect(FileIgnore.match("node_modules/bar")).toBe(true)
9
- expect(FileIgnore.match("node_modules/bar/")).toBe(true)
10
- })
@@ -1,198 +0,0 @@
1
- import { test, expect, describe } from "bun:test"
2
- import path from "path"
3
- import fs from "fs/promises"
4
- import { Filesystem } from "../../src/util/filesystem"
5
- import { File } from "../../src/file"
6
- import { Instance } from "../../src/project/instance"
7
- import { tmpdir } from "../fixture/fixture"
8
-
9
- describe("Filesystem.contains", () => {
10
- test("allows paths within project", () => {
11
- expect(Filesystem.contains("/project", "/project/src")).toBe(true)
12
- expect(Filesystem.contains("/project", "/project/src/file.ts")).toBe(true)
13
- expect(Filesystem.contains("/project", "/project")).toBe(true)
14
- })
15
-
16
- test("blocks ../ traversal", () => {
17
- expect(Filesystem.contains("/project", "/project/../etc")).toBe(false)
18
- expect(Filesystem.contains("/project", "/project/src/../../etc")).toBe(false)
19
- expect(Filesystem.contains("/project", "/etc/passwd")).toBe(false)
20
- })
21
-
22
- test("blocks absolute paths outside project", () => {
23
- expect(Filesystem.contains("/project", "/etc/passwd")).toBe(false)
24
- expect(Filesystem.contains("/project", "/tmp/file")).toBe(false)
25
- expect(Filesystem.contains("/home/user/project", "/home/user/other")).toBe(false)
26
- })
27
-
28
- test("handles prefix collision edge cases", () => {
29
- expect(Filesystem.contains("/project", "/project-other/file")).toBe(false)
30
- expect(Filesystem.contains("/project", "/projectfile")).toBe(false)
31
- })
32
- })
33
-
34
- /*
35
- * Integration tests for File.read() and File.list() path traversal protection.
36
- *
37
- * These tests verify the HTTP API code path is protected. The HTTP endpoints
38
- * in server.ts (GET /file/content, GET /file) call File.read()/File.list()
39
- * directly - they do NOT go through ReadTool or the agent permission layer.
40
- *
41
- * This is a SEPARATE code path from ReadTool, which has its own checks.
42
- */
43
- describe("File.read path traversal protection", () => {
44
- test("rejects ../ traversal attempting to read /etc/passwd", async () => {
45
- await using tmp = await tmpdir({
46
- init: async (dir) => {
47
- await Bun.write(path.join(dir, "allowed.txt"), "allowed content")
48
- },
49
- })
50
-
51
- await Instance.provide({
52
- directory: tmp.path,
53
- fn: async () => {
54
- await expect(File.read("../../../etc/passwd")).rejects.toThrow("Access denied: path escapes project directory")
55
- },
56
- })
57
- })
58
-
59
- test("rejects deeply nested traversal", async () => {
60
- await using tmp = await tmpdir()
61
-
62
- await Instance.provide({
63
- directory: tmp.path,
64
- fn: async () => {
65
- await expect(File.read("src/nested/../../../../../../../etc/passwd")).rejects.toThrow(
66
- "Access denied: path escapes project directory",
67
- )
68
- },
69
- })
70
- })
71
-
72
- test("allows valid paths within project", async () => {
73
- await using tmp = await tmpdir({
74
- init: async (dir) => {
75
- await Bun.write(path.join(dir, "valid.txt"), "valid content")
76
- },
77
- })
78
-
79
- await Instance.provide({
80
- directory: tmp.path,
81
- fn: async () => {
82
- const result = await File.read("valid.txt")
83
- expect(result.content).toBe("valid content")
84
- },
85
- })
86
- })
87
- })
88
-
89
- describe("File.list path traversal protection", () => {
90
- test("rejects ../ traversal attempting to list /etc", async () => {
91
- await using tmp = await tmpdir()
92
-
93
- await Instance.provide({
94
- directory: tmp.path,
95
- fn: async () => {
96
- await expect(File.list("../../../etc")).rejects.toThrow("Access denied: path escapes project directory")
97
- },
98
- })
99
- })
100
-
101
- test("allows valid subdirectory listing", async () => {
102
- await using tmp = await tmpdir({
103
- init: async (dir) => {
104
- await Bun.write(path.join(dir, "subdir", "file.txt"), "content")
105
- },
106
- })
107
-
108
- await Instance.provide({
109
- directory: tmp.path,
110
- fn: async () => {
111
- const result = await File.list("subdir")
112
- expect(Array.isArray(result)).toBe(true)
113
- },
114
- })
115
- })
116
- })
117
-
118
- describe("Instance.containsPath", () => {
119
- test("returns true for path inside directory", async () => {
120
- await using tmp = await tmpdir({ git: true })
121
-
122
- await Instance.provide({
123
- directory: tmp.path,
124
- fn: () => {
125
- expect(Instance.containsPath(path.join(tmp.path, "foo.txt"))).toBe(true)
126
- expect(Instance.containsPath(path.join(tmp.path, "src", "file.ts"))).toBe(true)
127
- },
128
- })
129
- })
130
-
131
- test("returns true for path inside worktree but outside directory (monorepo subdirectory scenario)", async () => {
132
- await using tmp = await tmpdir({ git: true })
133
- const subdir = path.join(tmp.path, "packages", "lib")
134
- await fs.mkdir(subdir, { recursive: true })
135
-
136
- await Instance.provide({
137
- directory: subdir,
138
- fn: () => {
139
- // .opencode at worktree root, but we're running from packages/lib
140
- expect(Instance.containsPath(path.join(tmp.path, ".opencode", "state"))).toBe(true)
141
- // sibling package should also be accessible
142
- expect(Instance.containsPath(path.join(tmp.path, "packages", "other", "file.ts"))).toBe(true)
143
- // worktree root itself
144
- expect(Instance.containsPath(tmp.path)).toBe(true)
145
- },
146
- })
147
- })
148
-
149
- test("returns false for path outside both directory and worktree", async () => {
150
- await using tmp = await tmpdir({ git: true })
151
-
152
- await Instance.provide({
153
- directory: tmp.path,
154
- fn: () => {
155
- expect(Instance.containsPath("/etc/passwd")).toBe(false)
156
- expect(Instance.containsPath("/tmp/other-project")).toBe(false)
157
- },
158
- })
159
- })
160
-
161
- test("returns false for path with .. escaping worktree", async () => {
162
- await using tmp = await tmpdir({ git: true })
163
-
164
- await Instance.provide({
165
- directory: tmp.path,
166
- fn: () => {
167
- expect(Instance.containsPath(path.join(tmp.path, "..", "escape.txt"))).toBe(false)
168
- },
169
- })
170
- })
171
-
172
- test("handles directory === worktree (running from repo root)", async () => {
173
- await using tmp = await tmpdir({ git: true })
174
-
175
- await Instance.provide({
176
- directory: tmp.path,
177
- fn: () => {
178
- expect(Instance.directory).toBe(Instance.worktree)
179
- expect(Instance.containsPath(path.join(tmp.path, "file.txt"))).toBe(true)
180
- expect(Instance.containsPath("/etc/passwd")).toBe(false)
181
- },
182
- })
183
- })
184
-
185
- test("non-git project does not allow arbitrary paths via worktree='/'", async () => {
186
- await using tmp = await tmpdir() // no git: true
187
-
188
- await Instance.provide({
189
- directory: tmp.path,
190
- fn: () => {
191
- // worktree is "/" for non-git projects, but containsPath should NOT allow all paths
192
- expect(Instance.containsPath(path.join(tmp.path, "file.txt"))).toBe(true)
193
- expect(Instance.containsPath("/etc/passwd")).toBe(false)
194
- expect(Instance.containsPath("/tmp/other")).toBe(false)
195
- },
196
- })
197
- })
198
- })
@@ -1,45 +0,0 @@
1
- import { $ } from "bun"
2
- import * as fs from "fs/promises"
3
- import os from "os"
4
- import path from "path"
5
- import type { Config } from "../../src/config/config"
6
-
7
- // Strip null bytes from paths (defensive fix for CI environment issues)
8
- function sanitizePath(p: string): string {
9
- return p.replace(/\0/g, "")
10
- }
11
-
12
- type TmpDirOptions<T> = {
13
- git?: boolean
14
- config?: Partial<Config.Info>
15
- init?: (dir: string) => Promise<T>
16
- dispose?: (dir: string) => Promise<T>
17
- }
18
- export async function tmpdir<T>(options?: TmpDirOptions<T>) {
19
- const dirpath = sanitizePath(path.join(os.tmpdir(), "opencode-test-" + Math.random().toString(36).slice(2)))
20
- await fs.mkdir(dirpath, { recursive: true })
21
- if (options?.git) {
22
- await $`git init`.cwd(dirpath).quiet()
23
- await $`git commit --allow-empty -m "root commit ${dirpath}"`.cwd(dirpath).quiet()
24
- }
25
- if (options?.config) {
26
- await Bun.write(
27
- path.join(dirpath, "opencode.json"),
28
- JSON.stringify({
29
- $schema: "https://opencode.ai/config.json",
30
- ...options.config,
31
- }),
32
- )
33
- }
34
- const extra = await options?.init?.(dirpath)
35
- const realpath = sanitizePath(await fs.realpath(dirpath))
36
- const result = {
37
- [Symbol.asyncDispose]: async () => {
38
- await options?.dispose?.(dirpath)
39
- // await fs.rm(dirpath, { recursive: true, force: true })
40
- },
41
- path: realpath,
42
- extra: extra as T,
43
- }
44
- return result
45
- }
@@ -1,77 +0,0 @@
1
- // Simple JSON-RPC 2.0 LSP-like fake server over stdio
2
- // Implements a minimal LSP handshake and triggers a request upon notification
3
-
4
- const net = require("net")
5
-
6
- let nextId = 1
7
-
8
- function encode(message) {
9
- const json = JSON.stringify(message)
10
- const header = `Content-Length: ${Buffer.byteLength(json, "utf8")}\r\n\r\n`
11
- return Buffer.concat([Buffer.from(header, "utf8"), Buffer.from(json, "utf8")])
12
- }
13
-
14
- function decodeFrames(buffer) {
15
- const results = []
16
- let idx
17
- while ((idx = buffer.indexOf("\r\n\r\n")) !== -1) {
18
- const header = buffer.slice(0, idx).toString("utf8")
19
- const m = /Content-Length:\s*(\d+)/i.exec(header)
20
- const len = m ? parseInt(m[1], 10) : 0
21
- const bodyStart = idx + 4
22
- const bodyEnd = bodyStart + len
23
- if (buffer.length < bodyEnd) break
24
- const body = buffer.slice(bodyStart, bodyEnd).toString("utf8")
25
- results.push(body)
26
- buffer = buffer.slice(bodyEnd)
27
- }
28
- return { messages: results, rest: buffer }
29
- }
30
-
31
- let readBuffer = Buffer.alloc(0)
32
-
33
- process.stdin.on("data", (chunk) => {
34
- readBuffer = Buffer.concat([readBuffer, chunk])
35
- const { messages, rest } = decodeFrames(readBuffer)
36
- readBuffer = rest
37
- for (const m of messages) handle(m)
38
- })
39
-
40
- function send(msg) {
41
- process.stdout.write(encode(msg))
42
- }
43
-
44
- function sendRequest(method, params) {
45
- const id = nextId++
46
- send({ jsonrpc: "2.0", id, method, params })
47
- return id
48
- }
49
-
50
- function handle(raw) {
51
- let data
52
- try {
53
- data = JSON.parse(raw)
54
- } catch {
55
- return
56
- }
57
- if (data.method === "initialize") {
58
- send({ jsonrpc: "2.0", id: data.id, result: { capabilities: {} } })
59
- return
60
- }
61
- if (data.method === "initialized") {
62
- return
63
- }
64
- if (data.method === "workspace/didChangeConfiguration") {
65
- return
66
- }
67
- if (data.method === "test/trigger") {
68
- const method = data.params && data.params.method
69
- if (method) sendRequest(method, {})
70
- return
71
- }
72
- if (typeof data.id !== "undefined") {
73
- // Respond OK to any request from client to keep transport flowing
74
- send({ jsonrpc: "2.0", id: data.id, result: null })
75
- return
76
- }
77
- }
@@ -1,82 +0,0 @@
1
- import { describe, expect, test, afterEach } from "bun:test"
2
- import { Ide } from "../../src/ide"
3
-
4
- describe("ide", () => {
5
- const original = structuredClone(process.env)
6
-
7
- afterEach(() => {
8
- Object.keys(process.env).forEach((key) => {
9
- delete process.env[key]
10
- })
11
- Object.assign(process.env, original)
12
- })
13
-
14
- test("should detect Visual Studio Code", () => {
15
- process.env["TERM_PROGRAM"] = "vscode"
16
- process.env["GIT_ASKPASS"] = "/path/to/Visual Studio Code.app/Contents/Resources/app/extensions/git/dist/askpass.sh"
17
-
18
- expect(Ide.ide()).toBe("Visual Studio Code")
19
- })
20
-
21
- test("should detect Visual Studio Code Insiders", () => {
22
- process.env["TERM_PROGRAM"] = "vscode"
23
- process.env["GIT_ASKPASS"] =
24
- "/Applications/Visual Studio Code - Insiders.app/Contents/Resources/app/extensions/git/dist/askpass.sh"
25
-
26
- expect(Ide.ide()).toBe("Visual Studio Code - Insiders")
27
- })
28
-
29
- test("should detect Cursor", () => {
30
- process.env["TERM_PROGRAM"] = "vscode"
31
- process.env["GIT_ASKPASS"] = "/path/to/Cursor.app/Contents/Resources/app/extensions/git/dist/askpass.sh"
32
-
33
- expect(Ide.ide()).toBe("Cursor")
34
- })
35
-
36
- test("should detect VSCodium", () => {
37
- process.env["TERM_PROGRAM"] = "vscode"
38
- process.env["GIT_ASKPASS"] = "/path/to/VSCodium.app/Contents/Resources/app/extensions/git/dist/askpass.sh"
39
-
40
- expect(Ide.ide()).toBe("VSCodium")
41
- })
42
-
43
- test("should detect Windsurf", () => {
44
- process.env["TERM_PROGRAM"] = "vscode"
45
- process.env["GIT_ASKPASS"] = "/path/to/Windsurf.app/Contents/Resources/app/extensions/git/dist/askpass.sh"
46
-
47
- expect(Ide.ide()).toBe("Windsurf")
48
- })
49
-
50
- test("should return unknown when TERM_PROGRAM is not vscode", () => {
51
- process.env["TERM_PROGRAM"] = "iTerm2"
52
- process.env["GIT_ASKPASS"] =
53
- "/Applications/Visual Studio Code - Insiders.app/Contents/Resources/app/extensions/git/dist/askpass.sh"
54
-
55
- expect(Ide.ide()).toBe("unknown")
56
- })
57
-
58
- test("should return unknown when GIT_ASKPASS does not contain IDE name", () => {
59
- process.env["TERM_PROGRAM"] = "vscode"
60
- process.env["GIT_ASKPASS"] = "/path/to/unknown/askpass.sh"
61
-
62
- expect(Ide.ide()).toBe("unknown")
63
- })
64
-
65
- test("should recognize vscode-insiders OPENCODE_CALLER", () => {
66
- process.env["OPENCODE_CALLER"] = "vscode-insiders"
67
-
68
- expect(Ide.alreadyInstalled()).toBe(true)
69
- })
70
-
71
- test("should recognize vscode OPENCODE_CALLER", () => {
72
- process.env["OPENCODE_CALLER"] = "vscode"
73
-
74
- expect(Ide.alreadyInstalled()).toBe(true)
75
- })
76
-
77
- test("should return false for unknown OPENCODE_CALLER", () => {
78
- process.env["OPENCODE_CALLER"] = "unknown"
79
-
80
- expect(Ide.alreadyInstalled()).toBe(false)
81
- })
82
- })