@link-assistant/agent 0.0.8

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 (133) hide show
  1. package/EXAMPLES.md +383 -0
  2. package/LICENSE +24 -0
  3. package/MODELS.md +95 -0
  4. package/README.md +388 -0
  5. package/TOOLS.md +134 -0
  6. package/package.json +89 -0
  7. package/src/agent/agent.ts +150 -0
  8. package/src/agent/generate.txt +75 -0
  9. package/src/auth/index.ts +64 -0
  10. package/src/bun/index.ts +96 -0
  11. package/src/bus/global.ts +10 -0
  12. package/src/bus/index.ts +119 -0
  13. package/src/cli/bootstrap.js +41 -0
  14. package/src/cli/bootstrap.ts +17 -0
  15. package/src/cli/cmd/agent.ts +165 -0
  16. package/src/cli/cmd/cmd.ts +5 -0
  17. package/src/cli/cmd/export.ts +88 -0
  18. package/src/cli/cmd/mcp.ts +80 -0
  19. package/src/cli/cmd/models.ts +58 -0
  20. package/src/cli/cmd/run.ts +359 -0
  21. package/src/cli/cmd/stats.ts +276 -0
  22. package/src/cli/error.ts +27 -0
  23. package/src/command/index.ts +73 -0
  24. package/src/command/template/initialize.txt +10 -0
  25. package/src/config/config.ts +705 -0
  26. package/src/config/markdown.ts +41 -0
  27. package/src/file/ripgrep.ts +391 -0
  28. package/src/file/time.ts +38 -0
  29. package/src/file/watcher.ts +75 -0
  30. package/src/file.ts +6 -0
  31. package/src/flag/flag.ts +19 -0
  32. package/src/format/formatter.ts +248 -0
  33. package/src/format/index.ts +137 -0
  34. package/src/global/index.ts +52 -0
  35. package/src/id/id.ts +72 -0
  36. package/src/index.js +371 -0
  37. package/src/mcp/index.ts +289 -0
  38. package/src/patch/index.ts +622 -0
  39. package/src/project/bootstrap.ts +22 -0
  40. package/src/project/instance.ts +67 -0
  41. package/src/project/project.ts +105 -0
  42. package/src/project/state.ts +65 -0
  43. package/src/provider/models-macro.ts +11 -0
  44. package/src/provider/models.ts +98 -0
  45. package/src/provider/opencode.js +47 -0
  46. package/src/provider/provider.ts +636 -0
  47. package/src/provider/transform.ts +241 -0
  48. package/src/server/project.ts +48 -0
  49. package/src/server/server.ts +249 -0
  50. package/src/session/agent.js +204 -0
  51. package/src/session/compaction.ts +249 -0
  52. package/src/session/index.ts +380 -0
  53. package/src/session/message-v2.ts +758 -0
  54. package/src/session/message.ts +189 -0
  55. package/src/session/processor.ts +356 -0
  56. package/src/session/prompt/anthropic-20250930.txt +166 -0
  57. package/src/session/prompt/anthropic.txt +105 -0
  58. package/src/session/prompt/anthropic_spoof.txt +1 -0
  59. package/src/session/prompt/beast.txt +147 -0
  60. package/src/session/prompt/build-switch.txt +5 -0
  61. package/src/session/prompt/codex.txt +318 -0
  62. package/src/session/prompt/copilot-gpt-5.txt +143 -0
  63. package/src/session/prompt/gemini.txt +155 -0
  64. package/src/session/prompt/grok-code.txt +1 -0
  65. package/src/session/prompt/plan.txt +8 -0
  66. package/src/session/prompt/polaris.txt +107 -0
  67. package/src/session/prompt/qwen.txt +109 -0
  68. package/src/session/prompt/summarize-turn.txt +5 -0
  69. package/src/session/prompt/summarize.txt +10 -0
  70. package/src/session/prompt/title.txt +25 -0
  71. package/src/session/prompt.ts +1390 -0
  72. package/src/session/retry.ts +53 -0
  73. package/src/session/revert.ts +108 -0
  74. package/src/session/status.ts +75 -0
  75. package/src/session/summary.ts +179 -0
  76. package/src/session/system.ts +138 -0
  77. package/src/session/todo.ts +36 -0
  78. package/src/snapshot/index.ts +197 -0
  79. package/src/storage/storage.ts +226 -0
  80. package/src/tool/bash.ts +193 -0
  81. package/src/tool/bash.txt +121 -0
  82. package/src/tool/batch.ts +173 -0
  83. package/src/tool/batch.txt +28 -0
  84. package/src/tool/codesearch.ts +123 -0
  85. package/src/tool/codesearch.txt +12 -0
  86. package/src/tool/edit.ts +604 -0
  87. package/src/tool/edit.txt +10 -0
  88. package/src/tool/glob.ts +65 -0
  89. package/src/tool/glob.txt +6 -0
  90. package/src/tool/grep.ts +116 -0
  91. package/src/tool/grep.txt +8 -0
  92. package/src/tool/invalid.ts +17 -0
  93. package/src/tool/ls.ts +110 -0
  94. package/src/tool/ls.txt +1 -0
  95. package/src/tool/multiedit.ts +46 -0
  96. package/src/tool/multiedit.txt +41 -0
  97. package/src/tool/patch.ts +188 -0
  98. package/src/tool/patch.txt +1 -0
  99. package/src/tool/read.ts +201 -0
  100. package/src/tool/read.txt +12 -0
  101. package/src/tool/registry.ts +87 -0
  102. package/src/tool/task.ts +126 -0
  103. package/src/tool/task.txt +60 -0
  104. package/src/tool/todo.ts +39 -0
  105. package/src/tool/todoread.txt +14 -0
  106. package/src/tool/todowrite.txt +167 -0
  107. package/src/tool/tool.ts +66 -0
  108. package/src/tool/webfetch.ts +171 -0
  109. package/src/tool/webfetch.txt +14 -0
  110. package/src/tool/websearch.ts +133 -0
  111. package/src/tool/websearch.txt +11 -0
  112. package/src/tool/write.ts +33 -0
  113. package/src/tool/write.txt +8 -0
  114. package/src/util/binary.ts +41 -0
  115. package/src/util/context.ts +25 -0
  116. package/src/util/defer.ts +12 -0
  117. package/src/util/error.ts +54 -0
  118. package/src/util/eventloop.ts +20 -0
  119. package/src/util/filesystem.ts +69 -0
  120. package/src/util/fn.ts +11 -0
  121. package/src/util/iife.ts +3 -0
  122. package/src/util/keybind.ts +79 -0
  123. package/src/util/lazy.ts +11 -0
  124. package/src/util/locale.ts +39 -0
  125. package/src/util/lock.ts +98 -0
  126. package/src/util/log.ts +177 -0
  127. package/src/util/queue.ts +19 -0
  128. package/src/util/rpc.ts +42 -0
  129. package/src/util/scrap.ts +10 -0
  130. package/src/util/signal.ts +12 -0
  131. package/src/util/timeout.ts +14 -0
  132. package/src/util/token.ts +7 -0
  133. package/src/util/wildcard.ts +54 -0
