@stonerzju/opencode 1.2.17 → 1.2.18

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,110 +0,0 @@
1
- import { describe, test, expect, beforeAll, afterAll } from "bun:test"
2
- import { Discovery } from "../../src/skill/discovery"
3
- import { Filesystem } from "../../src/util/filesystem"
4
- import { rm } from "fs/promises"
5
- import path from "path"
6
-
7
- let CLOUDFLARE_SKILLS_URL: string
8
- let server: ReturnType<typeof Bun.serve>
9
- let downloadCount = 0
10
-
11
- const fixturePath = path.join(import.meta.dir, "../fixture/skills")
12
-
13
- beforeAll(async () => {
14
- await rm(Discovery.dir(), { recursive: true, force: true })
15
-
16
- server = Bun.serve({
17
- port: 0,
18
- async fetch(req) {
19
- const url = new URL(req.url)
20
-
21
- // route /.well-known/skills/* to the fixture directory
22
- if (url.pathname.startsWith("/.well-known/skills/")) {
23
- const filePath = url.pathname.replace("/.well-known/skills/", "")
24
- const fullPath = path.join(fixturePath, filePath)
25
-
26
- if (await Filesystem.exists(fullPath)) {
27
- if (!fullPath.endsWith("index.json")) {
28
- downloadCount++
29
- }
30
- return new Response(Bun.file(fullPath))
31
- }
32
- }
33
-
34
- return new Response("Not Found", { status: 404 })
35
- },
36
- })
37
-
38
- CLOUDFLARE_SKILLS_URL = `http://localhost:${server.port}/.well-known/skills/`
39
- })
40
-
41
- afterAll(async () => {
42
- server?.stop()
43
- await rm(Discovery.dir(), { recursive: true, force: true })
44
- })
45
-
46
- describe("Discovery.pull", () => {
47
- test("downloads skills from cloudflare url", async () => {
48
- const dirs = await Discovery.pull(CLOUDFLARE_SKILLS_URL)
49
- expect(dirs.length).toBeGreaterThan(0)
50
- for (const dir of dirs) {
51
- expect(dir).toStartWith(Discovery.dir())
52
- const md = path.join(dir, "SKILL.md")
53
- expect(await Filesystem.exists(md)).toBe(true)
54
- }
55
- })
56
-
57
- test("url without trailing slash works", async () => {
58
- const dirs = await Discovery.pull(CLOUDFLARE_SKILLS_URL.replace(/\/$/, ""))
59
- expect(dirs.length).toBeGreaterThan(0)
60
- for (const dir of dirs) {
61
- const md = path.join(dir, "SKILL.md")
62
- expect(await Filesystem.exists(md)).toBe(true)
63
- }
64
- })
65
-
66
- test("returns empty array for invalid url", async () => {
67
- const dirs = await Discovery.pull(`http://localhost:${server.port}/invalid-url/`)
68
- expect(dirs).toEqual([])
69
- })
70
-
71
- test("returns empty array for non-json response", async () => {
72
- // any url not explicitly handled in server returns 404 text "Not Found"
73
- const dirs = await Discovery.pull(`http://localhost:${server.port}/some-other-path/`)
74
- expect(dirs).toEqual([])
75
- })
76
-
77
- test("downloads reference files alongside SKILL.md", async () => {
78
- const dirs = await Discovery.pull(CLOUDFLARE_SKILLS_URL)
79
- // find a skill dir that should have reference files (e.g. agents-sdk)
80
- const agentsSdk = dirs.find((d) => d.endsWith(path.sep + "agents-sdk"))
81
- expect(agentsSdk).toBeDefined()
82
- if (agentsSdk) {
83
- const refs = path.join(agentsSdk, "references")
84
- expect(await Filesystem.exists(path.join(agentsSdk, "SKILL.md"))).toBe(true)
85
- // agents-sdk has reference files per the index
86
- const refDir = await Array.fromAsync(new Bun.Glob("**/*.md").scan({ cwd: refs, onlyFiles: true }))
87
- expect(refDir.length).toBeGreaterThan(0)
88
- }
89
- })
90
-
91
- test("caches downloaded files on second pull", async () => {
92
- // clear dir and downloadCount
93
- await rm(Discovery.dir(), { recursive: true, force: true })
94
- downloadCount = 0
95
-
96
- // first pull to populate cache
97
- const first = await Discovery.pull(CLOUDFLARE_SKILLS_URL)
98
- expect(first.length).toBeGreaterThan(0)
99
- const firstCount = downloadCount
100
- expect(firstCount).toBeGreaterThan(0)
101
-
102
- // second pull should return same results from cache
103
- const second = await Discovery.pull(CLOUDFLARE_SKILLS_URL)
104
- expect(second.length).toBe(first.length)
105
- expect(second.sort()).toEqual(first.sort())
106
-
107
- // second pull should NOT increment download count
108
- expect(downloadCount).toBe(firstCount)
109
- })
110
- })
@@ -1,388 +0,0 @@
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(path.join("skill", "test-skill", "SKILL.md"))
54
- },
55
- })
56
- })
57
-
58
- test("returns skill directories from Skill.dirs", async () => {
59
- await using tmp = await tmpdir({
60
- git: true,
61
- init: async (dir) => {
62
- const skillDir = path.join(dir, ".opencode", "skill", "dir-skill")
63
- await Bun.write(
64
- path.join(skillDir, "SKILL.md"),
65
- `---
66
- name: dir-skill
67
- description: Skill for dirs test.
68
- ---
69
-
70
- # Dir Skill
71
- `,
72
- )
73
- },
74
- })
75
-
76
- const home = process.env.OPENCODE_TEST_HOME
77
- process.env.OPENCODE_TEST_HOME = tmp.path
78
-
79
- try {
80
- await Instance.provide({
81
- directory: tmp.path,
82
- fn: async () => {
83
- const dirs = await Skill.dirs()
84
- const skillDir = path.join(tmp.path, ".opencode", "skill", "dir-skill")
85
- expect(dirs).toContain(skillDir)
86
- expect(dirs.length).toBe(1)
87
- },
88
- })
89
- } finally {
90
- process.env.OPENCODE_TEST_HOME = home
91
- }
92
- })
93
-
94
- test("discovers multiple skills from .opencode/skill/ directory", async () => {
95
- await using tmp = await tmpdir({
96
- git: true,
97
- init: async (dir) => {
98
- const skillDir1 = path.join(dir, ".opencode", "skill", "skill-one")
99
- const skillDir2 = path.join(dir, ".opencode", "skill", "skill-two")
100
- await Bun.write(
101
- path.join(skillDir1, "SKILL.md"),
102
- `---
103
- name: skill-one
104
- description: First test skill.
105
- ---
106
-
107
- # Skill One
108
- `,
109
- )
110
- await Bun.write(
111
- path.join(skillDir2, "SKILL.md"),
112
- `---
113
- name: skill-two
114
- description: Second test skill.
115
- ---
116
-
117
- # Skill Two
118
- `,
119
- )
120
- },
121
- })
122
-
123
- await Instance.provide({
124
- directory: tmp.path,
125
- fn: async () => {
126
- const skills = await Skill.all()
127
- expect(skills.length).toBe(2)
128
- expect(skills.find((s) => s.name === "skill-one")).toBeDefined()
129
- expect(skills.find((s) => s.name === "skill-two")).toBeDefined()
130
- },
131
- })
132
- })
133
-
134
- test("skips skills with missing frontmatter", async () => {
135
- await using tmp = await tmpdir({
136
- git: true,
137
- init: async (dir) => {
138
- const skillDir = path.join(dir, ".opencode", "skill", "no-frontmatter")
139
- await Bun.write(
140
- path.join(skillDir, "SKILL.md"),
141
- `# No Frontmatter
142
-
143
- Just some content without YAML frontmatter.
144
- `,
145
- )
146
- },
147
- })
148
-
149
- await Instance.provide({
150
- directory: tmp.path,
151
- fn: async () => {
152
- const skills = await Skill.all()
153
- expect(skills).toEqual([])
154
- },
155
- })
156
- })
157
-
158
- test("discovers skills from .claude/skills/ directory", async () => {
159
- await using tmp = await tmpdir({
160
- git: true,
161
- init: async (dir) => {
162
- const skillDir = path.join(dir, ".claude", "skills", "claude-skill")
163
- await Bun.write(
164
- path.join(skillDir, "SKILL.md"),
165
- `---
166
- name: claude-skill
167
- description: A skill in the .claude/skills directory.
168
- ---
169
-
170
- # Claude Skill
171
- `,
172
- )
173
- },
174
- })
175
-
176
- await Instance.provide({
177
- directory: tmp.path,
178
- fn: async () => {
179
- const skills = await Skill.all()
180
- expect(skills.length).toBe(1)
181
- const claudeSkill = skills.find((s) => s.name === "claude-skill")
182
- expect(claudeSkill).toBeDefined()
183
- expect(claudeSkill!.location).toContain(path.join(".claude", "skills", "claude-skill", "SKILL.md"))
184
- },
185
- })
186
- })
187
-
188
- test("discovers global skills from ~/.claude/skills/ directory", async () => {
189
- await using tmp = await tmpdir({ git: true })
190
-
191
- const originalHome = process.env.OPENCODE_TEST_HOME
192
- process.env.OPENCODE_TEST_HOME = tmp.path
193
-
194
- try {
195
- await createGlobalSkill(tmp.path)
196
- await Instance.provide({
197
- directory: tmp.path,
198
- fn: async () => {
199
- const skills = await Skill.all()
200
- expect(skills.length).toBe(1)
201
- expect(skills[0].name).toBe("global-test-skill")
202
- expect(skills[0].description).toBe("A global skill from ~/.claude/skills for testing.")
203
- expect(skills[0].location).toContain(path.join(".claude", "skills", "global-test-skill", "SKILL.md"))
204
- },
205
- })
206
- } finally {
207
- process.env.OPENCODE_TEST_HOME = originalHome
208
- }
209
- })
210
-
211
- test("returns empty array when no skills exist", async () => {
212
- await using tmp = await tmpdir({ git: true })
213
-
214
- await Instance.provide({
215
- directory: tmp.path,
216
- fn: async () => {
217
- const skills = await Skill.all()
218
- expect(skills).toEqual([])
219
- },
220
- })
221
- })
222
-
223
- test("discovers skills from .agents/skills/ directory", async () => {
224
- await using tmp = await tmpdir({
225
- git: true,
226
- init: async (dir) => {
227
- const skillDir = path.join(dir, ".agents", "skills", "agent-skill")
228
- await Bun.write(
229
- path.join(skillDir, "SKILL.md"),
230
- `---
231
- name: agent-skill
232
- description: A skill in the .agents/skills directory.
233
- ---
234
-
235
- # Agent Skill
236
- `,
237
- )
238
- },
239
- })
240
-
241
- await Instance.provide({
242
- directory: tmp.path,
243
- fn: async () => {
244
- const skills = await Skill.all()
245
- expect(skills.length).toBe(1)
246
- const agentSkill = skills.find((s) => s.name === "agent-skill")
247
- expect(agentSkill).toBeDefined()
248
- expect(agentSkill!.location).toContain(path.join(".agents", "skills", "agent-skill", "SKILL.md"))
249
- },
250
- })
251
- })
252
-
253
- test("discovers global skills from ~/.agents/skills/ directory", async () => {
254
- await using tmp = await tmpdir({ git: true })
255
-
256
- const originalHome = process.env.OPENCODE_TEST_HOME
257
- process.env.OPENCODE_TEST_HOME = tmp.path
258
-
259
- try {
260
- const skillDir = path.join(tmp.path, ".agents", "skills", "global-agent-skill")
261
- await fs.mkdir(skillDir, { recursive: true })
262
- await Bun.write(
263
- path.join(skillDir, "SKILL.md"),
264
- `---
265
- name: global-agent-skill
266
- description: A global skill from ~/.agents/skills for testing.
267
- ---
268
-
269
- # Global Agent Skill
270
-
271
- This skill is loaded from the global home directory.
272
- `,
273
- )
274
-
275
- await Instance.provide({
276
- directory: tmp.path,
277
- fn: async () => {
278
- const skills = await Skill.all()
279
- expect(skills.length).toBe(1)
280
- expect(skills[0].name).toBe("global-agent-skill")
281
- expect(skills[0].description).toBe("A global skill from ~/.agents/skills for testing.")
282
- expect(skills[0].location).toContain(path.join(".agents", "skills", "global-agent-skill", "SKILL.md"))
283
- },
284
- })
285
- } finally {
286
- process.env.OPENCODE_TEST_HOME = originalHome
287
- }
288
- })
289
-
290
- test("discovers skills from both .claude/skills/ and .agents/skills/", async () => {
291
- await using tmp = await tmpdir({
292
- git: true,
293
- init: async (dir) => {
294
- const claudeDir = path.join(dir, ".claude", "skills", "claude-skill")
295
- const agentDir = path.join(dir, ".agents", "skills", "agent-skill")
296
- await Bun.write(
297
- path.join(claudeDir, "SKILL.md"),
298
- `---
299
- name: claude-skill
300
- description: A skill in the .claude/skills directory.
301
- ---
302
-
303
- # Claude Skill
304
- `,
305
- )
306
- await Bun.write(
307
- path.join(agentDir, "SKILL.md"),
308
- `---
309
- name: agent-skill
310
- description: A skill in the .agents/skills directory.
311
- ---
312
-
313
- # Agent Skill
314
- `,
315
- )
316
- },
317
- })
318
-
319
- await Instance.provide({
320
- directory: tmp.path,
321
- fn: async () => {
322
- const skills = await Skill.all()
323
- expect(skills.length).toBe(2)
324
- expect(skills.find((s) => s.name === "claude-skill")).toBeDefined()
325
- expect(skills.find((s) => s.name === "agent-skill")).toBeDefined()
326
- },
327
- })
328
- })
329
-
330
- test("properly resolves directories that skills live in", async () => {
331
- await using tmp = await tmpdir({
332
- git: true,
333
- init: async (dir) => {
334
- const opencodeSkillDir = path.join(dir, ".opencode", "skill", "agent-skill")
335
- const opencodeSkillsDir = path.join(dir, ".opencode", "skills", "agent-skill")
336
- const claudeDir = path.join(dir, ".claude", "skills", "claude-skill")
337
- const agentDir = path.join(dir, ".agents", "skills", "agent-skill")
338
- await Bun.write(
339
- path.join(claudeDir, "SKILL.md"),
340
- `---
341
- name: claude-skill
342
- description: A skill in the .claude/skills directory.
343
- ---
344
-
345
- # Claude Skill
346
- `,
347
- )
348
- await Bun.write(
349
- path.join(agentDir, "SKILL.md"),
350
- `---
351
- name: agent-skill
352
- description: A skill in the .agents/skills directory.
353
- ---
354
-
355
- # Agent Skill
356
- `,
357
- )
358
- await Bun.write(
359
- path.join(opencodeSkillDir, "SKILL.md"),
360
- `---
361
- name: opencode-skill
362
- description: A skill in the .opencode/skill directory.
363
- ---
364
-
365
- # OpenCode Skill
366
- `,
367
- )
368
- await Bun.write(
369
- path.join(opencodeSkillsDir, "SKILL.md"),
370
- `---
371
- name: opencode-skill
372
- description: A skill in the .opencode/skills directory.
373
- ---
374
-
375
- # OpenCode Skill
376
- `,
377
- )
378
- },
379
- })
380
-
381
- await Instance.provide({
382
- directory: tmp.path,
383
- fn: async () => {
384
- const dirs = await Skill.dirs()
385
- expect(dirs.length).toBe(4)
386
- },
387
- })
388
- })