chad-code 1.3.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 (338) hide show
  1. package/AGENTS.md +27 -0
  2. package/Dockerfile +18 -0
  3. package/README.md +15 -0
  4. package/README.npm.md +64 -0
  5. package/bin/chad-code +84 -0
  6. package/bunfig.toml +7 -0
  7. package/eslint.config.js +29 -0
  8. package/package.json +107 -0
  9. package/parsers-config.ts +253 -0
  10. package/script/build.ts +167 -0
  11. package/script/postinstall.mjs +122 -0
  12. package/script/publish-registries.ts +187 -0
  13. package/script/publish.ts +93 -0
  14. package/script/schema.ts +47 -0
  15. package/src/acp/README.md +164 -0
  16. package/src/acp/agent.ts +1086 -0
  17. package/src/acp/session.ts +101 -0
  18. package/src/acp/types.ts +22 -0
  19. package/src/agent/agent.ts +253 -0
  20. package/src/agent/generate.txt +75 -0
  21. package/src/agent/prompt/compaction.txt +12 -0
  22. package/src/agent/prompt/explore.txt +18 -0
  23. package/src/agent/prompt/summary.txt +11 -0
  24. package/src/agent/prompt/title.txt +36 -0
  25. package/src/auth/index.ts +70 -0
  26. package/src/bun/index.ts +130 -0
  27. package/src/bus/bus-event.ts +43 -0
  28. package/src/bus/global.ts +10 -0
  29. package/src/bus/index.ts +105 -0
  30. package/src/cli/bootstrap.ts +17 -0
  31. package/src/cli/cmd/acp.ts +69 -0
  32. package/src/cli/cmd/agent.ts +257 -0
  33. package/src/cli/cmd/auth.ts +132 -0
  34. package/src/cli/cmd/cmd.ts +7 -0
  35. package/src/cli/cmd/debug/agent.ts +28 -0
  36. package/src/cli/cmd/debug/config.ts +15 -0
  37. package/src/cli/cmd/debug/file.ts +91 -0
  38. package/src/cli/cmd/debug/index.ts +45 -0
  39. package/src/cli/cmd/debug/lsp.ts +48 -0
  40. package/src/cli/cmd/debug/ripgrep.ts +83 -0
  41. package/src/cli/cmd/debug/scrap.ts +15 -0
  42. package/src/cli/cmd/debug/skill.ts +15 -0
  43. package/src/cli/cmd/debug/snapshot.ts +48 -0
  44. package/src/cli/cmd/export.ts +88 -0
  45. package/src/cli/cmd/generate.ts +38 -0
  46. package/src/cli/cmd/github.ts +32 -0
  47. package/src/cli/cmd/import.ts +98 -0
  48. package/src/cli/cmd/mcp.ts +670 -0
  49. package/src/cli/cmd/models.ts +42 -0
  50. package/src/cli/cmd/pr.ts +112 -0
  51. package/src/cli/cmd/run.ts +374 -0
  52. package/src/cli/cmd/serve.ts +16 -0
  53. package/src/cli/cmd/session.ts +135 -0
  54. package/src/cli/cmd/stats.ts +402 -0
  55. package/src/cli/cmd/tui/app.tsx +705 -0
  56. package/src/cli/cmd/tui/attach.ts +32 -0
  57. package/src/cli/cmd/tui/component/border.tsx +21 -0
  58. package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
  59. package/src/cli/cmd/tui/component/dialog-command.tsx +124 -0
  60. package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
  61. package/src/cli/cmd/tui/component/dialog-model.tsx +232 -0
  62. package/src/cli/cmd/tui/component/dialog-provider.tsx +228 -0
  63. package/src/cli/cmd/tui/component/dialog-session-list.tsx +115 -0
  64. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
  65. package/src/cli/cmd/tui/component/dialog-stash.tsx +86 -0
  66. package/src/cli/cmd/tui/component/dialog-status.tsx +162 -0
  67. package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
  68. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
  69. package/src/cli/cmd/tui/component/did-you-know.tsx +85 -0
  70. package/src/cli/cmd/tui/component/logo.tsx +43 -0
  71. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +654 -0
  72. package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
  73. package/src/cli/cmd/tui/component/prompt/index.tsx +1078 -0
  74. package/src/cli/cmd/tui/component/prompt/stash.tsx +101 -0
  75. package/src/cli/cmd/tui/component/textarea-keybindings.ts +73 -0
  76. package/src/cli/cmd/tui/component/tips.ts +92 -0
  77. package/src/cli/cmd/tui/component/todo-item.tsx +32 -0
  78. package/src/cli/cmd/tui/context/args.tsx +14 -0
  79. package/src/cli/cmd/tui/context/directory.ts +13 -0
  80. package/src/cli/cmd/tui/context/exit.tsx +23 -0
  81. package/src/cli/cmd/tui/context/helper.tsx +25 -0
  82. package/src/cli/cmd/tui/context/keybind.tsx +101 -0
  83. package/src/cli/cmd/tui/context/kv.tsx +49 -0
  84. package/src/cli/cmd/tui/context/local.tsx +392 -0
  85. package/src/cli/cmd/tui/context/prompt.tsx +18 -0
  86. package/src/cli/cmd/tui/context/route.tsx +46 -0
  87. package/src/cli/cmd/tui/context/sdk.tsx +75 -0
  88. package/src/cli/cmd/tui/context/sync.tsx +384 -0
  89. package/src/cli/cmd/tui/context/theme/aura.json +69 -0
  90. package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
  91. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +233 -0
  92. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
  93. package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
  94. package/src/cli/cmd/tui/context/theme/chad.json +245 -0
  95. package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
  96. package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
  97. package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
  98. package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
  99. package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
  100. package/src/cli/cmd/tui/context/theme/github.json +233 -0
  101. package/src/cli/cmd/tui/context/theme/gruvbox.json +95 -0
  102. package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
  103. package/src/cli/cmd/tui/context/theme/lucent-orng.json +227 -0
  104. package/src/cli/cmd/tui/context/theme/material.json +235 -0
  105. package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
  106. package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
  107. package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
  108. package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
  109. package/src/cli/cmd/tui/context/theme/nord.json +223 -0
  110. package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
  111. package/src/cli/cmd/tui/context/theme/orng.json +245 -0
  112. package/src/cli/cmd/tui/context/theme/osaka-jade.json +93 -0
  113. package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
  114. package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
  115. package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
  116. package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
  117. package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
  118. package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
  119. package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
  120. package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
  121. package/src/cli/cmd/tui/context/theme.tsx +1137 -0
  122. package/src/cli/cmd/tui/event.ts +46 -0
  123. package/src/cli/cmd/tui/routes/home.tsx +138 -0
  124. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +64 -0
  125. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +109 -0
  126. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +26 -0
  127. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
  128. package/src/cli/cmd/tui/routes/session/footer.tsx +88 -0
  129. package/src/cli/cmd/tui/routes/session/header.tsx +125 -0
  130. package/src/cli/cmd/tui/routes/session/index.tsx +1814 -0
  131. package/src/cli/cmd/tui/routes/session/permission.tsx +416 -0
  132. package/src/cli/cmd/tui/routes/session/sidebar.tsx +318 -0
  133. package/src/cli/cmd/tui/spawn.ts +48 -0
  134. package/src/cli/cmd/tui/thread.ts +111 -0
  135. package/src/cli/cmd/tui/ui/dialog-alert.tsx +57 -0
  136. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +83 -0
  137. package/src/cli/cmd/tui/ui/dialog-export-options.tsx +204 -0
  138. package/src/cli/cmd/tui/ui/dialog-help.tsx +38 -0
  139. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +77 -0
  140. package/src/cli/cmd/tui/ui/dialog-select.tsx +345 -0
  141. package/src/cli/cmd/tui/ui/dialog.tsx +171 -0
  142. package/src/cli/cmd/tui/ui/link.tsx +28 -0
  143. package/src/cli/cmd/tui/ui/spinner.ts +368 -0
  144. package/src/cli/cmd/tui/ui/toast.tsx +100 -0
  145. package/src/cli/cmd/tui/util/clipboard.ts +127 -0
  146. package/src/cli/cmd/tui/util/editor.ts +32 -0
  147. package/src/cli/cmd/tui/util/signal.ts +7 -0
  148. package/src/cli/cmd/tui/util/terminal.ts +114 -0
  149. package/src/cli/cmd/tui/util/transcript.ts +98 -0
  150. package/src/cli/cmd/tui/worker.ts +68 -0
  151. package/src/cli/cmd/uninstall.ts +344 -0
  152. package/src/cli/cmd/upgrade.ts +67 -0
  153. package/src/cli/cmd/web.ts +73 -0
  154. package/src/cli/error.ts +56 -0
  155. package/src/cli/network.ts +53 -0
  156. package/src/cli/ui.ts +87 -0
  157. package/src/cli/upgrade.ts +25 -0
  158. package/src/command/index.ts +131 -0
  159. package/src/command/template/initialize.txt +10 -0
  160. package/src/command/template/review.txt +97 -0
  161. package/src/config/config.ts +1124 -0
  162. package/src/config/markdown.ts +41 -0
  163. package/src/env/index.ts +26 -0
  164. package/src/file/ignore.ts +83 -0
  165. package/src/file/index.ts +411 -0
  166. package/src/file/ripgrep.ts +402 -0
  167. package/src/file/time.ts +64 -0
  168. package/src/file/watcher.ts +117 -0
  169. package/src/flag/flag.ts +52 -0
  170. package/src/format/formatter.ts +359 -0
  171. package/src/format/index.ts +137 -0
  172. package/src/global/index.ts +55 -0
  173. package/src/id/id.ts +73 -0
  174. package/src/ide/index.ts +77 -0
  175. package/src/index.ts +159 -0
  176. package/src/installation/index.ts +198 -0
  177. package/src/lsp/client.ts +252 -0
  178. package/src/lsp/index.ts +485 -0
  179. package/src/lsp/language.ts +119 -0
  180. package/src/lsp/server.ts +2023 -0
  181. package/src/mcp/auth.ts +135 -0
  182. package/src/mcp/index.ts +874 -0
  183. package/src/mcp/oauth-callback.ts +200 -0
  184. package/src/mcp/oauth-provider.ts +154 -0
  185. package/src/patch/index.ts +622 -0
  186. package/src/permission/arity.ts +163 -0
  187. package/src/permission/index.ts +210 -0
  188. package/src/permission/next.ts +268 -0
  189. package/src/plugin/index.ts +106 -0
  190. package/src/project/bootstrap.ts +31 -0
  191. package/src/project/instance.ts +78 -0
  192. package/src/project/project.ts +263 -0
  193. package/src/project/state.ts +65 -0
  194. package/src/project/vcs.ts +76 -0
  195. package/src/provider/auth.ts +143 -0
  196. package/src/provider/models-macro.ts +4 -0
  197. package/src/provider/models.ts +77 -0
  198. package/src/provider/provider.ts +516 -0
  199. package/src/provider/transform.ts +114 -0
  200. package/src/pty/index.ts +212 -0
  201. package/src/server/error.ts +36 -0
  202. package/src/server/mdns.ts +57 -0
  203. package/src/server/project.ts +79 -0
  204. package/src/server/server.ts +2866 -0
  205. package/src/server/tui.ts +71 -0
  206. package/src/session/compaction.ts +225 -0
  207. package/src/session/index.ts +469 -0
  208. package/src/session/llm.ts +213 -0
  209. package/src/session/message-v2.ts +742 -0
  210. package/src/session/message.ts +189 -0
  211. package/src/session/processor.ts +402 -0
  212. package/src/session/prompt/anthropic-20250930.txt +166 -0
  213. package/src/session/prompt/anthropic.txt +105 -0
  214. package/src/session/prompt/anthropic_spoof.txt +1 -0
  215. package/src/session/prompt/beast.txt +147 -0
  216. package/src/session/prompt/build-switch.txt +5 -0
  217. package/src/session/prompt/codex.txt +318 -0
  218. package/src/session/prompt/copilot-gpt-5.txt +143 -0
  219. package/src/session/prompt/gemini.txt +155 -0
  220. package/src/session/prompt/max-steps.txt +16 -0
  221. package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
  222. package/src/session/prompt/plan.txt +26 -0
  223. package/src/session/prompt/qwen.txt +109 -0
  224. package/src/session/prompt.ts +1621 -0
  225. package/src/session/retry.ts +90 -0
  226. package/src/session/revert.ts +108 -0
  227. package/src/session/status.ts +76 -0
  228. package/src/session/summary.ts +194 -0
  229. package/src/session/system.ts +108 -0
  230. package/src/session/todo.ts +37 -0
  231. package/src/share/share-next.ts +194 -0
  232. package/src/share/share.ts +23 -0
  233. package/src/shell/shell.ts +67 -0
  234. package/src/skill/index.ts +1 -0
  235. package/src/skill/skill.ts +124 -0
  236. package/src/snapshot/index.ts +197 -0
  237. package/src/storage/storage.ts +226 -0
  238. package/src/tool/bash.ts +262 -0
  239. package/src/tool/bash.txt +116 -0
  240. package/src/tool/batch.ts +175 -0
  241. package/src/tool/batch.txt +24 -0
  242. package/src/tool/codesearch.ts +132 -0
  243. package/src/tool/codesearch.txt +12 -0
  244. package/src/tool/edit.ts +655 -0
  245. package/src/tool/edit.txt +10 -0
  246. package/src/tool/glob.ts +75 -0
  247. package/src/tool/glob.txt +6 -0
  248. package/src/tool/grep.ts +132 -0
  249. package/src/tool/grep.txt +8 -0
  250. package/src/tool/invalid.ts +17 -0
  251. package/src/tool/ls.ts +119 -0
  252. package/src/tool/ls.txt +1 -0
  253. package/src/tool/lsp.ts +94 -0
  254. package/src/tool/lsp.txt +19 -0
  255. package/src/tool/multiedit.ts +46 -0
  256. package/src/tool/multiedit.txt +41 -0
  257. package/src/tool/patch.ts +210 -0
  258. package/src/tool/patch.txt +1 -0
  259. package/src/tool/read.ts +191 -0
  260. package/src/tool/read.txt +12 -0
  261. package/src/tool/registry.ts +137 -0
  262. package/src/tool/skill.ts +77 -0
  263. package/src/tool/task.ts +167 -0
  264. package/src/tool/task.txt +60 -0
  265. package/src/tool/todo.ts +53 -0
  266. package/src/tool/todoread.txt +14 -0
  267. package/src/tool/todowrite.txt +167 -0
  268. package/src/tool/tool.ts +73 -0
  269. package/src/tool/webfetch.ts +182 -0
  270. package/src/tool/webfetch.txt +13 -0
  271. package/src/tool/websearch.ts +144 -0
  272. package/src/tool/websearch.txt +11 -0
  273. package/src/tool/write.ts +84 -0
  274. package/src/tool/write.txt +8 -0
  275. package/src/util/archive.ts +16 -0
  276. package/src/util/color.ts +19 -0
  277. package/src/util/context.ts +25 -0
  278. package/src/util/defer.ts +12 -0
  279. package/src/util/eventloop.ts +20 -0
  280. package/src/util/filesystem.ts +83 -0
  281. package/src/util/fn.ts +11 -0
  282. package/src/util/iife.ts +3 -0
  283. package/src/util/keybind.ts +102 -0
  284. package/src/util/lazy.ts +18 -0
  285. package/src/util/locale.ts +81 -0
  286. package/src/util/lock.ts +98 -0
  287. package/src/util/log.ts +180 -0
  288. package/src/util/queue.ts +32 -0
  289. package/src/util/rpc.ts +42 -0
  290. package/src/util/scrap.ts +10 -0
  291. package/src/util/signal.ts +12 -0
  292. package/src/util/timeout.ts +14 -0
  293. package/src/util/token.ts +7 -0
  294. package/src/util/wildcard.ts +54 -0
  295. package/src/worktree/index.ts +217 -0
  296. package/sst-env.d.ts +9 -0
  297. package/test/agent/agent.test.ts +448 -0
  298. package/test/bun.test.ts +53 -0
  299. package/test/cli/github-action.test.ts +129 -0
  300. package/test/cli/github-remote.test.ts +80 -0
  301. package/test/cli/tui/transcript.test.ts +297 -0
  302. package/test/config/agent-color.test.ts +66 -0
  303. package/test/config/config.test.ts +870 -0
  304. package/test/config/markdown.test.ts +89 -0
  305. package/test/file/ignore.test.ts +10 -0
  306. package/test/file/path-traversal.test.ts +115 -0
  307. package/test/fixture/fixture.ts +45 -0
  308. package/test/fixture/lsp/fake-lsp-server.js +77 -0
  309. package/test/ide/ide.test.ts +82 -0
  310. package/test/keybind.test.ts +421 -0
  311. package/test/lsp/client.test.ts +95 -0
  312. package/test/mcp/headers.test.ts +153 -0
  313. package/test/patch/patch.test.ts +348 -0
  314. package/test/permission/arity.test.ts +33 -0
  315. package/test/permission/next.test.ts +652 -0
  316. package/test/preload.ts +63 -0
  317. package/test/project/project.test.ts +120 -0
  318. package/test/provider/amazon-bedrock.test.ts +236 -0
  319. package/test/provider/provider.test.ts +2127 -0
  320. package/test/provider/transform.test.ts +980 -0
  321. package/test/server/session-select.test.ts +78 -0
  322. package/test/session/compaction.test.ts +251 -0
  323. package/test/session/message-v2.test.ts +570 -0
  324. package/test/session/retry.test.ts +131 -0
  325. package/test/session/revert-compact.test.ts +285 -0
  326. package/test/session/session.test.ts +71 -0
  327. package/test/skill/skill.test.ts +185 -0
  328. package/test/snapshot/snapshot.test.ts +939 -0
  329. package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
  330. package/test/tool/bash.test.ts +232 -0
  331. package/test/tool/grep.test.ts +109 -0
  332. package/test/tool/patch.test.ts +261 -0
  333. package/test/tool/read.test.ts +167 -0
  334. package/test/util/iife.test.ts +36 -0
  335. package/test/util/lazy.test.ts +50 -0
  336. package/test/util/timeout.test.ts +21 -0
  337. package/test/util/wildcard.test.ts +55 -0
  338. package/tsconfig.json +16 -0
