@stonerzju/opencode 1.2.17 → 1.2.19

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 (261) hide show
  1. package/bin/opencode +29 -157
  2. package/package.json +29 -29
  3. package/src/acp/agent.ts +4 -4
  4. package/src/acp/session.ts +1 -1
  5. package/src/agent/agent.ts +3 -3
  6. package/src/bun/index.ts +2 -2
  7. package/src/cli/cmd/acp.ts +3 -3
  8. package/src/cli/cmd/debug/file.ts +1 -1
  9. package/src/cli/cmd/github.ts +2 -2
  10. package/src/cli/cmd/pr.ts +1 -1
  11. package/src/cli/cmd/tui/app.tsx +24 -24
  12. package/src/cli/cmd/tui/attach.ts +3 -3
  13. package/src/cli/cmd/tui/component/dialog-agent.tsx +3 -3
  14. package/src/cli/cmd/tui/component/dialog-command.tsx +3 -3
  15. package/src/cli/cmd/tui/component/dialog-mcp.tsx +5 -5
  16. package/src/cli/cmd/tui/component/dialog-model.tsx +4 -4
  17. package/src/cli/cmd/tui/component/dialog-provider.tsx +4 -4
  18. package/src/cli/cmd/tui/component/dialog-session-list.tsx +5 -5
  19. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +3 -3
  20. package/src/cli/cmd/tui/component/dialog-skill.tsx +3 -3
  21. package/src/cli/cmd/tui/component/dialog-stash.tsx +3 -3
  22. package/src/cli/cmd/tui/component/dialog-status.tsx +2 -2
  23. package/src/cli/cmd/tui/component/dialog-tag.tsx +3 -3
  24. package/src/cli/cmd/tui/component/logo.tsx +2 -2
  25. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +6 -6
  26. package/src/cli/cmd/tui/component/prompt/frecency.tsx +2 -2
  27. package/src/cli/cmd/tui/component/prompt/history.tsx +2 -2
  28. package/src/cli/cmd/tui/component/prompt/index.tsx +14 -14
  29. package/src/cli/cmd/tui/component/prompt/stash.tsx +2 -2
  30. package/src/cli/cmd/tui/component/textarea-keybindings.ts +1 -1
  31. package/src/cli/cmd/tui/component/tips.tsx +1 -1
  32. package/src/cli/cmd/tui/context/directory.ts +1 -1
  33. package/src/cli/cmd/tui/context/exit.tsx +1 -1
  34. package/src/cli/cmd/tui/context/keybind.tsx +2 -2
  35. package/src/cli/cmd/tui/context/kv.tsx +2 -2
  36. package/src/cli/cmd/tui/context/local.tsx +6 -6
  37. package/src/cli/cmd/tui/context/sync.tsx +4 -4
  38. package/src/cli/cmd/tui/context/theme/opencode.json +245 -0
  39. package/src/cli/cmd/tui/context/theme.tsx +2 -2
  40. package/src/cli/cmd/tui/context/tui-config.tsx +1 -1
  41. package/src/cli/cmd/tui/event.ts +2 -2
  42. package/src/cli/cmd/tui/routes/home.tsx +6 -6
  43. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +6 -6
  44. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +6 -6
  45. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +2 -2
  46. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +3 -3
  47. package/src/cli/cmd/tui/routes/session/header.tsx +5 -5
  48. package/src/cli/cmd/tui/routes/session/index.tsx +32 -32
  49. package/src/cli/cmd/tui/routes/session/permission.tsx +4 -4
  50. package/src/cli/cmd/tui/routes/session/sidebar.tsx +4 -4
  51. package/src/cli/cmd/tui/thread.ts +9 -9
  52. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +1 -1
  53. package/src/cli/cmd/tui/ui/dialog-help.tsx +2 -2
  54. package/src/cli/cmd/tui/ui/dialog-select.tsx +5 -5
  55. package/src/cli/cmd/tui/ui/dialog.tsx +3 -3
  56. package/src/cli/cmd/tui/ui/toast.tsx +1 -1
  57. package/src/cli/cmd/tui/util/editor.ts +3 -3
  58. package/src/cli/cmd/tui/util/transcript.ts +1 -1
  59. package/src/cli/cmd/tui/worker.ts +10 -10
  60. package/src/cli/error.ts +1 -1
  61. package/src/cli/ui.ts +1 -1
  62. package/src/cli/upgrade.ts +4 -4
  63. package/src/command/index.ts +1 -1
  64. package/src/config/config.ts +10 -10
  65. package/src/config/markdown.ts +1 -1
  66. package/src/config/migrate-tui-config.ts +5 -5
  67. package/src/config/paths.ts +4 -4
  68. package/src/config/tui.ts +4 -4
  69. package/src/control/control.sql.ts +1 -1
  70. package/src/control/index.ts +1 -1
  71. package/src/control-plane/adaptors/worktree.ts +1 -1
  72. package/src/control-plane/session-proxy-middleware.ts +1 -1
  73. package/src/control-plane/workspace.sql.ts +1 -1
  74. package/src/control-plane/workspace.ts +7 -7
  75. package/src/file/index.ts +1 -1
  76. package/src/file/ripgrep.ts +2 -2
  77. package/src/file/watcher.ts +5 -5
  78. package/src/format/formatter.ts +1 -1
  79. package/src/ide/index.ts +3 -3
  80. package/src/index.ts +1 -1
  81. package/src/installation/index.ts +3 -3
  82. package/src/lsp/client.ts +3 -3
  83. package/src/lsp/index.ts +3 -3
  84. package/src/mcp/index.ts +4 -4
  85. package/src/permission/index.ts +2 -2
  86. package/src/permission/next.ts +10 -10
  87. package/src/plugin/codex.ts +1 -1
  88. package/src/plugin/copilot.ts +2 -2
  89. package/src/plugin/index.ts +1 -1
  90. package/src/project/bootstrap.ts +2 -2
  91. package/src/project/instance.ts +4 -4
  92. package/src/project/project.sql.ts +1 -1
  93. package/src/project/project.ts +5 -5
  94. package/src/project/state.ts +1 -1
  95. package/src/project/vcs.ts +4 -4
  96. package/src/provider/auth.ts +4 -4
  97. package/src/provider/error.ts +1 -1
  98. package/src/provider/models-snapshot.ts +2 -0
  99. package/src/provider/models.ts +1 -1
  100. package/src/provider/provider.ts +2 -2
  101. package/src/provider/transform.ts +2 -2
  102. package/src/pty/index.ts +5 -5
  103. package/src/question/index.ts +5 -5
  104. package/src/server/event.ts +1 -1
  105. package/src/server/mdns.ts +1 -1
  106. package/src/server/routes/global.ts +3 -3
  107. package/src/server/routes/permission.ts +1 -1
  108. package/src/server/routes/pty.ts +1 -1
  109. package/src/server/routes/session.ts +4 -4
  110. package/src/server/routes/tui.ts +1 -1
  111. package/src/server/server.ts +3 -3
  112. package/src/session/compaction.ts +7 -7
  113. package/src/session/index.ts +10 -10
  114. package/src/session/instruction.ts +1 -1
  115. package/src/session/llm.ts +11 -11
  116. package/src/session/message-v2.ts +10 -10
  117. package/src/session/message.ts +1 -1
  118. package/src/session/processor.ts +10 -10
  119. package/src/session/prompt.ts +8 -8
  120. package/src/session/retry.ts +2 -2
  121. package/src/session/revert.ts +1 -1
  122. package/src/session/session.sql.ts +3 -3
  123. package/src/session/status.ts +3 -3
  124. package/src/session/summary.ts +5 -5
  125. package/src/session/system.ts +1 -1
  126. package/src/session/todo.ts +2 -2
  127. package/src/share/share-next.ts +7 -7
  128. package/src/share/share.sql.ts +1 -1
  129. package/src/shell/shell.ts +3 -3
  130. package/src/skill/skill.ts +6 -6
  131. package/src/storage/db.ts +1 -1
  132. package/src/storage/storage.ts +1 -1
  133. package/src/tool/bash.ts +6 -6
  134. package/src/tool/edit.ts +1 -1
  135. package/src/tool/registry.ts +2 -2
  136. package/src/tool/skill.ts +1 -1
  137. package/src/tool/task.ts +3 -3
  138. package/src/util/array.ts +10 -0
  139. package/src/util/binary.ts +41 -0
  140. package/src/util/encode.ts +51 -0
  141. package/src/util/error.ts +54 -0
  142. package/src/util/identifier.ts +48 -0
  143. package/src/util/lazy.ts +4 -16
  144. package/src/util/path.ts +37 -0
  145. package/src/util/retry.ts +41 -0
  146. package/src/util/slug.ts +74 -0
  147. package/src/worktree/index.ts +3 -3
  148. package/AGENTS.md +0 -10
  149. package/BUN_SHELL_MIGRATION_PLAN.md +0 -136
  150. package/Dockerfile +0 -18
  151. package/README.md +0 -15
  152. package/bunfig.toml +0 -7
  153. package/drizzle.config.ts +0 -10
  154. package/script/build.ts +0 -224
  155. package/script/check-migrations.ts +0 -16
  156. package/script/postinstall.mjs +0 -131
  157. package/script/publish.ts +0 -181
  158. package/script/schema.ts +0 -63
  159. package/script/seed-e2e.ts +0 -50
  160. package/sst-env.d.ts +0 -10
  161. package/test/AGENTS.md +0 -81
  162. package/test/acp/agent-interface.test.ts +0 -51
  163. package/test/acp/event-subscription.test.ts +0 -683
  164. package/test/agent/agent.test.ts +0 -689
  165. package/test/bun.test.ts +0 -53
  166. package/test/cli/github-action.test.ts +0 -197
  167. package/test/cli/github-remote.test.ts +0 -80
  168. package/test/cli/import.test.ts +0 -38
  169. package/test/cli/plugin-auth-picker.test.ts +0 -120
  170. package/test/cli/tui/transcript.test.ts +0 -322
  171. package/test/config/agent-color.test.ts +0 -71
  172. package/test/config/config.test.ts +0 -1886
  173. package/test/config/fixtures/empty-frontmatter.md +0 -4
  174. package/test/config/fixtures/frontmatter.md +0 -28
  175. package/test/config/fixtures/markdown-header.md +0 -11
  176. package/test/config/fixtures/no-frontmatter.md +0 -1
  177. package/test/config/fixtures/weird-model-id.md +0 -13
  178. package/test/config/markdown.test.ts +0 -228
  179. package/test/config/tui.test.ts +0 -510
  180. package/test/control-plane/session-proxy-middleware.test.ts +0 -147
  181. package/test/control-plane/sse.test.ts +0 -56
  182. package/test/control-plane/workspace-server-sse.test.ts +0 -65
  183. package/test/control-plane/workspace-sync.test.ts +0 -97
  184. package/test/file/ignore.test.ts +0 -10
  185. package/test/file/index.test.ts +0 -394
  186. package/test/file/path-traversal.test.ts +0 -198
  187. package/test/file/ripgrep.test.ts +0 -39
  188. package/test/file/time.test.ts +0 -361
  189. package/test/fixture/db.ts +0 -11
  190. package/test/fixture/fixture.ts +0 -45
  191. package/test/fixture/lsp/fake-lsp-server.js +0 -77
  192. package/test/fixture/skills/agents-sdk/SKILL.md +0 -152
  193. package/test/fixture/skills/agents-sdk/references/callable.md +0 -92
  194. package/test/fixture/skills/cloudflare/SKILL.md +0 -211
  195. package/test/fixture/skills/index.json +0 -6
  196. package/test/ide/ide.test.ts +0 -82
  197. package/test/keybind.test.ts +0 -421
  198. package/test/lsp/client.test.ts +0 -95
  199. package/test/mcp/headers.test.ts +0 -153
  200. package/test/mcp/oauth-browser.test.ts +0 -249
  201. package/test/memory/abort-leak.test.ts +0 -136
  202. package/test/patch/patch.test.ts +0 -348
  203. package/test/permission/arity.test.ts +0 -33
  204. package/test/permission/next.test.ts +0 -689
  205. package/test/permission-task.test.ts +0 -319
  206. package/test/plugin/auth-override.test.ts +0 -44
  207. package/test/plugin/codex.test.ts +0 -123
  208. package/test/preload.ts +0 -80
  209. package/test/project/project.test.ts +0 -348
  210. package/test/project/worktree-remove.test.ts +0 -65
  211. package/test/provider/amazon-bedrock.test.ts +0 -446
  212. package/test/provider/copilot/convert-to-copilot-messages.test.ts +0 -523
  213. package/test/provider/copilot/copilot-chat-model.test.ts +0 -592
  214. package/test/provider/gitlab-duo.test.ts +0 -262
  215. package/test/provider/provider.test.ts +0 -2220
  216. package/test/provider/transform.test.ts +0 -2353
  217. package/test/pty/pty-output-isolation.test.ts +0 -140
  218. package/test/question/question.test.ts +0 -300
  219. package/test/scheduler.test.ts +0 -73
  220. package/test/server/global-session-list.test.ts +0 -89
  221. package/test/server/session-list.test.ts +0 -90
  222. package/test/server/session-select.test.ts +0 -78
  223. package/test/session/compaction.test.ts +0 -423
  224. package/test/session/instruction.test.ts +0 -170
  225. package/test/session/llm.test.ts +0 -667
  226. package/test/session/message-v2.test.ts +0 -924
  227. package/test/session/prompt.test.ts +0 -211
  228. package/test/session/retry.test.ts +0 -188
  229. package/test/session/revert-compact.test.ts +0 -285
  230. package/test/session/session.test.ts +0 -71
  231. package/test/session/structured-output-integration.test.ts +0 -233
  232. package/test/session/structured-output.test.ts +0 -385
  233. package/test/skill/discovery.test.ts +0 -110
  234. package/test/skill/skill.test.ts +0 -388
  235. package/test/snapshot/snapshot.test.ts +0 -1180
  236. package/test/storage/json-migration.test.ts +0 -846
  237. package/test/tool/__snapshots__/tool.test.ts.snap +0 -9
  238. package/test/tool/apply_patch.test.ts +0 -566
  239. package/test/tool/bash.test.ts +0 -402
  240. package/test/tool/edit.test.ts +0 -496
  241. package/test/tool/external-directory.test.ts +0 -127
  242. package/test/tool/fixtures/large-image.png +0 -0
  243. package/test/tool/fixtures/models-api.json +0 -38413
  244. package/test/tool/grep.test.ts +0 -110
  245. package/test/tool/question.test.ts +0 -107
  246. package/test/tool/read.test.ts +0 -504
  247. package/test/tool/registry.test.ts +0 -122
  248. package/test/tool/skill.test.ts +0 -112
  249. package/test/tool/truncation.test.ts +0 -160
  250. package/test/tool/webfetch.test.ts +0 -100
  251. package/test/tool/write.test.ts +0 -348
  252. package/test/util/filesystem.test.ts +0 -443
  253. package/test/util/format.test.ts +0 -59
  254. package/test/util/glob.test.ts +0 -164
  255. package/test/util/iife.test.ts +0 -36
  256. package/test/util/lazy.test.ts +0 -50
  257. package/test/util/lock.test.ts +0 -72
  258. package/test/util/process.test.ts +0 -59
  259. package/test/util/timeout.test.ts +0 -21
  260. package/test/util/wildcard.test.ts +0 -90
  261. package/tsconfig.json +0 -16