@@ -0,0 +1,197 @@
1
+ import { $ } from "bun"
2
+ import path from "path"
3
+ import fs from "fs/promises"
4
+ import { Log } from "../util/log"
5
+ import { Global } from "../global"
6
+ import z from "zod"
7
+ import { Config } from "../config/config"
8
+ import { Instance } from "../project/instance"
9
+
10
+ export namespace Snapshot {
11
+ const log = Log.create({ service: "snapshot" })
12
+
13
+ export async function track() {
14
+ if (Instance.project.vcs !== "git") return
15
+ const cfg = await Config.get()
16
+ if (cfg.snapshot === false) return
17
+ const git = gitdir()
18
+ if (await fs.mkdir(git, { recursive: true })) {
19
+ await $`git init`
20
+ .env({
21
+ ...process.env,
22
+ GIT_DIR: git,
23
+ GIT_WORK_TREE: Instance.worktree,
24
+ })
25
+ .quiet()
26
+ .nothrow()
27
+ // Configure git to not convert line endings on Windows
28
+ await $`git --git-dir ${git} config core.autocrlf false`.quiet().nothrow()
29
+ log.info("initialized")
30
+ }
31
+ await $`git --git-dir ${git} --work-tree ${Instance.worktree} add .`.quiet().cwd(Instance.directory).nothrow()
32
+ const hash = await $`git --git-dir ${git} --work-tree ${Instance.worktree} write-tree`
33
+ .quiet()
34
+ .cwd(Instance.directory)
35
+ .nothrow()
36
+ .text()
37
+ log.info("tracking", { hash, cwd: Instance.directory, git })
38
+ return hash.trim()
39
+ }
40
+
41
+ export const Patch = z.object({
42
+ hash: z.string(),
43
+ files: z.string().array(),
44
+ })
45
+ export type Patch = z.infer<typeof Patch>
46
+
47
+ export async function patch(hash: string): Promise<Patch> {
48
+ const git = gitdir()
49
+ await $`git --git-dir ${git} --work-tree ${Instance.worktree} add .`.quiet().cwd(Instance.directory).nothrow()
50
+ const result =
51
+ await $`git -c core.autocrlf=false --git-dir ${git} --work-tree ${Instance.worktree} diff --name-only ${hash} -- .`
52
+ .quiet()
53
+ .cwd(Instance.directory)
54
+ .nothrow()
55
+
56
+ // If git diff fails, return empty patch
57
+ if (result.exitCode !== 0) {
58
+ log.warn("failed to get diff", { hash, exitCode: result.exitCode })
59
+ return { hash, files: [] }
60
+ }
61
+
62
+ const files = result.text()
63
+ return {
64
+ hash,
65
+ files: files
66
+ .trim()
67
+ .split("\n")
68
+ .map((x) => x.trim())
69
+ .filter(Boolean)
70
+ .map((x) => path.join(Instance.worktree, x)),
71
+ }
72
+ }
73
+
74
+ export async function restore(snapshot: string) {
75
+ log.info("restore", { commit: snapshot })
76
+ const git = gitdir()
77
+ const result =
78
+ await $`git --git-dir ${git} --work-tree ${Instance.worktree} read-tree ${snapshot} && git --git-dir ${git} --work-tree ${Instance.worktree} checkout-index -a -f`
79
+ .quiet()
80
+ .cwd(Instance.worktree)
81
+ .nothrow()
82
+
83
+ if (result.exitCode !== 0) {
84
+ log.error("failed to restore snapshot", {
85
+ snapshot,
86
+ exitCode: result.exitCode,
87
+ stderr: result.stderr.toString(),
88
+ stdout: result.stdout.toString(),
89
+ })
90
+ }
91
+ }
92
+
93
+ export async function revert(patches: Patch[]) {
94
+ const files = new Set<string>()
95
+ const git = gitdir()
96
+ for (const item of patches) {
97
+ for (const file of item.files) {
98
+ if (files.has(file)) continue
99
+ log.info("reverting", { file, hash: item.hash })
100
+ const result = await $`git --git-dir ${git} --work-tree ${Instance.worktree} checkout ${item.hash} -- ${file}`
101
+ .quiet()
102
+ .cwd(Instance.worktree)
103
+ .nothrow()
104
+ if (result.exitCode !== 0) {
105
+ const relativePath = path.relative(Instance.worktree, file)
106
+ const checkTree =
107
+ await $`git --git-dir ${git} --work-tree ${Instance.worktree} ls-tree ${item.hash} -- ${relativePath}`
108
+ .quiet()
109
+ .cwd(Instance.worktree)
110
+ .nothrow()
111
+ if (checkTree.exitCode === 0 && checkTree.text().trim()) {
112
+ log.info("file existed in snapshot but checkout failed, keeping", {
113
+ file,
114
+ })
115
+ } else {
116
+ log.info("file did not exist in snapshot, deleting", { file })
117
+ await fs.unlink(file).catch(() => {})
118
+ }
119
+ }
120
+ files.add(file)
121
+ }
122
+ }
123
+ }
124
+
125
+ export async function diff(hash: string) {
126
+ const git = gitdir()
127
+ await $`git --git-dir ${git} --work-tree ${Instance.worktree} add .`.quiet().cwd(Instance.directory).nothrow()
128
+ const result =
129
+ await $`git -c core.autocrlf=false --git-dir ${git} --work-tree ${Instance.worktree} diff ${hash} -- .`
130
+ .quiet()
131
+ .cwd(Instance.worktree)
132
+ .nothrow()
133
+
134
+ if (result.exitCode !== 0) {
135
+ log.warn("failed to get diff", {
136
+ hash,
137
+ exitCode: result.exitCode,
138
+ stderr: result.stderr.toString(),
139
+ stdout: result.stdout.toString(),
140
+ })
141
+ return ""
142
+ }
143
+
144
+ return result.text().trim()
145
+ }
146
+
147
+ export const FileDiff = z
148
+ .object({
149
+ file: z.string(),
150
+ before: z.string(),
151
+ after: z.string(),
152
+ additions: z.number(),
153
+ deletions: z.number(),
154
+ })
155
+ .meta({
156
+ ref: "FileDiff",
157
+ })
158
+ export type FileDiff = z.infer<typeof FileDiff>
159
+ export async function diffFull(from: string, to: string): Promise<FileDiff[]> {
160
+ const git = gitdir()
161
+ const result: FileDiff[] = []
162
+ for await (const line of $`git -c core.autocrlf=false --git-dir ${git} --work-tree ${Instance.worktree} diff --no-renames --numstat ${from} ${to} -- .`
163
+ .quiet()
164
+ .cwd(Instance.directory)
165
+ .nothrow()
166
+ .lines()) {
167
+ if (!line) continue
168
+ const [additions, deletions, file] = line.split("\t")
169
+ const isBinaryFile = additions === "-" && deletions === "-"
170
+ const before = isBinaryFile
171
+ ? ""
172
+ : await $`git -c core.autocrlf=false --git-dir ${git} --work-tree ${Instance.worktree} show ${from}:${file}`
173
+ .quiet()
174
+ .nothrow()
175
+ .text()
176
+ const after = isBinaryFile
177
+ ? ""
178
+ : await $`git -c core.autocrlf=false --git-dir ${git} --work-tree ${Instance.worktree} show ${to}:${file}`
179
+ .quiet()
180
+ .nothrow()
181
+ .text()
182
+ result.push({
183
+ file,
184
+ before,
185
+ after,
186
+ additions: parseInt(additions),
187
+ deletions: parseInt(deletions),
188
+ })
189
+ }
190
+ return result
191
+ }
192
+
193
+ function gitdir() {
194
+ const project = Instance.project
195
+ return path.join(Global.Path.data, "snapshot", project.id)
196
+ }
197
+ }
@@ -0,0 +1,226 @@
1
+ import { Log } from "../util/log"
2
+ import path from "path"
3
+ import fs from "fs/promises"
4
+ import { Global } from "../global"
5
+ import { lazy } from "../util/lazy"
6
+ import { Lock } from "../util/lock"
7
+ import { $ } from "bun"
8
+ import { NamedError } from "../util/error"
9
+ import z from "zod"
10
+
11
+ export namespace Storage {
12
+ const log = Log.create({ service: "storage" })
13
+
14
+ type Migration = (dir: string) => Promise<void>
15
+
16
+ export const NotFoundError = NamedError.create(
17
+ "NotFoundError",
18
+ z.object({
19
+ message: z.string(),
20
+ }),
21
+ )
22
+
23
+ const MIGRATIONS: Migration[] = [
24
+ async (dir) => {
25
+ const project = path.resolve(dir, "../project")
26
+ if (!fs.exists(project)) return
27
+ for await (const projectDir of new Bun.Glob("*").scan({
28
+ cwd: project,
29
+ onlyFiles: false,
30
+ })) {
31
+ log.info(`migrating project ${projectDir}`)
32
+ let projectID = projectDir
33
+ const fullProjectDir = path.join(project, projectDir)
34
+ let worktree = "/"
35
+
36
+ if (projectID !== "global") {
37
+ for await (const msgFile of new Bun.Glob("storage/session/message/*/*.json").scan({
38
+ cwd: path.join(project, projectDir),
39
+ absolute: true,
40
+ })) {
41
+ const json = await Bun.file(msgFile).json()
42
+ worktree = json.path?.root
43
+ if (worktree) break
44
+ }
45
+ if (!worktree) continue
46
+ if (!(await fs.exists(worktree))) continue
47
+ const [id] = await $`git rev-list --max-parents=0 --all`
48
+ .quiet()
49
+ .nothrow()
50
+ .cwd(worktree)
51
+ .text()
52
+ .then((x) =>
53
+ x
54
+ .split("\n")
55
+ .filter(Boolean)
56
+ .map((x) => x.trim())
57
+ .toSorted(),
58
+ )
59
+ if (!id) continue
60
+ projectID = id
61
+
62
+ await Bun.write(
63
+ path.join(dir, "project", projectID + ".json"),
64
+ JSON.stringify({
65
+ id,
66
+ vcs: "git",
67
+ worktree,
68
+ time: {
69
+ created: Date.now(),
70
+ initialized: Date.now(),
71
+ },
72
+ }),
73
+ )
74
+
75
+ log.info(`migrating sessions for project ${projectID}`)
76
+ for await (const sessionFile of new Bun.Glob("storage/session/info/*.json").scan({
77
+ cwd: fullProjectDir,
78
+ absolute: true,
79
+ })) {
80
+ const dest = path.join(dir, "session", projectID, path.basename(sessionFile))
81
+ log.info("copying", {
82
+ sessionFile,
83
+ dest,
84
+ })
85
+ const session = await Bun.file(sessionFile).json()
86
+ await Bun.write(dest, JSON.stringify(session))
87
+ log.info(`migrating messages for session ${session.id}`)
88
+ for await (const msgFile of new Bun.Glob(`storage/session/message/${session.id}/*.json`).scan({
89
+ cwd: fullProjectDir,
90
+ absolute: true,
91
+ })) {
92
+ const dest = path.join(dir, "message", session.id, path.basename(msgFile))
93
+ log.info("copying", {
94
+ msgFile,
95
+ dest,
96
+ })
97
+ const message = await Bun.file(msgFile).json()
98
+ await Bun.write(dest, JSON.stringify(message))
99
+
100
+ log.info(`migrating parts for message ${message.id}`)
101
+ for await (const partFile of new Bun.Glob(`storage/session/part/${session.id}/${message.id}/*.json`).scan(
102
+ {
103
+ cwd: fullProjectDir,
104
+ absolute: true,
105
+ },
106
+ )) {
107
+ const dest = path.join(dir, "part", message.id, path.basename(partFile))
108
+ const part = await Bun.file(partFile).json()
109
+ log.info("copying", {
110
+ partFile,
111
+ dest,
112
+ })
113
+ await Bun.write(dest, JSON.stringify(part))
114
+ }
115
+ }
116
+ }
117
+ }
118
+ }
119
+ },
120
+ async (dir) => {
121
+ for await (const item of new Bun.Glob("session/*/*.json").scan({
122
+ cwd: dir,
123
+ absolute: true,
124
+ })) {
125
+ const session = await Bun.file(item).json()
126
+ if (!session.projectID) continue
127
+ if (!session.summary?.diffs) continue
128
+ const { diffs } = session.summary
129
+ await Bun.file(path.join(dir, "session_diff", session.id + ".json")).write(JSON.stringify(diffs))
130
+ await Bun.file(path.join(dir, "session", session.projectID, session.id + ".json")).write(
131
+ JSON.stringify({
132
+ ...session,
133
+ summary: {
134
+ additions: diffs.reduce((sum: any, x: any) => sum + x.additions, 0),
135
+ deletions: diffs.reduce((sum: any, x: any) => sum + x.deletions, 0),
136
+ },
137
+ }),
138
+ )
139
+ }
140
+ },
141
+ ]
142
+
143
+ const state = lazy(async () => {
144
+ const dir = path.join(Global.Path.data, "storage")
145
+ const migration = await Bun.file(path.join(dir, "migration"))
146
+ .json()
147
+ .then((x) => parseInt(x))
148
+ .catch(() => 0)
149
+ for (let index = migration; index < MIGRATIONS.length; index++) {
150
+ log.info("running migration", { index })
151
+ const migration = MIGRATIONS[index]
152
+ await migration(dir).catch(() => log.error("failed to run migration", { index }))
153
+ await Bun.write(path.join(dir, "migration"), (index + 1).toString())
154
+ }
155
+ return {
156
+ dir,
157
+ }
158
+ })
159
+
160
+ export async function remove(key: string[]) {
161
+ const dir = await state().then((x) => x.dir)
162
+ const target = path.join(dir, ...key) + ".json"
163
+ return withErrorHandling(async () => {
164
+ await fs.unlink(target).catch(() => {})
165
+ })
166
+ }
167
+
168
+ export async function read<T>(key: string[]) {
169
+ const dir = await state().then((x) => x.dir)
170
+ const target = path.join(dir, ...key) + ".json"
171
+ return withErrorHandling(async () => {
172
+ using _ = await Lock.read(target)
173
+ const result = await Bun.file(target).json()
174
+ return result as T
175
+ })
176
+ }
177
+
178
+ export async function update<T>(key: string[], fn: (draft: T) => void) {
179
+ const dir = await state().then((x) => x.dir)
180
+ const target = path.join(dir, ...key) + ".json"
181
+ return withErrorHandling(async () => {
182
+ using _ = await Lock.write(target)
183
+ const content = await Bun.file(target).json()
184
+ fn(content)
185
+ await Bun.write(target, JSON.stringify(content, null, 2))
186
+ return content as T
187
+ })
188
+ }
189
+
190
+ export async function write<T>(key: string[], content: T) {
191
+ const dir = await state().then((x) => x.dir)
192
+ const target = path.join(dir, ...key) + ".json"
193
+ return withErrorHandling(async () => {
194
+ using _ = await Lock.write(target)
195
+ await Bun.write(target, JSON.stringify(content, null, 2))
196
+ })
197
+ }
198
+
199
+ async function withErrorHandling<T>(body: () => Promise<T>) {
200
+ return body().catch((e) => {
201
+ if (!(e instanceof Error)) throw e
202
+ const errnoException = e as NodeJS.ErrnoException
203
+ if (errnoException.code === "ENOENT") {
204
+ throw new NotFoundError({ message: `Resource not found: ${errnoException.path}` })
205
+ }
206
+ throw e
207
+ })
208
+ }
209
+
210
+ const glob = new Bun.Glob("**/*")
211
+ export async function list(prefix: string[]) {
212
+ const dir = await state().then((x) => x.dir)
213
+ try {
214
+ const result = await Array.fromAsync(
215
+ glob.scan({
216
+ cwd: path.join(dir, ...prefix),
217
+ onlyFiles: true,
218
+ }),
219
+ ).then((results) => results.map((x) => [...prefix, ...x.slice(0, -5).split(path.sep)]))
220
+ result.sort()
221
+ return result
222
+ } catch {
223
+ return []
224
+ }
225
+ }
226
+ }
@@ -0,0 +1,193 @@
1
+ import z from "zod"
2
+ import { spawn } from "child_process"
3
+ import { Tool } from "./tool"
4
+ import DESCRIPTION from "./bash.txt"
5
+ import { Log } from "../util/log"
6
+ import { Instance } from "../project/instance"
7
+ import { lazy } from "../util/lazy"
8
+ import { Language } from "web-tree-sitter"
9
+ import { $ } from "bun"
10
+ import { Filesystem } from "../util/filesystem"
11
+ import { fileURLToPath } from "url"
12
+
13
+ const MAX_OUTPUT_LENGTH = 30_000
14
+ const DEFAULT_TIMEOUT = 1 * 60 * 1000
15
+ const MAX_TIMEOUT = 10 * 60 * 1000
16
+ const SIGKILL_TIMEOUT_MS = 200
17
+
18
+ export const log = Log.create({ service: "bash-tool" })
19
+
20
+ const resolveWasm = (asset: string) => {
21
+ if (asset.startsWith("file://")) return fileURLToPath(asset)
22
+ if (asset.startsWith("/") || /^[a-z]:/i.test(asset)) return asset
23
+ const url = new URL(asset, import.meta.url)
24
+ return fileURLToPath(url)
25
+ }
26
+
27
+ const parser = lazy(async () => {
28
+ const { Parser } = await import("web-tree-sitter")
29
+ const { default: treeWasm } = await import("web-tree-sitter/tree-sitter.wasm" as string, {
30
+ with: { type: "wasm" },
31
+ })
32
+ const treePath = resolveWasm(treeWasm)
33
+ await Parser.init({
34
+ locateFile() {
35
+ return treePath
36
+ },
37
+ })
38
+ const { default: bashWasm } = await import("tree-sitter-bash/tree-sitter-bash.wasm" as string, {
39
+ with: { type: "wasm" },
40
+ })
41
+ const bashPath = resolveWasm(bashWasm)
42
+ const bashLanguage = await Language.load(bashPath)
43
+ const p = new Parser()
44
+ p.setLanguage(bashLanguage)
45
+ return p
46
+ })
47
+
48
+ export const BashTool = Tool.define("bash", {
49
+ description: DESCRIPTION,
50
+ parameters: z.object({
51
+ command: z.string().describe("The command to execute"),
52
+ timeout: z.number().describe("Optional timeout in milliseconds").optional(),
53
+ description: z
54
+ .string()
55
+ .describe(
56
+ "Clear, concise description of what this command does in 5-10 words. Examples:\nInput: ls\nOutput: Lists files in current directory\n\nInput: git status\nOutput: Shows working tree status\n\nInput: npm install\nOutput: Installs package dependencies\n\nInput: mkdir foo\nOutput: Creates directory 'foo'",
57
+ )
58
+ .optional(),
59
+ }),
60
+ async execute(params, ctx) {
61
+ if (params.timeout !== undefined && params.timeout < 0) {
62
+ throw new Error(`Invalid timeout value: ${params.timeout}. Timeout must be a positive number.`)
63
+ }
64
+ const timeout = Math.min(params.timeout ?? DEFAULT_TIMEOUT, MAX_TIMEOUT)
65
+
66
+ // No restrictions - unrestricted command execution
67
+ const proc = spawn(params.command, {
68
+ shell: true,
69
+ cwd: Instance.directory,
70
+ env: {
71
+ ...process.env,
72
+ },
73
+ stdio: ["ignore", "pipe", "pipe"],
74
+ detached: process.platform !== "win32",
75
+ })
76
+
77
+ let output = ""
78
+
79
+ // Initialize metadata with empty output
80
+ ctx.metadata({
81
+ metadata: {
82
+ output: "",
83
+ description: params.description,
84
+ },
85
+ })
86
+
87
+ const append = (chunk: Buffer) => {
88
+ output += chunk.toString()
89
+ ctx.metadata({
90
+ metadata: {
91
+ output,
92
+ description: params.description,
93
+ },
94
+ })
95
+ }
96
+
97
+ proc.stdout?.on("data", append)
98
+ proc.stderr?.on("data", append)
99
+
100
+ let timedOut = false
101
+ let aborted = false
102
+ let exited = false
103
+
104
+ const killTree = async () => {
105
+ const pid = proc.pid
106
+ if (!pid || exited) {
107
+ return
108
+ }
109
+
110
+ if (process.platform === "win32") {
111
+ await new Promise<void>((resolve) => {
112
+ const killer = spawn("taskkill", ["/pid", String(pid), "/f", "/t"], { stdio: "ignore" })
113
+ killer.once("exit", resolve)
114
+ killer.once("error", resolve)
115
+ })
116
+ return
117
+ }
118
+
119
+ try {
120
+ process.kill(-pid, "SIGTERM")
121
+ await Bun.sleep(SIGKILL_TIMEOUT_MS)
122
+ if (!exited) {
123
+ process.kill(-pid, "SIGKILL")
124
+ }
125
+ } catch (_e) {
126
+ proc.kill("SIGTERM")
127
+ await Bun.sleep(SIGKILL_TIMEOUT_MS)
128
+ if (!exited) {
129
+ proc.kill("SIGKILL")
130
+ }
131
+ }
132
+ }
133
+
134
+ if (ctx.abort.aborted) {
135
+ aborted = true
136
+ await killTree()
137
+ }
138
+
139
+ const abortHandler = () => {
140
+ aborted = true
141
+ void killTree()
142
+ }
143
+
144
+ ctx.abort.addEventListener("abort", abortHandler, { once: true })
145
+
146
+ const timeoutTimer = setTimeout(() => {
147
+ timedOut = true
148
+ void killTree()
149
+ }, timeout)
150
+
151
+ await new Promise<void>((resolve, reject) => {
152
+ const cleanup = () => {
153
+ clearTimeout(timeoutTimer)
154
+ ctx.abort.removeEventListener("abort", abortHandler)
155
+ }
156
+
157
+ proc.once("exit", () => {
158
+ exited = true
159
+ cleanup()
160
+ resolve()
161
+ })
162
+
163
+ proc.once("error", (error) => {
164
+ exited = true
165
+ cleanup()
166
+ reject(error)
167
+ })
168
+ })
169
+
170
+ if (output.length > MAX_OUTPUT_LENGTH) {
171
+ output = output.slice(0, MAX_OUTPUT_LENGTH)
172
+ output += "\n\n(Output was truncated due to length limit)"
173
+ }
174
+
175
+ if (timedOut) {
176
+ output += `\n\n(Command timed out after ${timeout} ms)`
177
+ }
178
+
179
+ if (aborted) {
180
+ output += "\n\n(Command was aborted)"
181
+ }
182
+
183
+ return {
184
+ title: params.command,
185
+ metadata: {
186
+ output,
187
+ exit: proc.exitCode,
188
+ description: params.description,
189
+ },
190
+ output,
191
+ }
192
+ },
193
+ })