@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,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
- })