@@ -0,0 +1,285 @@
1
+ import { describe, expect, test, beforeEach, afterEach } from "bun:test"
2
+ import path from "path"
3
+ import { Session } from "../../src/session"
4
+ import { SessionRevert } from "../../src/session/revert"
5
+ import { SessionCompaction } from "../../src/session/compaction"
6
+ import { MessageV2 } from "../../src/session/message-v2"
7
+ import { Log } from "../../src/util/log"
8
+ import { Instance } from "../../src/project/instance"
9
+ import { Identifier } from "../../src/id/id"
10
+ import { tmpdir } from "../fixture/fixture"
11
+
12
+ const projectRoot = path.join(__dirname, "../..")
13
+ Log.init({ print: false })
14
+
15
+ describe("revert + compact workflow", () => {
16
+ test("should properly handle compact command after revert", async () => {
17
+ await using tmp = await tmpdir({ git: true })
18
+ await Instance.provide({
19
+ directory: tmp.path,
20
+ fn: async () => {
21
+ // Create a session
22
+ const session = await Session.create({})
23
+ const sessionID = session.id
24
+
25
+ // Create a user message
26
+ const userMsg1 = await Session.updateMessage({
27
+ id: Identifier.ascending("message"),
28
+ role: "user",
29
+ sessionID,
30
+ agent: "default",
31
+ model: {
32
+ providerID: "openai",
33
+ modelID: "gpt-4",
34
+ },
35
+ time: {
36
+ created: Date.now(),
37
+ },
38
+ })
39
+
40
+ // Add a text part to the user message
41
+ await Session.updatePart({
42
+ id: Identifier.ascending("part"),
43
+ messageID: userMsg1.id,
44
+ sessionID,
45
+ type: "text",
46
+ text: "Hello, please help me",
47
+ })
48
+
49
+ // Create an assistant response message
50
+ const assistantMsg1: MessageV2.Assistant = {
51
+ id: Identifier.ascending("message"),
52
+ role: "assistant",
53
+ sessionID,
54
+ mode: "default",
55
+ agent: "default",
56
+ path: {
57
+ cwd: tmp.path,
58
+ root: tmp.path,
59
+ },
60
+ cost: 0,
61
+ tokens: {
62
+ output: 0,
63
+ input: 0,
64
+ reasoning: 0,
65
+ cache: { read: 0, write: 0 },
66
+ },
67
+ modelID: "gpt-4",
68
+ providerID: "openai",
69
+ parentID: userMsg1.id,
70
+ time: {
71
+ created: Date.now(),
72
+ },
73
+ finish: "end_turn",
74
+ }
75
+ await Session.updateMessage(assistantMsg1)
76
+
77
+ // Add a text part to the assistant message
78
+ await Session.updatePart({
79
+ id: Identifier.ascending("part"),
80
+ messageID: assistantMsg1.id,
81
+ sessionID,
82
+ type: "text",
83
+ text: "Sure, I'll help you!",
84
+ })
85
+
86
+ // Create another user message
87
+ const userMsg2 = await Session.updateMessage({
88
+ id: Identifier.ascending("message"),
89
+ role: "user",
90
+ sessionID,
91
+ agent: "default",
92
+ model: {
93
+ providerID: "openai",
94
+ modelID: "gpt-4",
95
+ },
96
+ time: {
97
+ created: Date.now(),
98
+ },
99
+ })
100
+
101
+ await Session.updatePart({
102
+ id: Identifier.ascending("part"),
103
+ messageID: userMsg2.id,
104
+ sessionID,
105
+ type: "text",
106
+ text: "What's the capital of France?",
107
+ })
108
+
109
+ // Create another assistant response
110
+ const assistantMsg2: MessageV2.Assistant = {
111
+ id: Identifier.ascending("message"),
112
+ role: "assistant",
113
+ sessionID,
114
+ mode: "default",
115
+ agent: "default",
116
+ path: {
117
+ cwd: tmp.path,
118
+ root: tmp.path,
119
+ },
120
+ cost: 0,
121
+ tokens: {
122
+ output: 0,
123
+ input: 0,
124
+ reasoning: 0,
125
+ cache: { read: 0, write: 0 },
126
+ },
127
+ modelID: "gpt-4",
128
+ providerID: "openai",
129
+ parentID: userMsg2.id,
130
+ time: {
131
+ created: Date.now(),
132
+ },
133
+ finish: "end_turn",
134
+ }
135
+ await Session.updateMessage(assistantMsg2)
136
+
137
+ await Session.updatePart({
138
+ id: Identifier.ascending("part"),
139
+ messageID: assistantMsg2.id,
140
+ sessionID,
141
+ type: "text",
142
+ text: "The capital of France is Paris.",
143
+ })
144
+
145
+ // Verify messages before revert
146
+ let messages = await Session.messages({ sessionID })
147
+ expect(messages.length).toBe(4) // 2 user + 2 assistant messages
148
+ const messageIds = messages.map((m) => m.info.id)
149
+ expect(messageIds).toContain(userMsg1.id)
150
+ expect(messageIds).toContain(userMsg2.id)
151
+ expect(messageIds).toContain(assistantMsg1.id)
152
+ expect(messageIds).toContain(assistantMsg2.id)
153
+
154
+ // Revert the last user message (userMsg2)
155
+ await SessionRevert.revert({
156
+ sessionID,
157
+ messageID: userMsg2.id,
158
+ })
159
+
160
+ // Check that revert state is set
161
+ let sessionInfo = await Session.get(sessionID)
162
+ expect(sessionInfo.revert).toBeDefined()
163
+ const revertMessageID = sessionInfo.revert?.messageID
164
+ expect(revertMessageID).toBeDefined()
165
+
166
+ // Messages should still be in the list (not removed yet, just marked for revert)
167
+ messages = await Session.messages({ sessionID })
168
+ expect(messages.length).toBe(4)
169
+
170
+ // Now clean up the revert state (this is what the compact endpoint should do)
171
+ await SessionRevert.cleanup(sessionInfo)
172
+
173
+ // After cleanup, the reverted messages (those after the revert point) should be removed
174
+ messages = await Session.messages({ sessionID })
175
+ const remainingIds = messages.map((m) => m.info.id)
176
+ // The revert point is somewhere in the message chain, so we should have fewer messages
177
+ expect(messages.length).toBeLessThan(4)
178
+ // userMsg2 and assistantMsg2 should be removed (they come after the revert point)
179
+ expect(remainingIds).not.toContain(userMsg2.id)
180
+ expect(remainingIds).not.toContain(assistantMsg2.id)
181
+
182
+ // Revert state should be cleared
183
+ sessionInfo = await Session.get(sessionID)
184
+ expect(sessionInfo.revert).toBeUndefined()
185
+
186
+ // Clean up
187
+ await Session.remove(sessionID)
188
+ },
189
+ })
190
+ })
191
+
192
+ test("should properly clean up revert state before creating compaction message", async () => {
193
+ await using tmp = await tmpdir({ git: true })
194
+ await Instance.provide({
195
+ directory: tmp.path,
196
+ fn: async () => {
197
+ // Create a session
198
+ const session = await Session.create({})
199
+ const sessionID = session.id
200
+
201
+ // Create initial messages
202
+ const userMsg = await Session.updateMessage({
203
+ id: Identifier.ascending("message"),
204
+ role: "user",
205
+ sessionID,
206
+ agent: "default",
207
+ model: {
208
+ providerID: "openai",
209
+ modelID: "gpt-4",
210
+ },
211
+ time: {
212
+ created: Date.now(),
213
+ },
214
+ })
215
+
216
+ await Session.updatePart({
217
+ id: Identifier.ascending("part"),
218
+ messageID: userMsg.id,
219
+ sessionID,
220
+ type: "text",
221
+ text: "Hello",
222
+ })
223
+
224
+ const assistantMsg: MessageV2.Assistant = {
225
+ id: Identifier.ascending("message"),
226
+ role: "assistant",
227
+ sessionID,
228
+ mode: "default",
229
+ agent: "default",
230
+ path: {
231
+ cwd: tmp.path,
232
+ root: tmp.path,
233
+ },
234
+ cost: 0,
235
+ tokens: {
236
+ output: 0,
237
+ input: 0,
238
+ reasoning: 0,
239
+ cache: { read: 0, write: 0 },
240
+ },
241
+ modelID: "gpt-4",
242
+ providerID: "openai",
243
+ parentID: userMsg.id,
244
+ time: {
245
+ created: Date.now(),
246
+ },
247
+ finish: "end_turn",
248
+ }
249
+ await Session.updateMessage(assistantMsg)
250
+
251
+ await Session.updatePart({
252
+ id: Identifier.ascending("part"),
253
+ messageID: assistantMsg.id,
254
+ sessionID,
255
+ type: "text",
256
+ text: "Hi there!",
257
+ })
258
+
259
+ // Revert the user message
260
+ await SessionRevert.revert({
261
+ sessionID,
262
+ messageID: userMsg.id,
263
+ })
264
+
265
+ // Check that revert state is set
266
+ let sessionInfo = await Session.get(sessionID)
267
+ expect(sessionInfo.revert).toBeDefined()
268
+
269
+ // Simulate what the compact endpoint does: cleanup revert before creating compaction
270
+ await SessionRevert.cleanup(sessionInfo)
271
+
272
+ // Verify revert state is cleared
273
+ sessionInfo = await Session.get(sessionID)
274
+ expect(sessionInfo.revert).toBeUndefined()
275
+
276
+ // Verify messages are properly cleaned up
277
+ const messages = await Session.messages({ sessionID })
278
+ expect(messages.length).toBe(0) // All messages should be reverted
279
+
280
+ // Clean up
281
+ await Session.remove(sessionID)
282
+ },
283
+ })
284
+ })
285
+ })
@@ -0,0 +1,71 @@
1
+ import { describe, expect, test } from "bun:test"
2
+ import path from "path"
3
+ import { Session } from "../../src/session"
4
+ import { Bus } from "../../src/bus"
5
+ import { Log } from "../../src/util/log"
6
+ import { Instance } from "../../src/project/instance"
7
+
8
+ const projectRoot = path.join(__dirname, "../..")
9
+ Log.init({ print: false })
10
+
11
+ describe("session.started event", () => {
12
+ test("should emit session.started event when session is created", async () => {
13
+ await Instance.provide({
14
+ directory: projectRoot,
15
+ fn: async () => {
16
+ let eventReceived = false
17
+ let receivedInfo: Session.Info | undefined
18
+
19
+ const unsub = Bus.subscribe(Session.Event.Created, (event) => {
20
+ eventReceived = true
21
+ receivedInfo = event.properties.info as Session.Info
22
+ })
23
+
24
+ const session = await Session.create({})
25
+
26
+ await new Promise((resolve) => setTimeout(resolve, 100))
27
+
28
+ unsub()
29
+
30
+ expect(eventReceived).toBe(true)
31
+ expect(receivedInfo).toBeDefined()
32
+ expect(receivedInfo?.id).toBe(session.id)
33
+ expect(receivedInfo?.projectID).toBe(session.projectID)
34
+ expect(receivedInfo?.directory).toBe(session.directory)
35
+ expect(receivedInfo?.title).toBe(session.title)
36
+
37
+ await Session.remove(session.id)
38
+ },
39
+ })
40
+ })
41
+
42
+ test("session.started event should be emitted before session.updated", async () => {
43
+ await Instance.provide({
44
+ directory: projectRoot,
45
+ fn: async () => {
46
+ const events: string[] = []
47
+
48
+ const unsubStarted = Bus.subscribe(Session.Event.Created, () => {
49
+ events.push("started")
50
+ })
51
+
52
+ const unsubUpdated = Bus.subscribe(Session.Event.Updated, () => {
53
+ events.push("updated")
54
+ })
55
+
56
+ const session = await Session.create({})
57
+
58
+ await new Promise((resolve) => setTimeout(resolve, 100))
59
+
60
+ unsubStarted()
61
+ unsubUpdated()
62
+
63
+ expect(events).toContain("started")
64
+ expect(events).toContain("updated")
65
+ expect(events.indexOf("started")).toBeLessThan(events.indexOf("updated"))
66
+
67
+ await Session.remove(session.id)
68
+ },
69
+ })
70
+ })
71
+ })
@@ -0,0 +1,185 @@
1
+ import { test, expect } from "bun:test"
2
+ import { Skill } from "../../src/skill"
3
+ import { Instance } from "../../src/project/instance"
4
+ import { tmpdir } from "../fixture/fixture"
5
+ import path from "path"
6
+ import fs from "fs/promises"
7
+
8
+ async function createGlobalSkill(homeDir: string) {
9
+ const skillDir = path.join(homeDir, ".claude", "skills", "global-test-skill")
10
+ await fs.mkdir(skillDir, { recursive: true })
11
+ await Bun.write(
12
+ path.join(skillDir, "SKILL.md"),
13
+ `---
14
+ name: global-test-skill
15
+ description: A global skill from ~/.claude/skills for testing.
16
+ ---
17
+
18
+ # Global Test Skill
19
+
20
+ This skill is loaded from the global home directory.
21
+ `,
22
+ )
23
+ }
24
+
25
+ test("discovers skills from .opencode/skill/ directory", async () => {
26
+ await using tmp = await tmpdir({
27
+ git: true,
28
+ init: async (dir) => {
29
+ const skillDir = path.join(dir, ".opencode", "skill", "test-skill")
30
+ await Bun.write(
31
+ path.join(skillDir, "SKILL.md"),
32
+ `---
33
+ name: test-skill
34
+ description: A test skill for verification.
35
+ ---
36
+
37
+ # Test Skill
38
+
39
+ Instructions here.
40
+ `,
41
+ )
42
+ },
43
+ })
44
+
45
+ await Instance.provide({
46
+ directory: tmp.path,
47
+ fn: async () => {
48
+ const skills = await Skill.all()
49
+ expect(skills.length).toBe(1)
50
+ const testSkill = skills.find((s) => s.name === "test-skill")
51
+ expect(testSkill).toBeDefined()
52
+ expect(testSkill!.description).toBe("A test skill for verification.")
53
+ expect(testSkill!.location).toContain("skill/test-skill/SKILL.md")
54
+ },
55
+ })
56
+ })
57
+
58
+ test("discovers multiple skills from .opencode/skill/ directory", async () => {
59
+ await using tmp = await tmpdir({
60
+ git: true,
61
+ init: async (dir) => {
62
+ const skillDir1 = path.join(dir, ".opencode", "skill", "skill-one")
63
+ const skillDir2 = path.join(dir, ".opencode", "skill", "skill-two")
64
+ await Bun.write(
65
+ path.join(skillDir1, "SKILL.md"),
66
+ `---
67
+ name: skill-one
68
+ description: First test skill.
69
+ ---
70
+
71
+ # Skill One
72
+ `,
73
+ )
74
+ await Bun.write(
75
+ path.join(skillDir2, "SKILL.md"),
76
+ `---
77
+ name: skill-two
78
+ description: Second test skill.
79
+ ---
80
+
81
+ # Skill Two
82
+ `,
83
+ )
84
+ },
85
+ })
86
+
87
+ await Instance.provide({
88
+ directory: tmp.path,
89
+ fn: async () => {
90
+ const skills = await Skill.all()
91
+ expect(skills.length).toBe(2)
92
+ expect(skills.find((s) => s.name === "skill-one")).toBeDefined()
93
+ expect(skills.find((s) => s.name === "skill-two")).toBeDefined()
94
+ },
95
+ })
96
+ })
97
+
98
+ test("skips skills with missing frontmatter", async () => {
99
+ await using tmp = await tmpdir({
100
+ git: true,
101
+ init: async (dir) => {
102
+ const skillDir = path.join(dir, ".opencode", "skill", "no-frontmatter")
103
+ await Bun.write(
104
+ path.join(skillDir, "SKILL.md"),
105
+ `# No Frontmatter
106
+
107
+ Just some content without YAML frontmatter.
108
+ `,
109
+ )
110
+ },
111
+ })
112
+
113
+ await Instance.provide({
114
+ directory: tmp.path,
115
+ fn: async () => {
116
+ const skills = await Skill.all()
117
+ expect(skills).toEqual([])
118
+ },
119
+ })
120
+ })
121
+
122
+ test("discovers skills from .claude/skills/ directory", async () => {
123
+ await using tmp = await tmpdir({
124
+ git: true,
125
+ init: async (dir) => {
126
+ const skillDir = path.join(dir, ".claude", "skills", "claude-skill")
127
+ await Bun.write(
128
+ path.join(skillDir, "SKILL.md"),
129
+ `---
130
+ name: claude-skill
131
+ description: A skill in the .claude/skills directory.
132
+ ---
133
+
134
+ # Claude Skill
135
+ `,
136
+ )
137
+ },
138
+ })
139
+
140
+ await Instance.provide({
141
+ directory: tmp.path,
142
+ fn: async () => {
143
+ const skills = await Skill.all()
144
+ expect(skills.length).toBe(1)
145
+ const claudeSkill = skills.find((s) => s.name === "claude-skill")
146
+ expect(claudeSkill).toBeDefined()
147
+ expect(claudeSkill!.location).toContain(".claude/skills/claude-skill/SKILL.md")
148
+ },
149
+ })
150
+ })
151
+
152
+ test("discovers global skills from ~/.claude/skills/ directory", async () => {
153
+ await using tmp = await tmpdir({ git: true })
154
+
155
+ const originalHome = process.env.OPENCODE_TEST_HOME
156
+ process.env.OPENCODE_TEST_HOME = tmp.path
157
+
158
+ try {
159
+ await createGlobalSkill(tmp.path)
160
+ await Instance.provide({
161
+ directory: tmp.path,
162
+ fn: async () => {
163
+ const skills = await Skill.all()
164
+ expect(skills.length).toBe(1)
165
+ expect(skills[0].name).toBe("global-test-skill")
166
+ expect(skills[0].description).toBe("A global skill from ~/.claude/skills for testing.")
167
+ expect(skills[0].location).toContain(".claude/skills/global-test-skill/SKILL.md")
168
+ },
169
+ })
170
+ } finally {
171
+ process.env.OPENCODE_TEST_HOME = originalHome
172
+ }
173
+ })
174
+
175
+ test("returns empty array when no skills exist", async () => {
176
+ await using tmp = await tmpdir({ git: true })
177
+
178
+ await Instance.provide({
179
+ directory: tmp.path,
180
+ fn: async () => {
181
+ const skills = await Skill.all()
182
+ expect(skills).toEqual([])
183
+ },
184
+ })
185
+ })