@stonerzju/opencode 1.2.17 → 1.2.19
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/bin/opencode +29 -157
- package/package.json +29 -29
- package/src/acp/agent.ts +4 -4
- package/src/acp/session.ts +1 -1
- package/src/agent/agent.ts +3 -3
- package/src/bun/index.ts +2 -2
- package/src/cli/cmd/acp.ts +3 -3
- package/src/cli/cmd/debug/file.ts +1 -1
- package/src/cli/cmd/github.ts +2 -2
- package/src/cli/cmd/pr.ts +1 -1
- package/src/cli/cmd/tui/app.tsx +24 -24
- package/src/cli/cmd/tui/attach.ts +3 -3
- package/src/cli/cmd/tui/component/dialog-agent.tsx +3 -3
- package/src/cli/cmd/tui/component/dialog-command.tsx +3 -3
- package/src/cli/cmd/tui/component/dialog-mcp.tsx +5 -5
- package/src/cli/cmd/tui/component/dialog-model.tsx +4 -4
- package/src/cli/cmd/tui/component/dialog-provider.tsx +4 -4
- package/src/cli/cmd/tui/component/dialog-session-list.tsx +5 -5
- package/src/cli/cmd/tui/component/dialog-session-rename.tsx +3 -3
- package/src/cli/cmd/tui/component/dialog-skill.tsx +3 -3
- package/src/cli/cmd/tui/component/dialog-stash.tsx +3 -3
- package/src/cli/cmd/tui/component/dialog-status.tsx +2 -2
- package/src/cli/cmd/tui/component/dialog-tag.tsx +3 -3
- package/src/cli/cmd/tui/component/logo.tsx +2 -2
- package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +6 -6
- package/src/cli/cmd/tui/component/prompt/frecency.tsx +2 -2
- package/src/cli/cmd/tui/component/prompt/history.tsx +2 -2
- package/src/cli/cmd/tui/component/prompt/index.tsx +14 -14
- package/src/cli/cmd/tui/component/prompt/stash.tsx +2 -2
- package/src/cli/cmd/tui/component/textarea-keybindings.ts +1 -1
- package/src/cli/cmd/tui/component/tips.tsx +1 -1
- package/src/cli/cmd/tui/context/directory.ts +1 -1
- package/src/cli/cmd/tui/context/exit.tsx +1 -1
- package/src/cli/cmd/tui/context/keybind.tsx +2 -2
- package/src/cli/cmd/tui/context/kv.tsx +2 -2
- package/src/cli/cmd/tui/context/local.tsx +6 -6
- package/src/cli/cmd/tui/context/sync.tsx +4 -4
- package/src/cli/cmd/tui/context/theme/opencode.json +245 -0
- package/src/cli/cmd/tui/context/theme.tsx +2 -2
- package/src/cli/cmd/tui/context/tui-config.tsx +1 -1
- package/src/cli/cmd/tui/event.ts +2 -2
- package/src/cli/cmd/tui/routes/home.tsx +6 -6
- package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +6 -6
- package/src/cli/cmd/tui/routes/session/dialog-message.tsx +6 -6
- package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +2 -2
- package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +3 -3
- package/src/cli/cmd/tui/routes/session/header.tsx +5 -5
- package/src/cli/cmd/tui/routes/session/index.tsx +32 -32
- package/src/cli/cmd/tui/routes/session/permission.tsx +4 -4
- package/src/cli/cmd/tui/routes/session/sidebar.tsx +4 -4
- package/src/cli/cmd/tui/thread.ts +9 -9
- package/src/cli/cmd/tui/ui/dialog-confirm.tsx +1 -1
- package/src/cli/cmd/tui/ui/dialog-help.tsx +2 -2
- package/src/cli/cmd/tui/ui/dialog-select.tsx +5 -5
- package/src/cli/cmd/tui/ui/dialog.tsx +3 -3
- package/src/cli/cmd/tui/ui/toast.tsx +1 -1
- package/src/cli/cmd/tui/util/editor.ts +3 -3
- package/src/cli/cmd/tui/util/transcript.ts +1 -1
- package/src/cli/cmd/tui/worker.ts +10 -10
- package/src/cli/error.ts +1 -1
- package/src/cli/ui.ts +1 -1
- package/src/cli/upgrade.ts +4 -4
- package/src/command/index.ts +1 -1
- package/src/config/config.ts +10 -10
- package/src/config/markdown.ts +1 -1
- package/src/config/migrate-tui-config.ts +5 -5
- package/src/config/paths.ts +4 -4
- package/src/config/tui.ts +4 -4
- package/src/control/control.sql.ts +1 -1
- package/src/control/index.ts +1 -1
- package/src/control-plane/adaptors/worktree.ts +1 -1
- package/src/control-plane/session-proxy-middleware.ts +1 -1
- package/src/control-plane/workspace.sql.ts +1 -1
- package/src/control-plane/workspace.ts +7 -7
- package/src/file/index.ts +1 -1
- package/src/file/ripgrep.ts +2 -2
- package/src/file/watcher.ts +5 -5
- package/src/format/formatter.ts +1 -1
- package/src/ide/index.ts +3 -3
- package/src/index.ts +1 -1
- package/src/installation/index.ts +3 -3
- package/src/lsp/client.ts +3 -3
- package/src/lsp/index.ts +3 -3
- package/src/mcp/index.ts +4 -4
- package/src/permission/index.ts +2 -2
- package/src/permission/next.ts +10 -10
- package/src/plugin/codex.ts +1 -1
- package/src/plugin/copilot.ts +2 -2
- package/src/plugin/index.ts +1 -1
- package/src/project/bootstrap.ts +2 -2
- package/src/project/instance.ts +4 -4
- package/src/project/project.sql.ts +1 -1
- package/src/project/project.ts +5 -5
- package/src/project/state.ts +1 -1
- package/src/project/vcs.ts +4 -4
- package/src/provider/auth.ts +4 -4
- package/src/provider/error.ts +1 -1
- package/src/provider/models-snapshot.ts +2 -0
- package/src/provider/models.ts +1 -1
- package/src/provider/provider.ts +2 -2
- package/src/provider/transform.ts +2 -2
- package/src/pty/index.ts +5 -5
- package/src/question/index.ts +5 -5
- package/src/server/event.ts +1 -1
- package/src/server/mdns.ts +1 -1
- package/src/server/routes/global.ts +3 -3
- package/src/server/routes/permission.ts +1 -1
- package/src/server/routes/pty.ts +1 -1
- package/src/server/routes/session.ts +4 -4
- package/src/server/routes/tui.ts +1 -1
- package/src/server/server.ts +3 -3
- package/src/session/compaction.ts +7 -7
- package/src/session/index.ts +10 -10
- package/src/session/instruction.ts +1 -1
- package/src/session/llm.ts +11 -11
- package/src/session/message-v2.ts +10 -10
- package/src/session/message.ts +1 -1
- package/src/session/processor.ts +10 -10
- package/src/session/prompt.ts +8 -8
- package/src/session/retry.ts +2 -2
- package/src/session/revert.ts +1 -1
- package/src/session/session.sql.ts +3 -3
- package/src/session/status.ts +3 -3
- package/src/session/summary.ts +5 -5
- package/src/session/system.ts +1 -1
- package/src/session/todo.ts +2 -2
- package/src/share/share-next.ts +7 -7
- package/src/share/share.sql.ts +1 -1
- package/src/shell/shell.ts +3 -3
- package/src/skill/skill.ts +6 -6
- package/src/storage/db.ts +1 -1
- package/src/storage/storage.ts +1 -1
- package/src/tool/bash.ts +6 -6
- package/src/tool/edit.ts +1 -1
- package/src/tool/registry.ts +2 -2
- package/src/tool/skill.ts +1 -1
- package/src/tool/task.ts +3 -3
- package/src/util/array.ts +10 -0
- package/src/util/binary.ts +41 -0
- package/src/util/encode.ts +51 -0
- package/src/util/error.ts +54 -0
- package/src/util/identifier.ts +48 -0
- package/src/util/lazy.ts +4 -16
- package/src/util/path.ts +37 -0
- package/src/util/retry.ts +41 -0
- package/src/util/slug.ts +74 -0
- package/src/worktree/index.ts +3 -3
- package/AGENTS.md +0 -10
- package/BUN_SHELL_MIGRATION_PLAN.md +0 -136
- package/Dockerfile +0 -18
- package/README.md +0 -15
- package/bunfig.toml +0 -7
- package/drizzle.config.ts +0 -10
- package/script/build.ts +0 -224
- package/script/check-migrations.ts +0 -16
- package/script/postinstall.mjs +0 -131
- package/script/publish.ts +0 -181
- package/script/schema.ts +0 -63
- package/script/seed-e2e.ts +0 -50
- package/sst-env.d.ts +0 -10
- package/test/AGENTS.md +0 -81
- package/test/acp/agent-interface.test.ts +0 -51
- package/test/acp/event-subscription.test.ts +0 -683
- package/test/agent/agent.test.ts +0 -689
- package/test/bun.test.ts +0 -53
- package/test/cli/github-action.test.ts +0 -197
- package/test/cli/github-remote.test.ts +0 -80
- package/test/cli/import.test.ts +0 -38
- package/test/cli/plugin-auth-picker.test.ts +0 -120
- package/test/cli/tui/transcript.test.ts +0 -322
- package/test/config/agent-color.test.ts +0 -71
- package/test/config/config.test.ts +0 -1886
- package/test/config/fixtures/empty-frontmatter.md +0 -4
- package/test/config/fixtures/frontmatter.md +0 -28
- package/test/config/fixtures/markdown-header.md +0 -11
- package/test/config/fixtures/no-frontmatter.md +0 -1
- package/test/config/fixtures/weird-model-id.md +0 -13
- package/test/config/markdown.test.ts +0 -228
- package/test/config/tui.test.ts +0 -510
- package/test/control-plane/session-proxy-middleware.test.ts +0 -147
- package/test/control-plane/sse.test.ts +0 -56
- package/test/control-plane/workspace-server-sse.test.ts +0 -65
- package/test/control-plane/workspace-sync.test.ts +0 -97
- package/test/file/ignore.test.ts +0 -10
- package/test/file/index.test.ts +0 -394
- package/test/file/path-traversal.test.ts +0 -198
- package/test/file/ripgrep.test.ts +0 -39
- package/test/file/time.test.ts +0 -361
- package/test/fixture/db.ts +0 -11
- package/test/fixture/fixture.ts +0 -45
- package/test/fixture/lsp/fake-lsp-server.js +0 -77
- package/test/fixture/skills/agents-sdk/SKILL.md +0 -152
- package/test/fixture/skills/agents-sdk/references/callable.md +0 -92
- package/test/fixture/skills/cloudflare/SKILL.md +0 -211
- package/test/fixture/skills/index.json +0 -6
- package/test/ide/ide.test.ts +0 -82
- package/test/keybind.test.ts +0 -421
- package/test/lsp/client.test.ts +0 -95
- package/test/mcp/headers.test.ts +0 -153
- package/test/mcp/oauth-browser.test.ts +0 -249
- package/test/memory/abort-leak.test.ts +0 -136
- package/test/patch/patch.test.ts +0 -348
- package/test/permission/arity.test.ts +0 -33
- package/test/permission/next.test.ts +0 -689
- package/test/permission-task.test.ts +0 -319
- package/test/plugin/auth-override.test.ts +0 -44
- package/test/plugin/codex.test.ts +0 -123
- package/test/preload.ts +0 -80
- package/test/project/project.test.ts +0 -348
- package/test/project/worktree-remove.test.ts +0 -65
- package/test/provider/amazon-bedrock.test.ts +0 -446
- package/test/provider/copilot/convert-to-copilot-messages.test.ts +0 -523
- package/test/provider/copilot/copilot-chat-model.test.ts +0 -592
- package/test/provider/gitlab-duo.test.ts +0 -262
- package/test/provider/provider.test.ts +0 -2220
- package/test/provider/transform.test.ts +0 -2353
- package/test/pty/pty-output-isolation.test.ts +0 -140
- package/test/question/question.test.ts +0 -300
- package/test/scheduler.test.ts +0 -73
- package/test/server/global-session-list.test.ts +0 -89
- package/test/server/session-list.test.ts +0 -90
- package/test/server/session-select.test.ts +0 -78
- package/test/session/compaction.test.ts +0 -423
- package/test/session/instruction.test.ts +0 -170
- package/test/session/llm.test.ts +0 -667
- package/test/session/message-v2.test.ts +0 -924
- package/test/session/prompt.test.ts +0 -211
- package/test/session/retry.test.ts +0 -188
- package/test/session/revert-compact.test.ts +0 -285
- package/test/session/session.test.ts +0 -71
- package/test/session/structured-output-integration.test.ts +0 -233
- package/test/session/structured-output.test.ts +0 -385
- package/test/skill/discovery.test.ts +0 -110
- package/test/skill/skill.test.ts +0 -388
- package/test/snapshot/snapshot.test.ts +0 -1180
- package/test/storage/json-migration.test.ts +0 -846
- package/test/tool/__snapshots__/tool.test.ts.snap +0 -9
- package/test/tool/apply_patch.test.ts +0 -566
- package/test/tool/bash.test.ts +0 -402
- package/test/tool/edit.test.ts +0 -496
- package/test/tool/external-directory.test.ts +0 -127
- package/test/tool/fixtures/large-image.png +0 -0
- package/test/tool/fixtures/models-api.json +0 -38413
- package/test/tool/grep.test.ts +0 -110
- package/test/tool/question.test.ts +0 -107
- package/test/tool/read.test.ts +0 -504
- package/test/tool/registry.test.ts +0 -122
- package/test/tool/skill.test.ts +0 -112
- package/test/tool/truncation.test.ts +0 -160
- package/test/tool/webfetch.test.ts +0 -100
- package/test/tool/write.test.ts +0 -348
- package/test/util/filesystem.test.ts +0 -443
- package/test/util/format.test.ts +0 -59
- package/test/util/glob.test.ts +0 -164
- package/test/util/iife.test.ts +0 -36
- package/test/util/lazy.test.ts +0 -50
- package/test/util/lock.test.ts +0 -72
- package/test/util/process.test.ts +0 -59
- package/test/util/timeout.test.ts +0 -21
- package/test/util/wildcard.test.ts +0 -90
- 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
|
-
})
|
package/test/skill/skill.test.ts
DELETED
|
@@ -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
|
-
})
|