@@ -1,421 +0,0 @@
1
- import { describe, test, expect } from "bun:test"
2
- import { Keybind } from "../src/util/keybind"
3
-
4
- describe("Keybind.toString", () => {
5
- test("should convert simple key to string", () => {
6
- const info: Keybind.Info = { ctrl: false, meta: false, shift: false, leader: false, name: "f" }
7
- expect(Keybind.toString(info)).toBe("f")
8
- })
9
-
10
- test("should convert ctrl modifier to string", () => {
11
- const info: Keybind.Info = { ctrl: true, meta: false, shift: false, leader: false, name: "x" }
12
- expect(Keybind.toString(info)).toBe("ctrl+x")
13
- })
14
-
15
- test("should convert leader key to string", () => {
16
- const info: Keybind.Info = { ctrl: false, meta: false, shift: false, leader: true, name: "f" }
17
- expect(Keybind.toString(info)).toBe("<leader> f")
18
- })
19
-
20
- test("should convert multiple modifiers to string", () => {
21
- const info: Keybind.Info = { ctrl: true, meta: true, shift: false, leader: false, name: "g" }
22
- expect(Keybind.toString(info)).toBe("ctrl+alt+g")
23
- })
24
-
25
- test("should convert all modifiers to string", () => {
26
- const info: Keybind.Info = { ctrl: true, meta: true, shift: true, leader: true, name: "h" }
27
- expect(Keybind.toString(info)).toBe("<leader> ctrl+alt+shift+h")
28
- })
29
-
30
- test("should convert shift modifier to string", () => {
31
- const info: Keybind.Info = {
32
- ctrl: false,
33
- meta: false,
34
- shift: true,
35
- leader: false,
36
- name: "return",
37
- }
38
- expect(Keybind.toString(info)).toBe("shift+return")
39
- })
40
-
41
- test("should convert function key to string", () => {
42
- const info: Keybind.Info = { ctrl: false, meta: false, shift: false, leader: false, name: "f2" }
43
- expect(Keybind.toString(info)).toBe("f2")
44
- })
45
-
46
- test("should convert special key to string", () => {
47
- const info: Keybind.Info = {
48
- ctrl: false,
49
- meta: false,
50
- shift: false,
51
- leader: false,
52
- name: "pgup",
53
- }
54
- expect(Keybind.toString(info)).toBe("pgup")
55
- })
56
-
57
- test("should handle empty name", () => {
58
- const info: Keybind.Info = { ctrl: true, meta: false, shift: false, leader: false, name: "" }
59
- expect(Keybind.toString(info)).toBe("ctrl")
60
- })
61
-
62
- test("should handle only modifiers", () => {
63
- const info: Keybind.Info = { ctrl: true, meta: true, shift: true, leader: true, name: "" }
64
- expect(Keybind.toString(info)).toBe("<leader> ctrl+alt+shift")
65
- })
66
-
67
- test("should handle only leader with no other parts", () => {
68
- const info: Keybind.Info = { ctrl: false, meta: false, shift: false, leader: true, name: "" }
69
- expect(Keybind.toString(info)).toBe("<leader>")
70
- })
71
-
72
- test("should convert super modifier to string", () => {
73
- const info: Keybind.Info = { ctrl: false, meta: false, shift: false, super: true, leader: false, name: "z" }
74
- expect(Keybind.toString(info)).toBe("super+z")
75
- })
76
-
77
- test("should convert super+shift modifier to string", () => {
78
- const info: Keybind.Info = { ctrl: false, meta: false, shift: true, super: true, leader: false, name: "z" }
79
- expect(Keybind.toString(info)).toBe("super+shift+z")
80
- })
81
-
82
- test("should handle super with ctrl modifier", () => {
83
- const info: Keybind.Info = { ctrl: true, meta: false, shift: false, super: true, leader: false, name: "a" }
84
- expect(Keybind.toString(info)).toBe("ctrl+super+a")
85
- })
86
-
87
- test("should handle super with all modifiers", () => {
88
- const info: Keybind.Info = { ctrl: true, meta: true, shift: true, super: true, leader: false, name: "x" }
89
- expect(Keybind.toString(info)).toBe("ctrl+alt+super+shift+x")
90
- })
91
-
92
- test("should handle undefined super field (omitted)", () => {
93
- const info: Keybind.Info = { ctrl: true, meta: false, shift: false, leader: false, name: "c" }
94
- expect(Keybind.toString(info)).toBe("ctrl+c")
95
- })
96
- })
97
-
98
- describe("Keybind.match", () => {
99
- test("should match identical keybinds", () => {
100
- const a: Keybind.Info = { ctrl: true, meta: false, shift: false, leader: false, name: "x" }
101
- const b: Keybind.Info = { ctrl: true, meta: false, shift: false, leader: false, name: "x" }
102
- expect(Keybind.match(a, b)).toBe(true)
103
- })
104
-
105
- test("should not match different key names", () => {
106
- const a: Keybind.Info = { ctrl: true, meta: false, shift: false, leader: false, name: "x" }
107
- const b: Keybind.Info = { ctrl: true, meta: false, shift: false, leader: false, name: "y" }
108
- expect(Keybind.match(a, b)).toBe(false)
109
- })
110
-
111
- test("should not match different modifiers", () => {
112
- const a: Keybind.Info = { ctrl: true, meta: false, shift: false, leader: false, name: "x" }
113
- const b: Keybind.Info = { ctrl: false, meta: false, shift: false, leader: false, name: "x" }
114
- expect(Keybind.match(a, b)).toBe(false)
115
- })
116
-
117
- test("should match leader keybinds", () => {
118
- const a: Keybind.Info = { ctrl: false, meta: false, shift: false, leader: true, name: "f" }
119
- const b: Keybind.Info = { ctrl: false, meta: false, shift: false, leader: true, name: "f" }
120
- expect(Keybind.match(a, b)).toBe(true)
121
- })
122
-
123
- test("should not match leader vs non-leader", () => {
124
- const a: Keybind.Info = { ctrl: false, meta: false, shift: false, leader: true, name: "f" }
125
- const b: Keybind.Info = { ctrl: false, meta: false, shift: false, leader: false, name: "f" }
126
- expect(Keybind.match(a, b)).toBe(false)
127
- })
128
-
129
- test("should match complex keybinds", () => {
130
- const a: Keybind.Info = { ctrl: true, meta: true, shift: false, leader: false, name: "g" }
131
- const b: Keybind.Info = { ctrl: true, meta: true, shift: false, leader: false, name: "g" }
132
- expect(Keybind.match(a, b)).toBe(true)
133
- })
134
-
135
- test("should not match with one modifier different", () => {
136
- const a: Keybind.Info = { ctrl: true, meta: true, shift: false, leader: false, name: "g" }
137
- const b: Keybind.Info = { ctrl: true, meta: true, shift: true, leader: false, name: "g" }
138
- expect(Keybind.match(a, b)).toBe(false)
139
- })
140
-
141
- test("should match simple key without modifiers", () => {
142
- const a: Keybind.Info = { ctrl: false, meta: false, shift: false, leader: false, name: "a" }
143
- const b: Keybind.Info = { ctrl: false, meta: false, shift: false, leader: false, name: "a" }
144
- expect(Keybind.match(a, b)).toBe(true)
145
- })
146
-
147
- test("should match super modifier keybinds", () => {
148
- const a: Keybind.Info = { ctrl: false, meta: false, shift: false, super: true, leader: false, name: "z" }
149
- const b: Keybind.Info = { ctrl: false, meta: false, shift: false, super: true, leader: false, name: "z" }
150
- expect(Keybind.match(a, b)).toBe(true)
151
- })
152
-
153
- test("should not match super vs non-super", () => {
154
- const a: Keybind.Info = { ctrl: false, meta: false, shift: false, super: true, leader: false, name: "z" }
155
- const b: Keybind.Info = { ctrl: false, meta: false, shift: false, super: false, leader: false, name: "z" }
156
- expect(Keybind.match(a, b)).toBe(false)
157
- })
158
-
159
- test("should match undefined super with false super", () => {
160
- const a: Keybind.Info = { ctrl: true, meta: false, shift: false, leader: false, name: "c" }
161
- const b: Keybind.Info = { ctrl: true, meta: false, shift: false, super: false, leader: false, name: "c" }
162
- expect(Keybind.match(a, b)).toBe(true)
163
- })
164
-
165
- test("should match super+shift combination", () => {
166
- const a: Keybind.Info = { ctrl: false, meta: false, shift: true, super: true, leader: false, name: "z" }
167
- const b: Keybind.Info = { ctrl: false, meta: false, shift: true, super: true, leader: false, name: "z" }
168
- expect(Keybind.match(a, b)).toBe(true)
169
- })
170
-
171
- test("should not match when only super differs", () => {
172
- const a: Keybind.Info = { ctrl: true, meta: true, shift: true, super: true, leader: false, name: "a" }
173
- const b: Keybind.Info = { ctrl: true, meta: true, shift: true, super: false, leader: false, name: "a" }
174
- expect(Keybind.match(a, b)).toBe(false)
175
- })
176
- })
177
-
178
- describe("Keybind.parse", () => {
179
- test("should parse simple key", () => {
180
- const result = Keybind.parse("f")
181
- expect(result).toEqual([
182
- {
183
- ctrl: false,
184
- meta: false,
185
- shift: false,
186
- leader: false,
187
- name: "f",
188
- },
189
- ])
190
- })
191
-
192
- test("should parse leader key syntax", () => {
193
- const result = Keybind.parse("<leader>f")
194
- expect(result).toEqual([
195
- {
196
- ctrl: false,
197
- meta: false,
198
- shift: false,
199
- leader: true,
200
- name: "f",
201
- },
202
- ])
203
- })
204
-
205
- test("should parse ctrl modifier", () => {
206
- const result = Keybind.parse("ctrl+x")
207
- expect(result).toEqual([
208
- {
209
- ctrl: true,
210
- meta: false,
211
- shift: false,
212
- leader: false,
213
- name: "x",
214
- },
215
- ])
216
- })
217
-
218
- test("should parse multiple modifiers", () => {
219
- const result = Keybind.parse("ctrl+alt+u")
220
- expect(result).toEqual([
221
- {
222
- ctrl: true,
223
- meta: true,
224
- shift: false,
225
- leader: false,
226
- name: "u",
227
- },
228
- ])
229
- })
230
-
231
- test("should parse shift modifier", () => {
232
- const result = Keybind.parse("shift+f2")
233
- expect(result).toEqual([
234
- {
235
- ctrl: false,
236
- meta: false,
237
- shift: true,
238
- leader: false,
239
- name: "f2",
240
- },
241
- ])
242
- })
243
-
244
- test("should parse meta/alt modifier", () => {
245
- const result = Keybind.parse("meta+g")
246
- expect(result).toEqual([
247
- {
248
- ctrl: false,
249
- meta: true,
250
- shift: false,
251
- leader: false,
252
- name: "g",
253
- },
254
- ])
255
- })
256
-
257
- test("should parse leader with modifier", () => {
258
- const result = Keybind.parse("<leader>h")
259
- expect(result).toEqual([
260
- {
261
- ctrl: false,
262
- meta: false,
263
- shift: false,
264
- leader: true,
265
- name: "h",
266
- },
267
- ])
268
- })
269
-
270
- test("should parse multiple keybinds separated by comma", () => {
271
- const result = Keybind.parse("ctrl+c,<leader>q")
272
- expect(result).toEqual([
273
- {
274
- ctrl: true,
275
- meta: false,
276
- shift: false,
277
- leader: false,
278
- name: "c",
279
- },
280
- {
281
- ctrl: false,
282
- meta: false,
283
- shift: false,
284
- leader: true,
285
- name: "q",
286
- },
287
- ])
288
- })
289
-
290
- test("should parse shift+return combination", () => {
291
- const result = Keybind.parse("shift+return")
292
- expect(result).toEqual([
293
- {
294
- ctrl: false,
295
- meta: false,
296
- shift: true,
297
- leader: false,
298
- name: "return",
299
- },
300
- ])
301
- })
302
-
303
- test("should parse ctrl+j combination", () => {
304
- const result = Keybind.parse("ctrl+j")
305
- expect(result).toEqual([
306
- {
307
- ctrl: true,
308
- meta: false,
309
- shift: false,
310
- leader: false,
311
- name: "j",
312
- },
313
- ])
314
- })
315
-
316
- test("should handle 'none' value", () => {
317
- const result = Keybind.parse("none")
318
- expect(result).toEqual([])
319
- })
320
-
321
- test("should handle special keys", () => {
322
- const result = Keybind.parse("pgup")
323
- expect(result).toEqual([
324
- {
325
- ctrl: false,
326
- meta: false,
327
- shift: false,
328
- leader: false,
329
- name: "pgup",
330
- },
331
- ])
332
- })
333
-
334
- test("should handle function keys", () => {
335
- const result = Keybind.parse("f2")
336
- expect(result).toEqual([
337
- {
338
- ctrl: false,
339
- meta: false,
340
- shift: false,
341
- leader: false,
342
- name: "f2",
343
- },
344
- ])
345
- })
346
-
347
- test("should handle complex multi-modifier combination", () => {
348
- const result = Keybind.parse("ctrl+alt+g")
349
- expect(result).toEqual([
350
- {
351
- ctrl: true,
352
- meta: true,
353
- shift: false,
354
- leader: false,
355
- name: "g",
356
- },
357
- ])
358
- })
359
-
360
- test("should be case insensitive", () => {
361
- const result = Keybind.parse("CTRL+X")
362
- expect(result).toEqual([
363
- {
364
- ctrl: true,
365
- meta: false,
366
- shift: false,
367
- leader: false,
368
- name: "x",
369
- },
370
- ])
371
- })
372
-
373
- test("should parse super modifier", () => {
374
- const result = Keybind.parse("super+z")
375
- expect(result).toEqual([
376
- {
377
- ctrl: false,
378
- meta: false,
379
- shift: false,
380
- super: true,
381
- leader: false,
382
- name: "z",
383
- },
384
- ])
385
- })
386
-
387
- test("should parse super with shift modifier", () => {
388
- const result = Keybind.parse("super+shift+z")
389
- expect(result).toEqual([
390
- {
391
- ctrl: false,
392
- meta: false,
393
- shift: true,
394
- super: true,
395
- leader: false,
396
- name: "z",
397
- },
398
- ])
399
- })
400
-
401
- test("should parse multiple keybinds with super", () => {
402
- const result = Keybind.parse("ctrl+-,super+z")
403
- expect(result).toEqual([
404
- {
405
- ctrl: true,
406
- meta: false,
407
- shift: false,
408
- leader: false,
409
- name: "-",
410
- },
411
- {
412
- ctrl: false,
413
- meta: false,
414
- shift: false,
415
- super: true,
416
- leader: false,
417
- name: "z",
418
- },
419
- ])
420
- })
421
- })
@@ -1,95 +0,0 @@
1
- import { describe, expect, test, beforeEach } from "bun:test"
2
- import path from "path"
3
- import { LSPClient } from "../../src/lsp/client"
4
- import { LSPServer } from "../../src/lsp/server"
5
- import { Instance } from "../../src/project/instance"
6
- import { Log } from "../../src/util/log"
7
-
8
- // Minimal fake LSP server that speaks JSON-RPC over stdio
9
- function spawnFakeServer() {
10
- const { spawn } = require("child_process")
11
- const serverPath = path.join(__dirname, "../fixture/lsp/fake-lsp-server.js")
12
- return {
13
- process: spawn(process.execPath, [serverPath], {
14
- stdio: "pipe",
15
- }),
16
- }
17
- }
18
-
19
- describe("LSPClient interop", () => {
20
- beforeEach(async () => {
21
- await Log.init({ print: true })
22
- })
23
-
24
- test("handles workspace/workspaceFolders request", async () => {
25
- const handle = spawnFakeServer() as any
26
-
27
- const client = await Instance.provide({
28
- directory: process.cwd(),
29
- fn: () =>
30
- LSPClient.create({
31
- serverID: "fake",
32
- server: handle as unknown as LSPServer.Handle,
33
- root: process.cwd(),
34
- }),
35
- })
36
-
37
- await client.connection.sendNotification("test/trigger", {
38
- method: "workspace/workspaceFolders",
39
- })
40
-
41
- await new Promise((r) => setTimeout(r, 100))
42
-
43
- expect(client.connection).toBeDefined()
44
-
45
- await client.shutdown()
46
- })
47
-
48
- test("handles client/registerCapability request", async () => {
49
- const handle = spawnFakeServer() as any
50
-
51
- const client = await Instance.provide({
52
- directory: process.cwd(),
53
- fn: () =>
54
- LSPClient.create({
55
- serverID: "fake",
56
- server: handle as unknown as LSPServer.Handle,
57
- root: process.cwd(),
58
- }),
59
- })
60
-
61
- await client.connection.sendNotification("test/trigger", {
62
- method: "client/registerCapability",
63
- })
64
-
65
- await new Promise((r) => setTimeout(r, 100))
66
-
67
- expect(client.connection).toBeDefined()
68
-
69
- await client.shutdown()
70
- })
71
-
72
- test("handles client/unregisterCapability request", async () => {
73
- const handle = spawnFakeServer() as any
74
-
75
- const client = await Instance.provide({
76
- directory: process.cwd(),
77
- fn: () =>
78
- LSPClient.create({
79
- serverID: "fake",
80
- server: handle as unknown as LSPServer.Handle,
81
- root: process.cwd(),
82
- }),
83
- })
84
-
85
- await client.connection.sendNotification("test/trigger", {
86
- method: "client/unregisterCapability",
87
- })
88
-
89
- await new Promise((r) => setTimeout(r, 100))
90
-
91
- expect(client.connection).toBeDefined()
92
-
93
- await client.shutdown()
94
- })
95
- })
@@ -1,153 +0,0 @@
1
- import { test, expect, mock, beforeEach } from "bun:test"
2
-
3
- // Track what options were passed to each transport constructor
4
- const transportCalls: Array<{
5
- type: "streamable" | "sse"
6
- url: string
7
- options: { authProvider?: unknown; requestInit?: RequestInit }
8
- }> = []
9
-
10
- // Mock the transport constructors to capture their arguments
11
- mock.module("@modelcontextprotocol/sdk/client/streamableHttp.js", () => ({
12
- StreamableHTTPClientTransport: class MockStreamableHTTP {
13
- constructor(url: URL, options?: { authProvider?: unknown; requestInit?: RequestInit }) {
14
- transportCalls.push({
15
- type: "streamable",
16
- url: url.toString(),
17
- options: options ?? {},
18
- })
19
- }
20
- async start() {
21
- throw new Error("Mock transport cannot connect")
22
- }
23
- },
24
- }))
25
-
26
- mock.module("@modelcontextprotocol/sdk/client/sse.js", () => ({
27
- SSEClientTransport: class MockSSE {
28
- constructor(url: URL, options?: { authProvider?: unknown; requestInit?: RequestInit }) {
29
- transportCalls.push({
30
- type: "sse",
31
- url: url.toString(),
32
- options: options ?? {},
33
- })
34
- }
35
- async start() {
36
- throw new Error("Mock transport cannot connect")
37
- }
38
- },
39
- }))
40
-
41
- beforeEach(() => {
42
- transportCalls.length = 0
43
- })
44
-
45
- // Import MCP after mocking
46
- const { MCP } = await import("../../src/mcp/index")
47
- const { Instance } = await import("../../src/project/instance")
48
- const { tmpdir } = await import("../fixture/fixture")
49
-
50
- test("headers are passed to transports when oauth is enabled (default)", async () => {
51
- await using tmp = await tmpdir({
52
- init: async (dir) => {
53
- await Bun.write(
54
- `${dir}/opencode.json`,
55
- JSON.stringify({
56
- $schema: "https://opencode.ai/config.json",
57
- mcp: {
58
- "test-server": {
59
- type: "remote",
60
- url: "https://example.com/mcp",
61
- headers: {
62
- Authorization: "Bearer test-token",
63
- "X-Custom-Header": "custom-value",
64
- },
65
- },
66
- },
67
- }),
68
- )
69
- },
70
- })
71
-
72
- await Instance.provide({
73
- directory: tmp.path,
74
- fn: async () => {
75
- // Trigger MCP initialization - it will fail to connect but we can check the transport options
76
- await MCP.add("test-server", {
77
- type: "remote",
78
- url: "https://example.com/mcp",
79
- headers: {
80
- Authorization: "Bearer test-token",
81
- "X-Custom-Header": "custom-value",
82
- },
83
- }).catch(() => {})
84
-
85
- // Both transports should have been created with headers
86
- expect(transportCalls.length).toBeGreaterThanOrEqual(1)
87
-
88
- for (const call of transportCalls) {
89
- expect(call.options.requestInit).toBeDefined()
90
- expect(call.options.requestInit?.headers).toEqual({
91
- Authorization: "Bearer test-token",
92
- "X-Custom-Header": "custom-value",
93
- })
94
- // OAuth should be enabled by default, so authProvider should exist
95
- expect(call.options.authProvider).toBeDefined()
96
- }
97
- },
98
- })
99
- })
100
-
101
- test("headers are passed to transports when oauth is explicitly disabled", async () => {
102
- await using tmp = await tmpdir()
103
-
104
- await Instance.provide({
105
- directory: tmp.path,
106
- fn: async () => {
107
- transportCalls.length = 0
108
-
109
- await MCP.add("test-server-no-oauth", {
110
- type: "remote",
111
- url: "https://example.com/mcp",
112
- oauth: false,
113
- headers: {
114
- Authorization: "Bearer test-token",
115
- },
116
- }).catch(() => {})
117
-
118
- expect(transportCalls.length).toBeGreaterThanOrEqual(1)
119
-
120
- for (const call of transportCalls) {
121
- expect(call.options.requestInit).toBeDefined()
122
- expect(call.options.requestInit?.headers).toEqual({
123
- Authorization: "Bearer test-token",
124
- })
125
- // OAuth is disabled, so no authProvider
126
- expect(call.options.authProvider).toBeUndefined()
127
- }
128
- },
129
- })
130
- })
131
-
132
- test("no requestInit when headers are not provided", async () => {
133
- await using tmp = await tmpdir()
134
-
135
- await Instance.provide({
136
- directory: tmp.path,
137
- fn: async () => {
138
- transportCalls.length = 0
139
-
140
- await MCP.add("test-server-no-headers", {
141
- type: "remote",
142
- url: "https://example.com/mcp",
143
- }).catch(() => {})
144
-
145
- expect(transportCalls.length).toBeGreaterThanOrEqual(1)
146
-
147
- for (const call of transportCalls) {
148
- // No headers means requestInit should be undefined
149
- expect(call.options.requestInit).toBeUndefined()
150
- }
151
- },
152
- })
153
- })