mastracode 0.0.1

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 (336) hide show
  1. package/AGENTS.md +27 -0
  2. package/Dockerfile +18 -0
  3. package/README.md +15 -0
  4. package/bin/opencode +84 -0
  5. package/bunfig.toml +4 -0
  6. package/package.json +113 -0
  7. package/parsers-config.ts +239 -0
  8. package/script/build.ts +167 -0
  9. package/script/postinstall.mjs +122 -0
  10. package/script/publish-registries.ts +187 -0
  11. package/script/publish.ts +70 -0
  12. package/script/schema.ts +47 -0
  13. package/src/acp/README.md +164 -0
  14. package/src/acp/agent.ts +1051 -0
  15. package/src/acp/session.ts +101 -0
  16. package/src/acp/types.ts +22 -0
  17. package/src/agent/agent.ts +398 -0
  18. package/src/agent/generate.txt +75 -0
  19. package/src/agent/prompt/compaction.txt +12 -0
  20. package/src/agent/prompt/explore.txt +18 -0
  21. package/src/agent/prompt/summary.txt +10 -0
  22. package/src/agent/prompt/title.txt +36 -0
  23. package/src/auth/index.ts +70 -0
  24. package/src/bun/index.ts +114 -0
  25. package/src/bus/bus-event.ts +43 -0
  26. package/src/bus/global.ts +10 -0
  27. package/src/bus/index.ts +105 -0
  28. package/src/cli/bootstrap.ts +17 -0
  29. package/src/cli/cmd/acp.ts +88 -0
  30. package/src/cli/cmd/agent.ts +256 -0
  31. package/src/cli/cmd/auth.ts +391 -0
  32. package/src/cli/cmd/cmd.ts +7 -0
  33. package/src/cli/cmd/debug/config.ts +15 -0
  34. package/src/cli/cmd/debug/file.ts +91 -0
  35. package/src/cli/cmd/debug/index.ts +43 -0
  36. package/src/cli/cmd/debug/lsp.ts +48 -0
  37. package/src/cli/cmd/debug/ripgrep.ts +83 -0
  38. package/src/cli/cmd/debug/scrap.ts +15 -0
  39. package/src/cli/cmd/debug/skill.ts +15 -0
  40. package/src/cli/cmd/debug/snapshot.ts +48 -0
  41. package/src/cli/cmd/export.ts +88 -0
  42. package/src/cli/cmd/generate.ts +38 -0
  43. package/src/cli/cmd/github.ts +1408 -0
  44. package/src/cli/cmd/import.ts +98 -0
  45. package/src/cli/cmd/mcp.ts +654 -0
  46. package/src/cli/cmd/models.ts +77 -0
  47. package/src/cli/cmd/pr.ts +112 -0
  48. package/src/cli/cmd/run.ts +368 -0
  49. package/src/cli/cmd/serve.ts +31 -0
  50. package/src/cli/cmd/session.ts +106 -0
  51. package/src/cli/cmd/stats.ts +298 -0
  52. package/src/cli/cmd/tui/app.tsx +686 -0
  53. package/src/cli/cmd/tui/attach.ts +30 -0
  54. package/src/cli/cmd/tui/component/border.tsx +21 -0
  55. package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
  56. package/src/cli/cmd/tui/component/dialog-command.tsx +124 -0
  57. package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
  58. package/src/cli/cmd/tui/component/dialog-model.tsx +230 -0
  59. package/src/cli/cmd/tui/component/dialog-provider.tsx +224 -0
  60. package/src/cli/cmd/tui/component/dialog-session-list.tsx +102 -0
  61. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
  62. package/src/cli/cmd/tui/component/dialog-stash.tsx +86 -0
  63. package/src/cli/cmd/tui/component/dialog-status.tsx +162 -0
  64. package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
  65. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
  66. package/src/cli/cmd/tui/component/did-you-know.tsx +85 -0
  67. package/src/cli/cmd/tui/component/logo.tsx +27 -0
  68. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +574 -0
  69. package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
  70. package/src/cli/cmd/tui/component/prompt/index.tsx +1117 -0
  71. package/src/cli/cmd/tui/component/prompt/stash.tsx +101 -0
  72. package/src/cli/cmd/tui/component/tips.ts +103 -0
  73. package/src/cli/cmd/tui/component/todo-item.tsx +32 -0
  74. package/src/cli/cmd/tui/context/args.tsx +14 -0
  75. package/src/cli/cmd/tui/context/directory.ts +13 -0
  76. package/src/cli/cmd/tui/context/exit.tsx +23 -0
  77. package/src/cli/cmd/tui/context/helper.tsx +25 -0
  78. package/src/cli/cmd/tui/context/keybind.tsx +101 -0
  79. package/src/cli/cmd/tui/context/kv.tsx +49 -0
  80. package/src/cli/cmd/tui/context/local.tsx +339 -0
  81. package/src/cli/cmd/tui/context/prompt.tsx +18 -0
  82. package/src/cli/cmd/tui/context/route.tsx +46 -0
  83. package/src/cli/cmd/tui/context/sdk.tsx +74 -0
  84. package/src/cli/cmd/tui/context/sync.tsx +372 -0
  85. package/src/cli/cmd/tui/context/theme/aura.json +69 -0
  86. package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
  87. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +233 -0
  88. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
  89. package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
  90. package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
  91. package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
  92. package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
  93. package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
  94. package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
  95. package/src/cli/cmd/tui/context/theme/github.json +233 -0
  96. package/src/cli/cmd/tui/context/theme/gruvbox.json +95 -0
  97. package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
  98. package/src/cli/cmd/tui/context/theme/lucent-orng.json +227 -0
  99. package/src/cli/cmd/tui/context/theme/material.json +235 -0
  100. package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
  101. package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
  102. package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
  103. package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
  104. package/src/cli/cmd/tui/context/theme/nord.json +223 -0
  105. package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
  106. package/src/cli/cmd/tui/context/theme/orng.json +245 -0
  107. package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
  108. package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
  109. package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
  110. package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
  111. package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
  112. package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
  113. package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
  114. package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
  115. package/src/cli/cmd/tui/context/theme.tsx +1109 -0
  116. package/src/cli/cmd/tui/event.ts +40 -0
  117. package/src/cli/cmd/tui/routes/home.tsx +140 -0
  118. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +64 -0
  119. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +109 -0
  120. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +26 -0
  121. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
  122. package/src/cli/cmd/tui/routes/session/footer.tsx +88 -0
  123. package/src/cli/cmd/tui/routes/session/header.tsx +141 -0
  124. package/src/cli/cmd/tui/routes/session/index.tsx +1885 -0
  125. package/src/cli/cmd/tui/routes/session/sidebar.tsx +322 -0
  126. package/src/cli/cmd/tui/spawn.ts +60 -0
  127. package/src/cli/cmd/tui/thread.ts +120 -0
  128. package/src/cli/cmd/tui/ui/dialog-alert.tsx +57 -0
  129. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +83 -0
  130. package/src/cli/cmd/tui/ui/dialog-help.tsx +38 -0
  131. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +77 -0
  132. package/src/cli/cmd/tui/ui/dialog-select.tsx +332 -0
  133. package/src/cli/cmd/tui/ui/dialog.tsx +170 -0
  134. package/src/cli/cmd/tui/ui/spinner.ts +368 -0
  135. package/src/cli/cmd/tui/ui/toast.tsx +100 -0
  136. package/src/cli/cmd/tui/util/clipboard.ts +127 -0
  137. package/src/cli/cmd/tui/util/editor.ts +32 -0
  138. package/src/cli/cmd/tui/util/terminal.ts +114 -0
  139. package/src/cli/cmd/tui/worker.ts +63 -0
  140. package/src/cli/cmd/uninstall.ts +344 -0
  141. package/src/cli/cmd/upgrade.ts +67 -0
  142. package/src/cli/cmd/web.ts +84 -0
  143. package/src/cli/error.ts +56 -0
  144. package/src/cli/ui.ts +84 -0
  145. package/src/cli/upgrade.ts +25 -0
  146. package/src/command/index.ts +80 -0
  147. package/src/command/template/initialize.txt +10 -0
  148. package/src/command/template/review.txt +97 -0
  149. package/src/config/config.ts +997 -0
  150. package/src/config/markdown.ts +41 -0
  151. package/src/env/index.ts +26 -0
  152. package/src/file/ignore.ts +83 -0
  153. package/src/file/index.ts +328 -0
  154. package/src/file/ripgrep.ts +393 -0
  155. package/src/file/time.ts +64 -0
  156. package/src/file/watcher.ts +103 -0
  157. package/src/flag/flag.ts +46 -0
  158. package/src/format/formatter.ts +315 -0
  159. package/src/format/index.ts +137 -0
  160. package/src/global/index.ts +52 -0
  161. package/src/id/id.ts +73 -0
  162. package/src/ide/index.ts +76 -0
  163. package/src/index.ts +158 -0
  164. package/src/installation/index.ts +196 -0
  165. package/src/lsp/client.ts +229 -0
  166. package/src/lsp/index.ts +485 -0
  167. package/src/lsp/language.ts +116 -0
  168. package/src/lsp/server.ts +1895 -0
  169. package/src/mcp/auth.ts +135 -0
  170. package/src/mcp/index.ts +654 -0
  171. package/src/mcp/oauth-callback.ts +200 -0
  172. package/src/mcp/oauth-provider.ts +154 -0
  173. package/src/patch/index.ts +622 -0
  174. package/src/permission/index.ts +199 -0
  175. package/src/plugin/index.ts +91 -0
  176. package/src/project/bootstrap.ts +31 -0
  177. package/src/project/instance.ts +78 -0
  178. package/src/project/project.ts +221 -0
  179. package/src/project/state.ts +65 -0
  180. package/src/project/vcs.ts +76 -0
  181. package/src/provider/auth.ts +143 -0
  182. package/src/provider/models-macro.ts +11 -0
  183. package/src/provider/models.ts +106 -0
  184. package/src/provider/provider.ts +1056 -0
  185. package/src/provider/sdk/openai-compatible/src/README.md +5 -0
  186. package/src/provider/sdk/openai-compatible/src/index.ts +2 -0
  187. package/src/provider/sdk/openai-compatible/src/openai-compatible-provider.ts +100 -0
  188. package/src/provider/sdk/openai-compatible/src/responses/convert-to-openai-responses-input.ts +303 -0
  189. package/src/provider/sdk/openai-compatible/src/responses/map-openai-responses-finish-reason.ts +22 -0
  190. package/src/provider/sdk/openai-compatible/src/responses/openai-config.ts +18 -0
  191. package/src/provider/sdk/openai-compatible/src/responses/openai-error.ts +22 -0
  192. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-api-types.ts +207 -0
  193. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-language-model.ts +1713 -0
  194. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-prepare-tools.ts +177 -0
  195. package/src/provider/sdk/openai-compatible/src/responses/openai-responses-settings.ts +1 -0
  196. package/src/provider/sdk/openai-compatible/src/responses/tool/code-interpreter.ts +88 -0
  197. package/src/provider/sdk/openai-compatible/src/responses/tool/file-search.ts +128 -0
  198. package/src/provider/sdk/openai-compatible/src/responses/tool/image-generation.ts +115 -0
  199. package/src/provider/sdk/openai-compatible/src/responses/tool/local-shell.ts +65 -0
  200. package/src/provider/sdk/openai-compatible/src/responses/tool/web-search-preview.ts +104 -0
  201. package/src/provider/sdk/openai-compatible/src/responses/tool/web-search.ts +103 -0
  202. package/src/provider/transform.ts +455 -0
  203. package/src/pty/index.ts +231 -0
  204. package/src/server/error.ts +36 -0
  205. package/src/server/project.ts +79 -0
  206. package/src/server/server.ts +2642 -0
  207. package/src/server/tui.ts +71 -0
  208. package/src/session/compaction.ts +223 -0
  209. package/src/session/index.ts +458 -0
  210. package/src/session/llm-mastra.ts +412 -0
  211. package/src/session/llm-shared.ts +172 -0
  212. package/src/session/llm.ts +439 -0
  213. package/src/session/message-v2.ts +675 -0
  214. package/src/session/message.ts +189 -0
  215. package/src/session/processor.ts +171 -0
  216. package/src/session/prompt/anthropic-20250930.txt +166 -0
  217. package/src/session/prompt/anthropic.txt +105 -0
  218. package/src/session/prompt/anthropic_spoof.txt +1 -0
  219. package/src/session/prompt/beast.txt +147 -0
  220. package/src/session/prompt/build-switch.txt +5 -0
  221. package/src/session/prompt/codex.txt +318 -0
  222. package/src/session/prompt/copilot-gpt-5.txt +143 -0
  223. package/src/session/prompt/gemini.txt +155 -0
  224. package/src/session/prompt/max-steps.txt +16 -0
  225. package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
  226. package/src/session/prompt/plan.txt +26 -0
  227. package/src/session/prompt/polaris.txt +107 -0
  228. package/src/session/prompt/qwen.txt +109 -0
  229. package/src/session/prompt.ts +1454 -0
  230. package/src/session/retry.ts +86 -0
  231. package/src/session/revert.ts +108 -0
  232. package/src/session/status.ts +76 -0
  233. package/src/session/summary.ts +194 -0
  234. package/src/session/system.ts +120 -0
  235. package/src/session/todo.ts +37 -0
  236. package/src/share/share-next.ts +194 -0
  237. package/src/share/share.ts +87 -0
  238. package/src/shell/shell.ts +67 -0
  239. package/src/skill/index.ts +1 -0
  240. package/src/skill/skill.ts +83 -0
  241. package/src/snapshot/index.ts +197 -0
  242. package/src/storage/storage.ts +226 -0
  243. package/src/tool/bash.ts +306 -0
  244. package/src/tool/bash.txt +158 -0
  245. package/src/tool/batch.ts +175 -0
  246. package/src/tool/batch.txt +24 -0
  247. package/src/tool/codesearch.ts +138 -0
  248. package/src/tool/codesearch.txt +12 -0
  249. package/src/tool/edit.ts +675 -0
  250. package/src/tool/edit.txt +10 -0
  251. package/src/tool/glob.ts +65 -0
  252. package/src/tool/glob.txt +6 -0
  253. package/src/tool/grep.ts +121 -0
  254. package/src/tool/grep.txt +8 -0
  255. package/src/tool/invalid.ts +17 -0
  256. package/src/tool/ls.ts +110 -0
  257. package/src/tool/ls.txt +1 -0
  258. package/src/tool/lsp-diagnostics.ts +26 -0
  259. package/src/tool/lsp-diagnostics.txt +1 -0
  260. package/src/tool/lsp-hover.ts +31 -0
  261. package/src/tool/lsp-hover.txt +1 -0
  262. package/src/tool/lsp.ts +87 -0
  263. package/src/tool/lsp.txt +19 -0
  264. package/src/tool/multiedit.ts +46 -0
  265. package/src/tool/multiedit.txt +41 -0
  266. package/src/tool/patch.ts +233 -0
  267. package/src/tool/patch.txt +1 -0
  268. package/src/tool/read.ts +219 -0
  269. package/src/tool/read.txt +12 -0
  270. package/src/tool/registry.ts +162 -0
  271. package/src/tool/skill.ts +100 -0
  272. package/src/tool/task.ts +136 -0
  273. package/src/tool/task.txt +60 -0
  274. package/src/tool/todo.ts +39 -0
  275. package/src/tool/todoread.txt +14 -0
  276. package/src/tool/todowrite.txt +167 -0
  277. package/src/tool/tool.ts +71 -0
  278. package/src/tool/webfetch.ts +187 -0
  279. package/src/tool/webfetch.txt +13 -0
  280. package/src/tool/websearch.ts +150 -0
  281. package/src/tool/websearch.txt +11 -0
  282. package/src/tool/write.ts +110 -0
  283. package/src/tool/write.txt +8 -0
  284. package/src/util/archive.ts +16 -0
  285. package/src/util/color.ts +19 -0
  286. package/src/util/context.ts +25 -0
  287. package/src/util/defer.ts +12 -0
  288. package/src/util/eventloop.ts +20 -0
  289. package/src/util/filesystem.ts +83 -0
  290. package/src/util/fn.ts +11 -0
  291. package/src/util/iife.ts +3 -0
  292. package/src/util/keybind.ts +102 -0
  293. package/src/util/lazy.ts +11 -0
  294. package/src/util/locale.ts +81 -0
  295. package/src/util/lock.ts +98 -0
  296. package/src/util/log.ts +180 -0
  297. package/src/util/queue.ts +32 -0
  298. package/src/util/rpc.ts +42 -0
  299. package/src/util/scrap.ts +10 -0
  300. package/src/util/signal.ts +12 -0
  301. package/src/util/timeout.ts +14 -0
  302. package/src/util/token.ts +7 -0
  303. package/src/util/wildcard.ts +54 -0
  304. package/sst-env.d.ts +9 -0
  305. package/test/agent/agent.test.ts +146 -0
  306. package/test/bun.test.ts +53 -0
  307. package/test/cli/github-remote.test.ts +80 -0
  308. package/test/config/agent-color.test.ts +66 -0
  309. package/test/config/config.test.ts +535 -0
  310. package/test/config/markdown.test.ts +89 -0
  311. package/test/file/ignore.test.ts +10 -0
  312. package/test/fixture/fixture.ts +34 -0
  313. package/test/fixture/lsp/fake-lsp-server.js +77 -0
  314. package/test/ide/ide.test.ts +82 -0
  315. package/test/keybind.test.ts +421 -0
  316. package/test/lsp/client.test.ts +95 -0
  317. package/test/mcp/headers.test.ts +153 -0
  318. package/test/patch/patch.test.ts +348 -0
  319. package/test/preload.ts +57 -0
  320. package/test/project/project.test.ts +72 -0
  321. package/test/provider/provider.test.ts +1809 -0
  322. package/test/provider/transform.test.ts +411 -0
  323. package/test/session/retry.test.ts +61 -0
  324. package/test/session/session.test.ts +71 -0
  325. package/test/skill/skill.test.ts +131 -0
  326. package/test/snapshot/snapshot.test.ts +939 -0
  327. package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
  328. package/test/tool/bash.test.ts +434 -0
  329. package/test/tool/grep.test.ts +108 -0
  330. package/test/tool/patch.test.ts +259 -0
  331. package/test/tool/read.test.ts +42 -0
  332. package/test/util/iife.test.ts +36 -0
  333. package/test/util/lazy.test.ts +50 -0
  334. package/test/util/timeout.test.ts +21 -0
  335. package/test/util/wildcard.test.ts +55 -0
  336. package/tsconfig.json +16 -0
@@ -0,0 +1,77 @@
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
+ }
@@ -0,0 +1,82 @@
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
+ })
@@ -0,0 +1,421 @@
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
+ })
@@ -0,0 +1,95 @@
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
+ })