@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.
- 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
package/test/tool/bash.test.ts
DELETED
|
@@ -1,402 +0,0 @@
|
|
|
1
|
-
import { describe, expect, test } from "bun:test"
|
|
2
|
-
import os from "os"
|
|
3
|
-
import path from "path"
|
|
4
|
-
import { BashTool } from "../../src/tool/bash"
|
|
5
|
-
import { Instance } from "../../src/project/instance"
|
|
6
|
-
import { Filesystem } from "../../src/util/filesystem"
|
|
7
|
-
import { tmpdir } from "../fixture/fixture"
|
|
8
|
-
import type { PermissionNext } from "../../src/permission/next"
|
|
9
|
-
import { Truncate } from "../../src/tool/truncation"
|
|
10
|
-
|
|
11
|
-
const ctx = {
|
|
12
|
-
sessionID: "test",
|
|
13
|
-
messageID: "",
|
|
14
|
-
callID: "",
|
|
15
|
-
agent: "build",
|
|
16
|
-
abort: AbortSignal.any([]),
|
|
17
|
-
messages: [],
|
|
18
|
-
metadata: () => {},
|
|
19
|
-
ask: async () => {},
|
|
20
|
-
}
|
|
21
|
-
|
|
22
|
-
const projectRoot = path.join(__dirname, "../..")
|
|
23
|
-
|
|
24
|
-
describe("tool.bash", () => {
|
|
25
|
-
test("basic", async () => {
|
|
26
|
-
await Instance.provide({
|
|
27
|
-
directory: projectRoot,
|
|
28
|
-
fn: async () => {
|
|
29
|
-
const bash = await BashTool.init()
|
|
30
|
-
const result = await bash.execute(
|
|
31
|
-
{
|
|
32
|
-
command: "echo 'test'",
|
|
33
|
-
description: "Echo test message",
|
|
34
|
-
},
|
|
35
|
-
ctx,
|
|
36
|
-
)
|
|
37
|
-
expect(result.metadata.exit).toBe(0)
|
|
38
|
-
expect(result.metadata.output).toContain("test")
|
|
39
|
-
},
|
|
40
|
-
})
|
|
41
|
-
})
|
|
42
|
-
})
|
|
43
|
-
|
|
44
|
-
describe("tool.bash permissions", () => {
|
|
45
|
-
test("asks for bash permission with correct pattern", async () => {
|
|
46
|
-
await using tmp = await tmpdir({ git: true })
|
|
47
|
-
await Instance.provide({
|
|
48
|
-
directory: tmp.path,
|
|
49
|
-
fn: async () => {
|
|
50
|
-
const bash = await BashTool.init()
|
|
51
|
-
const requests: Array<Omit<PermissionNext.Request, "id" | "sessionID" | "tool">> = []
|
|
52
|
-
const testCtx = {
|
|
53
|
-
...ctx,
|
|
54
|
-
ask: async (req: Omit<PermissionNext.Request, "id" | "sessionID" | "tool">) => {
|
|
55
|
-
requests.push(req)
|
|
56
|
-
},
|
|
57
|
-
}
|
|
58
|
-
await bash.execute(
|
|
59
|
-
{
|
|
60
|
-
command: "echo hello",
|
|
61
|
-
description: "Echo hello",
|
|
62
|
-
},
|
|
63
|
-
testCtx,
|
|
64
|
-
)
|
|
65
|
-
expect(requests.length).toBe(1)
|
|
66
|
-
expect(requests[0].permission).toBe("bash")
|
|
67
|
-
expect(requests[0].patterns).toContain("echo hello")
|
|
68
|
-
},
|
|
69
|
-
})
|
|
70
|
-
})
|
|
71
|
-
|
|
72
|
-
test("asks for bash permission with multiple commands", async () => {
|
|
73
|
-
await using tmp = await tmpdir({ git: true })
|
|
74
|
-
await Instance.provide({
|
|
75
|
-
directory: tmp.path,
|
|
76
|
-
fn: async () => {
|
|
77
|
-
const bash = await BashTool.init()
|
|
78
|
-
const requests: Array<Omit<PermissionNext.Request, "id" | "sessionID" | "tool">> = []
|
|
79
|
-
const testCtx = {
|
|
80
|
-
...ctx,
|
|
81
|
-
ask: async (req: Omit<PermissionNext.Request, "id" | "sessionID" | "tool">) => {
|
|
82
|
-
requests.push(req)
|
|
83
|
-
},
|
|
84
|
-
}
|
|
85
|
-
await bash.execute(
|
|
86
|
-
{
|
|
87
|
-
command: "echo foo && echo bar",
|
|
88
|
-
description: "Echo twice",
|
|
89
|
-
},
|
|
90
|
-
testCtx,
|
|
91
|
-
)
|
|
92
|
-
expect(requests.length).toBe(1)
|
|
93
|
-
expect(requests[0].permission).toBe("bash")
|
|
94
|
-
expect(requests[0].patterns).toContain("echo foo")
|
|
95
|
-
expect(requests[0].patterns).toContain("echo bar")
|
|
96
|
-
},
|
|
97
|
-
})
|
|
98
|
-
})
|
|
99
|
-
|
|
100
|
-
test("asks for external_directory permission when cd to parent", async () => {
|
|
101
|
-
await using tmp = await tmpdir({ git: true })
|
|
102
|
-
await Instance.provide({
|
|
103
|
-
directory: tmp.path,
|
|
104
|
-
fn: async () => {
|
|
105
|
-
const bash = await BashTool.init()
|
|
106
|
-
const requests: Array<Omit<PermissionNext.Request, "id" | "sessionID" | "tool">> = []
|
|
107
|
-
const testCtx = {
|
|
108
|
-
...ctx,
|
|
109
|
-
ask: async (req: Omit<PermissionNext.Request, "id" | "sessionID" | "tool">) => {
|
|
110
|
-
requests.push(req)
|
|
111
|
-
},
|
|
112
|
-
}
|
|
113
|
-
await bash.execute(
|
|
114
|
-
{
|
|
115
|
-
command: "cd ../",
|
|
116
|
-
description: "Change to parent directory",
|
|
117
|
-
},
|
|
118
|
-
testCtx,
|
|
119
|
-
)
|
|
120
|
-
const extDirReq = requests.find((r) => r.permission === "external_directory")
|
|
121
|
-
expect(extDirReq).toBeDefined()
|
|
122
|
-
},
|
|
123
|
-
})
|
|
124
|
-
})
|
|
125
|
-
|
|
126
|
-
test("asks for external_directory permission when workdir is outside project", async () => {
|
|
127
|
-
await using tmp = await tmpdir({ git: true })
|
|
128
|
-
await Instance.provide({
|
|
129
|
-
directory: tmp.path,
|
|
130
|
-
fn: async () => {
|
|
131
|
-
const bash = await BashTool.init()
|
|
132
|
-
const requests: Array<Omit<PermissionNext.Request, "id" | "sessionID" | "tool">> = []
|
|
133
|
-
const testCtx = {
|
|
134
|
-
...ctx,
|
|
135
|
-
ask: async (req: Omit<PermissionNext.Request, "id" | "sessionID" | "tool">) => {
|
|
136
|
-
requests.push(req)
|
|
137
|
-
},
|
|
138
|
-
}
|
|
139
|
-
await bash.execute(
|
|
140
|
-
{
|
|
141
|
-
command: "ls",
|
|
142
|
-
workdir: os.tmpdir(),
|
|
143
|
-
description: "List temp dir",
|
|
144
|
-
},
|
|
145
|
-
testCtx,
|
|
146
|
-
)
|
|
147
|
-
const extDirReq = requests.find((r) => r.permission === "external_directory")
|
|
148
|
-
expect(extDirReq).toBeDefined()
|
|
149
|
-
expect(extDirReq!.patterns).toContain(path.join(os.tmpdir(), "*"))
|
|
150
|
-
},
|
|
151
|
-
})
|
|
152
|
-
})
|
|
153
|
-
|
|
154
|
-
test("asks for external_directory permission when file arg is outside project", async () => {
|
|
155
|
-
await using outerTmp = await tmpdir({
|
|
156
|
-
init: async (dir) => {
|
|
157
|
-
await Bun.write(path.join(dir, "outside.txt"), "x")
|
|
158
|
-
},
|
|
159
|
-
})
|
|
160
|
-
await using tmp = await tmpdir({ git: true })
|
|
161
|
-
await Instance.provide({
|
|
162
|
-
directory: tmp.path,
|
|
163
|
-
fn: async () => {
|
|
164
|
-
const bash = await BashTool.init()
|
|
165
|
-
const requests: Array<Omit<PermissionNext.Request, "id" | "sessionID" | "tool">> = []
|
|
166
|
-
const testCtx = {
|
|
167
|
-
...ctx,
|
|
168
|
-
ask: async (req: Omit<PermissionNext.Request, "id" | "sessionID" | "tool">) => {
|
|
169
|
-
requests.push(req)
|
|
170
|
-
},
|
|
171
|
-
}
|
|
172
|
-
const filepath = path.join(outerTmp.path, "outside.txt")
|
|
173
|
-
await bash.execute(
|
|
174
|
-
{
|
|
175
|
-
command: `cat ${filepath}`,
|
|
176
|
-
description: "Read external file",
|
|
177
|
-
},
|
|
178
|
-
testCtx,
|
|
179
|
-
)
|
|
180
|
-
const extDirReq = requests.find((r) => r.permission === "external_directory")
|
|
181
|
-
const expected = path.join(outerTmp.path, "*")
|
|
182
|
-
expect(extDirReq).toBeDefined()
|
|
183
|
-
expect(extDirReq!.patterns).toContain(expected)
|
|
184
|
-
expect(extDirReq!.always).toContain(expected)
|
|
185
|
-
},
|
|
186
|
-
})
|
|
187
|
-
})
|
|
188
|
-
|
|
189
|
-
test("does not ask for external_directory permission when rm inside project", async () => {
|
|
190
|
-
await using tmp = await tmpdir({ git: true })
|
|
191
|
-
await Instance.provide({
|
|
192
|
-
directory: tmp.path,
|
|
193
|
-
fn: async () => {
|
|
194
|
-
const bash = await BashTool.init()
|
|
195
|
-
const requests: Array<Omit<PermissionNext.Request, "id" | "sessionID" | "tool">> = []
|
|
196
|
-
const testCtx = {
|
|
197
|
-
...ctx,
|
|
198
|
-
ask: async (req: Omit<PermissionNext.Request, "id" | "sessionID" | "tool">) => {
|
|
199
|
-
requests.push(req)
|
|
200
|
-
},
|
|
201
|
-
}
|
|
202
|
-
|
|
203
|
-
await Bun.write(path.join(tmp.path, "tmpfile"), "x")
|
|
204
|
-
|
|
205
|
-
await bash.execute(
|
|
206
|
-
{
|
|
207
|
-
command: `rm -rf ${path.join(tmp.path, "nested")}`,
|
|
208
|
-
description: "remove nested dir",
|
|
209
|
-
},
|
|
210
|
-
testCtx,
|
|
211
|
-
)
|
|
212
|
-
|
|
213
|
-
const extDirReq = requests.find((r) => r.permission === "external_directory")
|
|
214
|
-
expect(extDirReq).toBeUndefined()
|
|
215
|
-
},
|
|
216
|
-
})
|
|
217
|
-
})
|
|
218
|
-
|
|
219
|
-
test("includes always patterns for auto-approval", async () => {
|
|
220
|
-
await using tmp = await tmpdir({ git: true })
|
|
221
|
-
await Instance.provide({
|
|
222
|
-
directory: tmp.path,
|
|
223
|
-
fn: async () => {
|
|
224
|
-
const bash = await BashTool.init()
|
|
225
|
-
const requests: Array<Omit<PermissionNext.Request, "id" | "sessionID" | "tool">> = []
|
|
226
|
-
const testCtx = {
|
|
227
|
-
...ctx,
|
|
228
|
-
ask: async (req: Omit<PermissionNext.Request, "id" | "sessionID" | "tool">) => {
|
|
229
|
-
requests.push(req)
|
|
230
|
-
},
|
|
231
|
-
}
|
|
232
|
-
await bash.execute(
|
|
233
|
-
{
|
|
234
|
-
command: "git log --oneline -5",
|
|
235
|
-
description: "Git log",
|
|
236
|
-
},
|
|
237
|
-
testCtx,
|
|
238
|
-
)
|
|
239
|
-
expect(requests.length).toBe(1)
|
|
240
|
-
expect(requests[0].always.length).toBeGreaterThan(0)
|
|
241
|
-
expect(requests[0].always.some((p) => p.endsWith("*"))).toBe(true)
|
|
242
|
-
},
|
|
243
|
-
})
|
|
244
|
-
})
|
|
245
|
-
|
|
246
|
-
test("does not ask for bash permission when command is cd only", async () => {
|
|
247
|
-
await using tmp = await tmpdir({ git: true })
|
|
248
|
-
await Instance.provide({
|
|
249
|
-
directory: tmp.path,
|
|
250
|
-
fn: async () => {
|
|
251
|
-
const bash = await BashTool.init()
|
|
252
|
-
const requests: Array<Omit<PermissionNext.Request, "id" | "sessionID" | "tool">> = []
|
|
253
|
-
const testCtx = {
|
|
254
|
-
...ctx,
|
|
255
|
-
ask: async (req: Omit<PermissionNext.Request, "id" | "sessionID" | "tool">) => {
|
|
256
|
-
requests.push(req)
|
|
257
|
-
},
|
|
258
|
-
}
|
|
259
|
-
await bash.execute(
|
|
260
|
-
{
|
|
261
|
-
command: "cd .",
|
|
262
|
-
description: "Stay in current directory",
|
|
263
|
-
},
|
|
264
|
-
testCtx,
|
|
265
|
-
)
|
|
266
|
-
const bashReq = requests.find((r) => r.permission === "bash")
|
|
267
|
-
expect(bashReq).toBeUndefined()
|
|
268
|
-
},
|
|
269
|
-
})
|
|
270
|
-
})
|
|
271
|
-
|
|
272
|
-
test("matches redirects in permission pattern", async () => {
|
|
273
|
-
await using tmp = await tmpdir({ git: true })
|
|
274
|
-
await Instance.provide({
|
|
275
|
-
directory: tmp.path,
|
|
276
|
-
fn: async () => {
|
|
277
|
-
const bash = await BashTool.init()
|
|
278
|
-
const requests: Array<Omit<PermissionNext.Request, "id" | "sessionID" | "tool">> = []
|
|
279
|
-
const testCtx = {
|
|
280
|
-
...ctx,
|
|
281
|
-
ask: async (req: Omit<PermissionNext.Request, "id" | "sessionID" | "tool">) => {
|
|
282
|
-
requests.push(req)
|
|
283
|
-
},
|
|
284
|
-
}
|
|
285
|
-
await bash.execute({ command: "cat > /tmp/output.txt", description: "Redirect ls output" }, testCtx)
|
|
286
|
-
const bashReq = requests.find((r) => r.permission === "bash")
|
|
287
|
-
expect(bashReq).toBeDefined()
|
|
288
|
-
expect(bashReq!.patterns).toContain("cat > /tmp/output.txt")
|
|
289
|
-
},
|
|
290
|
-
})
|
|
291
|
-
})
|
|
292
|
-
|
|
293
|
-
test("always pattern has space before wildcard to not include different commands", async () => {
|
|
294
|
-
await using tmp = await tmpdir({ git: true })
|
|
295
|
-
await Instance.provide({
|
|
296
|
-
directory: tmp.path,
|
|
297
|
-
fn: async () => {
|
|
298
|
-
const bash = await BashTool.init()
|
|
299
|
-
const requests: Array<Omit<PermissionNext.Request, "id" | "sessionID" | "tool">> = []
|
|
300
|
-
const testCtx = {
|
|
301
|
-
...ctx,
|
|
302
|
-
ask: async (req: Omit<PermissionNext.Request, "id" | "sessionID" | "tool">) => {
|
|
303
|
-
requests.push(req)
|
|
304
|
-
},
|
|
305
|
-
}
|
|
306
|
-
await bash.execute({ command: "ls -la", description: "List" }, testCtx)
|
|
307
|
-
const bashReq = requests.find((r) => r.permission === "bash")
|
|
308
|
-
expect(bashReq).toBeDefined()
|
|
309
|
-
const pattern = bashReq!.always[0]
|
|
310
|
-
expect(pattern).toBe("ls *")
|
|
311
|
-
},
|
|
312
|
-
})
|
|
313
|
-
})
|
|
314
|
-
})
|
|
315
|
-
|
|
316
|
-
describe("tool.bash truncation", () => {
|
|
317
|
-
test("truncates output exceeding line limit", async () => {
|
|
318
|
-
await Instance.provide({
|
|
319
|
-
directory: projectRoot,
|
|
320
|
-
fn: async () => {
|
|
321
|
-
const bash = await BashTool.init()
|
|
322
|
-
const lineCount = Truncate.MAX_LINES + 500
|
|
323
|
-
const result = await bash.execute(
|
|
324
|
-
{
|
|
325
|
-
command: `seq 1 ${lineCount}`,
|
|
326
|
-
description: "Generate lines exceeding limit",
|
|
327
|
-
},
|
|
328
|
-
ctx,
|
|
329
|
-
)
|
|
330
|
-
expect((result.metadata as any).truncated).toBe(true)
|
|
331
|
-
expect(result.output).toContain("truncated")
|
|
332
|
-
expect(result.output).toContain("The tool call succeeded but the output was truncated")
|
|
333
|
-
},
|
|
334
|
-
})
|
|
335
|
-
})
|
|
336
|
-
|
|
337
|
-
test("truncates output exceeding byte limit", async () => {
|
|
338
|
-
await Instance.provide({
|
|
339
|
-
directory: projectRoot,
|
|
340
|
-
fn: async () => {
|
|
341
|
-
const bash = await BashTool.init()
|
|
342
|
-
const byteCount = Truncate.MAX_BYTES + 10000
|
|
343
|
-
const result = await bash.execute(
|
|
344
|
-
{
|
|
345
|
-
command: `head -c ${byteCount} /dev/zero | tr '\\0' 'a'`,
|
|
346
|
-
description: "Generate bytes exceeding limit",
|
|
347
|
-
},
|
|
348
|
-
ctx,
|
|
349
|
-
)
|
|
350
|
-
expect((result.metadata as any).truncated).toBe(true)
|
|
351
|
-
expect(result.output).toContain("truncated")
|
|
352
|
-
expect(result.output).toContain("The tool call succeeded but the output was truncated")
|
|
353
|
-
},
|
|
354
|
-
})
|
|
355
|
-
})
|
|
356
|
-
|
|
357
|
-
test("does not truncate small output", async () => {
|
|
358
|
-
await Instance.provide({
|
|
359
|
-
directory: projectRoot,
|
|
360
|
-
fn: async () => {
|
|
361
|
-
const bash = await BashTool.init()
|
|
362
|
-
const result = await bash.execute(
|
|
363
|
-
{
|
|
364
|
-
command: "echo hello",
|
|
365
|
-
description: "Echo hello",
|
|
366
|
-
},
|
|
367
|
-
ctx,
|
|
368
|
-
)
|
|
369
|
-
expect((result.metadata as any).truncated).toBe(false)
|
|
370
|
-
const eol = process.platform === "win32" ? "\r\n" : "\n"
|
|
371
|
-
expect(result.output).toBe(`hello${eol}`)
|
|
372
|
-
},
|
|
373
|
-
})
|
|
374
|
-
})
|
|
375
|
-
|
|
376
|
-
test("full output is saved to file when truncated", async () => {
|
|
377
|
-
await Instance.provide({
|
|
378
|
-
directory: projectRoot,
|
|
379
|
-
fn: async () => {
|
|
380
|
-
const bash = await BashTool.init()
|
|
381
|
-
const lineCount = Truncate.MAX_LINES + 100
|
|
382
|
-
const result = await bash.execute(
|
|
383
|
-
{
|
|
384
|
-
command: `seq 1 ${lineCount}`,
|
|
385
|
-
description: "Generate lines for file check",
|
|
386
|
-
},
|
|
387
|
-
ctx,
|
|
388
|
-
)
|
|
389
|
-
expect((result.metadata as any).truncated).toBe(true)
|
|
390
|
-
|
|
391
|
-
const filepath = (result.metadata as any).outputPath
|
|
392
|
-
expect(filepath).toBeTruthy()
|
|
393
|
-
|
|
394
|
-
const saved = await Filesystem.readText(filepath)
|
|
395
|
-
const lines = saved.trim().split("\n")
|
|
396
|
-
expect(lines.length).toBe(lineCount)
|
|
397
|
-
expect(lines[0]).toBe("1")
|
|
398
|
-
expect(lines[lineCount - 1]).toBe(String(lineCount))
|
|
399
|
-
},
|
|
400
|
-
})
|
|
401
|
-
})
|
|
402
|
-
})
|