| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218 |
- import z from "zod/v4"
- import { exec } from "child_process"
- import { Tool } from "./tool"
- import DESCRIPTION from "./bash.txt"
- import { Permission } from "../permission"
- import { Filesystem } from "../util/filesystem"
- import { lazy } from "../util/lazy"
- import { Log } from "../util/log"
- import { Wildcard } from "../util/wildcard"
- import { $ } from "bun"
- import { Instance } from "../project/instance"
- import { Agent } from "../agent/agent"
- const MAX_OUTPUT_LENGTH = 30_000
- const DEFAULT_TIMEOUT = 1 * 60 * 1000
- const MAX_TIMEOUT = 10 * 60 * 1000
- const log = Log.create({ service: "bash-tool" })
- const parser = lazy(async () => {
- try {
- const { default: Parser } = await import("tree-sitter")
- const Bash = await import("tree-sitter-bash")
- const p = new Parser()
- p.setLanguage(Bash.language as any)
- return p
- } catch (e) {
- const { default: Parser } = await import("web-tree-sitter")
- const { default: treeWasm } = await import("web-tree-sitter/tree-sitter.wasm" as string, { with: { type: "wasm" } })
- await Parser.init({
- locateFile() {
- return treeWasm
- },
- })
- const { default: bashWasm } = await import("tree-sitter-bash/tree-sitter-bash.wasm" as string, {
- with: { type: "wasm" },
- })
- const bashLanguage = await Parser.Language.load(bashWasm)
- const p = new Parser()
- p.setLanguage(bashLanguage)
- return p
- }
- })
- export const BashTool = Tool.define("bash", {
- description: DESCRIPTION,
- parameters: z.object({
- command: z.string().describe("The command to execute"),
- timeout: z.number().describe("Optional timeout in milliseconds").optional(),
- description: z
- .string()
- .describe(
- "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'",
- ),
- }),
- async execute(params, ctx) {
- const timeout = Math.min(params.timeout ?? DEFAULT_TIMEOUT, MAX_TIMEOUT)
- const tree = await parser().then((p) => p.parse(params.command))
- const permissions = await Agent.get(ctx.agent).then((x) => x.permission.bash)
- const askPatterns = new Set<string>()
- for (const node of tree.rootNode.descendantsOfType("command")) {
- const command = []
- for (let i = 0; i < node.childCount; i++) {
- const child = node.child(i)
- if (!child) continue
- if (
- child.type !== "command_name" &&
- child.type !== "word" &&
- child.type !== "string" &&
- child.type !== "raw_string" &&
- child.type !== "concatenation"
- ) {
- continue
- }
- command.push(child.text)
- }
- // not an exhaustive list, but covers most common cases
- if (["cd", "rm", "cp", "mv", "mkdir", "touch", "chmod", "chown"].includes(command[0])) {
- for (const arg of command.slice(1)) {
- if (arg.startsWith("-") || (command[0] === "chmod" && arg.startsWith("+"))) continue
- const resolved = await $`realpath ${arg}`
- .quiet()
- .nothrow()
- .text()
- .then((x) => x.trim())
- log.info("resolved path", { arg, resolved })
- if (resolved && !Filesystem.contains(Instance.directory, resolved)) {
- throw new Error(
- `This command references paths outside of ${Instance.directory} so it is not allowed to be executed.`,
- )
- }
- }
- }
- // always allow cd if it passes above check
- if (command[0] !== "cd") {
- const action = Wildcard.all(node.text, permissions)
- if (action === "deny") {
- throw new Error(
- `The user has specifically restricted access to this command, you are not allowed to execute it. Here is the configuration: ${JSON.stringify(permissions)}`,
- )
- }
- if (action === "ask") {
- const pattern = (() => {
- let head = ""
- let sub: string | undefined
- for (let i = 0; i < node.childCount; i++) {
- const child = node.child(i)
- if (!child) continue
- if (child.type === "command_name") {
- if (!head) {
- head = child.text
- }
- continue
- }
- if (!sub && child.type === "word") {
- if (!child.text.startsWith("-")) sub = child.text
- }
- }
- if (!head) return
- return sub ? `${head} ${sub} *` : `${head} *`
- })()
- if (pattern) {
- askPatterns.add(pattern)
- }
- }
- }
- }
- if (askPatterns.size > 0) {
- const patterns = Array.from(askPatterns)
- await Permission.ask({
- type: "bash",
- pattern: patterns,
- sessionID: ctx.sessionID,
- messageID: ctx.messageID,
- callID: ctx.callID,
- title: params.command,
- metadata: {
- command: params.command,
- patterns,
- },
- })
- }
- const process = exec(params.command, {
- cwd: Instance.directory,
- signal: ctx.abort,
- timeout,
- })
- let output = ""
- // Initialize metadata with empty output
- ctx.metadata({
- metadata: {
- output: "",
- description: params.description,
- },
- })
- process.stdout?.on("data", (chunk) => {
- output += chunk.toString()
- ctx.metadata({
- metadata: {
- output: output,
- description: params.description,
- },
- })
- })
- process.stderr?.on("data", (chunk) => {
- output += chunk.toString()
- ctx.metadata({
- metadata: {
- output: output,
- description: params.description,
- },
- })
- })
- await new Promise<void>((resolve) => {
- process.on("close", () => {
- resolve()
- })
- })
- ctx.metadata({
- metadata: {
- output: output,
- exit: process.exitCode,
- description: params.description,
- },
- })
- if (output.length > MAX_OUTPUT_LENGTH) {
- output = output.slice(0, MAX_OUTPUT_LENGTH)
- output += "\n\n(Output was truncated due to length limit)"
- }
- if (process.signalCode === "SIGTERM" && params.timeout) {
- output += `\n\n(Command timed out after ${timeout} ms)`
- }
- return {
- title: params.command,
- metadata: {
- output,
- exit: process.exitCode,
- description: params.description,
- },
- output,
- }
- },
- })
|