@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,249 +0,0 @@
1
- import { test, expect, mock, beforeEach } from "bun:test"
2
- import { EventEmitter } from "events"
3
-
4
- // Track open() calls and control failure behavior
5
- let openShouldFail = false
6
- let openCalledWith: string | undefined
7
-
8
- mock.module("open", () => ({
9
- default: async (url: string) => {
10
- openCalledWith = url
11
-
12
- // Return a mock subprocess that emits an error if openShouldFail is true
13
- const subprocess = new EventEmitter()
14
- if (openShouldFail) {
15
- // Emit error asynchronously like a real subprocess would
16
- setTimeout(() => {
17
- subprocess.emit("error", new Error("spawn xdg-open ENOENT"))
18
- }, 10)
19
- }
20
- return subprocess
21
- },
22
- }))
23
-
24
- // Mock UnauthorizedError
25
- class MockUnauthorizedError extends Error {
26
- constructor() {
27
- super("Unauthorized")
28
- this.name = "UnauthorizedError"
29
- }
30
- }
31
-
32
- // Track what options were passed to each transport constructor
33
- const transportCalls: Array<{
34
- type: "streamable" | "sse"
35
- url: string
36
- options: { authProvider?: unknown }
37
- }> = []
38
-
39
- // Mock the transport constructors
40
- mock.module("@modelcontextprotocol/sdk/client/streamableHttp.js", () => ({
41
- StreamableHTTPClientTransport: class MockStreamableHTTP {
42
- url: string
43
- authProvider: { redirectToAuthorization?: (url: URL) => Promise<void> } | undefined
44
- constructor(url: URL, options?: { authProvider?: { redirectToAuthorization?: (url: URL) => Promise<void> } }) {
45
- this.url = url.toString()
46
- this.authProvider = options?.authProvider
47
- transportCalls.push({
48
- type: "streamable",
49
- url: url.toString(),
50
- options: options ?? {},
51
- })
52
- }
53
- async start() {
54
- // Simulate OAuth redirect by calling the authProvider's redirectToAuthorization
55
- if (this.authProvider?.redirectToAuthorization) {
56
- await this.authProvider.redirectToAuthorization(new URL("https://auth.example.com/authorize?client_id=test"))
57
- }
58
- throw new MockUnauthorizedError()
59
- }
60
- async finishAuth(_code: string) {
61
- // Mock successful auth completion
62
- }
63
- },
64
- }))
65
-
66
- mock.module("@modelcontextprotocol/sdk/client/sse.js", () => ({
67
- SSEClientTransport: class MockSSE {
68
- constructor(url: URL) {
69
- transportCalls.push({
70
- type: "sse",
71
- url: url.toString(),
72
- options: {},
73
- })
74
- }
75
- async start() {
76
- throw new Error("Mock SSE transport cannot connect")
77
- }
78
- },
79
- }))
80
-
81
- // Mock the MCP SDK Client to trigger OAuth flow
82
- mock.module("@modelcontextprotocol/sdk/client/index.js", () => ({
83
- Client: class MockClient {
84
- async connect(transport: { start: () => Promise<void> }) {
85
- await transport.start()
86
- }
87
- },
88
- }))
89
-
90
- // Mock UnauthorizedError in the auth module
91
- mock.module("@modelcontextprotocol/sdk/client/auth.js", () => ({
92
- UnauthorizedError: MockUnauthorizedError,
93
- }))
94
-
95
- beforeEach(() => {
96
- openShouldFail = false
97
- openCalledWith = undefined
98
- transportCalls.length = 0
99
- })
100
-
101
- // Import modules after mocking
102
- const { MCP } = await import("../../src/mcp/index")
103
- const { Bus } = await import("../../src/bus")
104
- const { McpOAuthCallback } = await import("../../src/mcp/oauth-callback")
105
- const { Instance } = await import("../../src/project/instance")
106
- const { tmpdir } = await import("../fixture/fixture")
107
-
108
- test("BrowserOpenFailed event is published when open() throws", async () => {
109
- await using tmp = await tmpdir({
110
- init: async (dir) => {
111
- await Bun.write(
112
- `${dir}/opencode.json`,
113
- JSON.stringify({
114
- $schema: "https://opencode.ai/config.json",
115
- mcp: {
116
- "test-oauth-server": {
117
- type: "remote",
118
- url: "https://example.com/mcp",
119
- },
120
- },
121
- }),
122
- )
123
- },
124
- })
125
-
126
- await Instance.provide({
127
- directory: tmp.path,
128
- fn: async () => {
129
- openShouldFail = true
130
-
131
- const events: Array<{ mcpName: string; url: string }> = []
132
- const unsubscribe = Bus.subscribe(MCP.BrowserOpenFailed, (evt) => {
133
- events.push(evt.properties)
134
- })
135
-
136
- // Run authenticate with a timeout to avoid waiting forever for the callback
137
- // Attach a handler immediately so callback shutdown rejections
138
- // don't show up as unhandled between tests.
139
- const authPromise = MCP.authenticate("test-oauth-server").catch(() => undefined)
140
-
141
- // Config.get() can be slow in tests, so give it plenty of time.
142
- await new Promise((resolve) => setTimeout(resolve, 2_000))
143
-
144
- // Stop the callback server and cancel any pending auth
145
- await McpOAuthCallback.stop()
146
-
147
- await authPromise
148
-
149
- unsubscribe()
150
-
151
- // Verify the BrowserOpenFailed event was published
152
- expect(events.length).toBe(1)
153
- expect(events[0].mcpName).toBe("test-oauth-server")
154
- expect(events[0].url).toContain("https://")
155
- },
156
- })
157
- })
158
-
159
- test("BrowserOpenFailed event is NOT published when open() succeeds", async () => {
160
- await using tmp = await tmpdir({
161
- init: async (dir) => {
162
- await Bun.write(
163
- `${dir}/opencode.json`,
164
- JSON.stringify({
165
- $schema: "https://opencode.ai/config.json",
166
- mcp: {
167
- "test-oauth-server-2": {
168
- type: "remote",
169
- url: "https://example.com/mcp",
170
- },
171
- },
172
- }),
173
- )
174
- },
175
- })
176
-
177
- await Instance.provide({
178
- directory: tmp.path,
179
- fn: async () => {
180
- openShouldFail = false
181
-
182
- const events: Array<{ mcpName: string; url: string }> = []
183
- const unsubscribe = Bus.subscribe(MCP.BrowserOpenFailed, (evt) => {
184
- events.push(evt.properties)
185
- })
186
-
187
- // Run authenticate with a timeout to avoid waiting forever for the callback
188
- const authPromise = MCP.authenticate("test-oauth-server-2").catch(() => undefined)
189
-
190
- // Config.get() can be slow in tests; also covers the ~500ms open() error-detection window.
191
- await new Promise((resolve) => setTimeout(resolve, 2_000))
192
-
193
- // Stop the callback server and cancel any pending auth
194
- await McpOAuthCallback.stop()
195
-
196
- await authPromise
197
-
198
- unsubscribe()
199
-
200
- // Verify NO BrowserOpenFailed event was published
201
- expect(events.length).toBe(0)
202
- // Verify open() was still called
203
- expect(openCalledWith).toBeDefined()
204
- },
205
- })
206
- })
207
-
208
- test("open() is called with the authorization URL", async () => {
209
- await using tmp = await tmpdir({
210
- init: async (dir) => {
211
- await Bun.write(
212
- `${dir}/opencode.json`,
213
- JSON.stringify({
214
- $schema: "https://opencode.ai/config.json",
215
- mcp: {
216
- "test-oauth-server-3": {
217
- type: "remote",
218
- url: "https://example.com/mcp",
219
- },
220
- },
221
- }),
222
- )
223
- },
224
- })
225
-
226
- await Instance.provide({
227
- directory: tmp.path,
228
- fn: async () => {
229
- openShouldFail = false
230
- openCalledWith = undefined
231
-
232
- // Run authenticate with a timeout to avoid waiting forever for the callback
233
- const authPromise = MCP.authenticate("test-oauth-server-3").catch(() => undefined)
234
-
235
- // Config.get() can be slow in tests; also covers the ~500ms open() error-detection window.
236
- await new Promise((resolve) => setTimeout(resolve, 2_000))
237
-
238
- // Stop the callback server and cancel any pending auth
239
- await McpOAuthCallback.stop()
240
-
241
- await authPromise
242
-
243
- // Verify open was called with a URL
244
- expect(openCalledWith).toBeDefined()
245
- expect(typeof openCalledWith).toBe("string")
246
- expect(openCalledWith!).toContain("https://")
247
- },
248
- })
249
- })
@@ -1,136 +0,0 @@
1
- import { describe, test, expect } from "bun:test"
2
- import path from "path"
3
- import { Instance } from "../../src/project/instance"
4
- import { WebFetchTool } from "../../src/tool/webfetch"
5
-
6
- const projectRoot = path.join(__dirname, "../..")
7
-
8
- const ctx = {
9
- sessionID: "test",
10
- messageID: "",
11
- callID: "",
12
- agent: "build",
13
- abort: new AbortController().signal,
14
- messages: [],
15
- metadata: () => {},
16
- ask: async () => {},
17
- }
18
-
19
- const MB = 1024 * 1024
20
- const ITERATIONS = 50
21
-
22
- const getHeapMB = () => {
23
- Bun.gc(true)
24
- return process.memoryUsage().heapUsed / MB
25
- }
26
-
27
- describe("memory: abort controller leak", () => {
28
- test("webfetch does not leak memory over many invocations", async () => {
29
- await Instance.provide({
30
- directory: projectRoot,
31
- fn: async () => {
32
- const tool = await WebFetchTool.init()
33
-
34
- // Warm up
35
- await tool.execute({ url: "https://example.com", format: "text" }, ctx).catch(() => {})
36
-
37
- Bun.gc(true)
38
- const baseline = getHeapMB()
39
-
40
- // Run many fetches
41
- for (let i = 0; i < ITERATIONS; i++) {
42
- await tool.execute({ url: "https://example.com", format: "text" }, ctx).catch(() => {})
43
- }
44
-
45
- Bun.gc(true)
46
- const after = getHeapMB()
47
- const growth = after - baseline
48
-
49
- console.log(`Baseline: ${baseline.toFixed(2)} MB`)
50
- console.log(`After ${ITERATIONS} fetches: ${after.toFixed(2)} MB`)
51
- console.log(`Growth: ${growth.toFixed(2)} MB`)
52
-
53
- // Memory growth should be minimal - less than 1MB per 10 requests
54
- // With the old closure pattern, this would grow ~0.5MB per request
55
- expect(growth).toBeLessThan(ITERATIONS / 10)
56
- },
57
- })
58
- }, 60000)
59
-
60
- test("compare closure vs bind pattern directly", async () => {
61
- const ITERATIONS = 500
62
-
63
- // Test OLD pattern: arrow function closure
64
- // Store closures in a map keyed by content to force retention
65
- const closureMap = new Map<string, () => void>()
66
- const timers: Timer[] = []
67
- const controllers: AbortController[] = []
68
-
69
- Bun.gc(true)
70
- Bun.sleepSync(100)
71
- const baseline = getHeapMB()
72
-
73
- for (let i = 0; i < ITERATIONS; i++) {
74
- // Simulate large response body like webfetch would have
75
- const content = `${i}:${"x".repeat(50 * 1024)}` // 50KB unique per iteration
76
- const controller = new AbortController()
77
- controllers.push(controller)
78
-
79
- // OLD pattern - closure captures `content`
80
- const handler = () => {
81
- // Actually use content so it can't be optimized away
82
- if (content.length > 1000000000) controller.abort()
83
- }
84
- closureMap.set(content, handler)
85
- const timeoutId = setTimeout(handler, 30000)
86
- timers.push(timeoutId)
87
- }
88
-
89
- Bun.gc(true)
90
- Bun.sleepSync(100)
91
- const after = getHeapMB()
92
- const oldGrowth = after - baseline
93
-
94
- console.log(`OLD pattern (closure): ${oldGrowth.toFixed(2)} MB growth (${closureMap.size} closures)`)
95
-
96
- // Cleanup after measuring
97
- timers.forEach(clearTimeout)
98
- controllers.forEach((c) => c.abort())
99
- closureMap.clear()
100
-
101
- // Test NEW pattern: bind
102
- Bun.gc(true)
103
- Bun.sleepSync(100)
104
- const baseline2 = getHeapMB()
105
- const handlers2: (() => void)[] = []
106
- const timers2: Timer[] = []
107
- const controllers2: AbortController[] = []
108
-
109
- for (let i = 0; i < ITERATIONS; i++) {
110
- const _content = `${i}:${"x".repeat(50 * 1024)}` // 50KB - won't be captured
111
- const controller = new AbortController()
112
- controllers2.push(controller)
113
-
114
- // NEW pattern - bind doesn't capture surrounding scope
115
- const handler = controller.abort.bind(controller)
116
- handlers2.push(handler)
117
- const timeoutId = setTimeout(handler, 30000)
118
- timers2.push(timeoutId)
119
- }
120
-
121
- Bun.gc(true)
122
- Bun.sleepSync(100)
123
- const after2 = getHeapMB()
124
- const newGrowth = after2 - baseline2
125
-
126
- // Cleanup after measuring
127
- timers2.forEach(clearTimeout)
128
- controllers2.forEach((c) => c.abort())
129
- handlers2.length = 0
130
-
131
- console.log(`NEW pattern (bind): ${newGrowth.toFixed(2)} MB growth`)
132
- console.log(`Improvement: ${(oldGrowth - newGrowth).toFixed(2)} MB saved`)
133
-
134
- expect(newGrowth).toBeLessThanOrEqual(oldGrowth)
135
- })
136
- })