@stonerzju/opencode 1.2.16-offline.1 → 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 (262) 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/package.json.bak +0 -140
  155. package/script/build.ts +0 -224
  156. package/script/check-migrations.ts +0 -16
  157. package/script/postinstall.mjs +0 -131
  158. package/script/publish.ts +0 -181
  159. package/script/schema.ts +0 -63
  160. package/script/seed-e2e.ts +0 -50
  161. package/sst-env.d.ts +0 -10
  162. package/test/AGENTS.md +0 -81
  163. package/test/acp/agent-interface.test.ts +0 -51
  164. package/test/acp/event-subscription.test.ts +0 -683
  165. package/test/agent/agent.test.ts +0 -689
  166. package/test/bun.test.ts +0 -53
  167. package/test/cli/github-action.test.ts +0 -197
  168. package/test/cli/github-remote.test.ts +0 -80
  169. package/test/cli/import.test.ts +0 -38
  170. package/test/cli/plugin-auth-picker.test.ts +0 -120
  171. package/test/cli/tui/transcript.test.ts +0 -322
  172. package/test/config/agent-color.test.ts +0 -71
  173. package/test/config/config.test.ts +0 -1886
  174. package/test/config/fixtures/empty-frontmatter.md +0 -4
  175. package/test/config/fixtures/frontmatter.md +0 -28
  176. package/test/config/fixtures/markdown-header.md +0 -11
  177. package/test/config/fixtures/no-frontmatter.md +0 -1
  178. package/test/config/fixtures/weird-model-id.md +0 -13
  179. package/test/config/markdown.test.ts +0 -228
  180. package/test/config/tui.test.ts +0 -510
  181. package/test/control-plane/session-proxy-middleware.test.ts +0 -147
  182. package/test/control-plane/sse.test.ts +0 -56
  183. package/test/control-plane/workspace-server-sse.test.ts +0 -65
  184. package/test/control-plane/workspace-sync.test.ts +0 -97
  185. package/test/file/ignore.test.ts +0 -10
  186. package/test/file/index.test.ts +0 -394
  187. package/test/file/path-traversal.test.ts +0 -198
  188. package/test/file/ripgrep.test.ts +0 -39
  189. package/test/file/time.test.ts +0 -361
  190. package/test/fixture/db.ts +0 -11
  191. package/test/fixture/fixture.ts +0 -45
  192. package/test/fixture/lsp/fake-lsp-server.js +0 -77
  193. package/test/fixture/skills/agents-sdk/SKILL.md +0 -152
  194. package/test/fixture/skills/agents-sdk/references/callable.md +0 -92
  195. package/test/fixture/skills/cloudflare/SKILL.md +0 -211
  196. package/test/fixture/skills/index.json +0 -6
  197. package/test/ide/ide.test.ts +0 -82
  198. package/test/keybind.test.ts +0 -421
  199. package/test/lsp/client.test.ts +0 -95
  200. package/test/mcp/headers.test.ts +0 -153
  201. package/test/mcp/oauth-browser.test.ts +0 -249
  202. package/test/memory/abort-leak.test.ts +0 -136
  203. package/test/patch/patch.test.ts +0 -348
  204. package/test/permission/arity.test.ts +0 -33
  205. package/test/permission/next.test.ts +0 -689
  206. package/test/permission-task.test.ts +0 -319
  207. package/test/plugin/auth-override.test.ts +0 -44
  208. package/test/plugin/codex.test.ts +0 -123
  209. package/test/preload.ts +0 -80
  210. package/test/project/project.test.ts +0 -348
  211. package/test/project/worktree-remove.test.ts +0 -65
  212. package/test/provider/amazon-bedrock.test.ts +0 -446
  213. package/test/provider/copilot/convert-to-copilot-messages.test.ts +0 -523
  214. package/test/provider/copilot/copilot-chat-model.test.ts +0 -592
  215. package/test/provider/gitlab-duo.test.ts +0 -262
  216. package/test/provider/provider.test.ts +0 -2220
  217. package/test/provider/transform.test.ts +0 -2353
  218. package/test/pty/pty-output-isolation.test.ts +0 -140
  219. package/test/question/question.test.ts +0 -300
  220. package/test/scheduler.test.ts +0 -73
  221. package/test/server/global-session-list.test.ts +0 -89
  222. package/test/server/session-list.test.ts +0 -90
  223. package/test/server/session-select.test.ts +0 -78
  224. package/test/session/compaction.test.ts +0 -423
  225. package/test/session/instruction.test.ts +0 -170
  226. package/test/session/llm.test.ts +0 -667
  227. package/test/session/message-v2.test.ts +0 -924
  228. package/test/session/prompt.test.ts +0 -211
  229. package/test/session/retry.test.ts +0 -188
  230. package/test/session/revert-compact.test.ts +0 -285
  231. package/test/session/session.test.ts +0 -71
  232. package/test/session/structured-output-integration.test.ts +0 -233
  233. package/test/session/structured-output.test.ts +0 -385
  234. package/test/skill/discovery.test.ts +0 -110
  235. package/test/skill/skill.test.ts +0 -388
  236. package/test/snapshot/snapshot.test.ts +0 -1180
  237. package/test/storage/json-migration.test.ts +0 -846
  238. package/test/tool/__snapshots__/tool.test.ts.snap +0 -9
  239. package/test/tool/apply_patch.test.ts +0 -566
  240. package/test/tool/bash.test.ts +0 -402
  241. package/test/tool/edit.test.ts +0 -496
  242. package/test/tool/external-directory.test.ts +0 -127
  243. package/test/tool/fixtures/large-image.png +0 -0
  244. package/test/tool/fixtures/models-api.json +0 -38413
  245. package/test/tool/grep.test.ts +0 -110
  246. package/test/tool/question.test.ts +0 -107
  247. package/test/tool/read.test.ts +0 -504
  248. package/test/tool/registry.test.ts +0 -122
  249. package/test/tool/skill.test.ts +0 -112
  250. package/test/tool/truncation.test.ts +0 -160
  251. package/test/tool/webfetch.test.ts +0 -100
  252. package/test/tool/write.test.ts +0 -348
  253. package/test/util/filesystem.test.ts +0 -443
  254. package/test/util/format.test.ts +0 -59
  255. package/test/util/glob.test.ts +0 -164
  256. package/test/util/iife.test.ts +0 -36
  257. package/test/util/lazy.test.ts +0 -50
  258. package/test/util/lock.test.ts +0 -72
  259. package/test/util/process.test.ts +0 -59
  260. package/test/util/timeout.test.ts +0 -21
  261. package/test/util/wildcard.test.ts +0 -90
  262. 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
- })