Fix coding agent tools to return error content instead of throwing

Tools now resolve with error messages in content blocks rather than
rejecting the promise. This matches the expected behavior where tools
always return a result, with errors indicated in the text content.

- read: Return error content for missing files
- edit: Return error content for missing files, text not found, multiple matches
- bash: Return error content for command failures, timeouts, and aborts
- All tools now include required details field in error results
This commit is contained in:
Mario Zechner 2025-11-12 10:52:12 +01:00
parent 84dcab219b
commit f147109da7
3 changed files with 33 additions and 16 deletions

View file

@ -13,7 +13,7 @@ export const bashTool: AgentTool<typeof bashSchema> = {
"Execute a bash command in the current working directory. Returns stdout and stderr. Commands run with a 30 second timeout.",
parameters: bashSchema,
execute: async (_toolCallId: string, { command }: { command: string }, signal?: AbortSignal) => {
return new Promise((resolve, reject) => {
return new Promise((resolve, _reject) => {
const child = spawn("sh", ["-c", command], {
detached: true,
stdio: ["ignore", "pipe", "pipe"],
@ -67,7 +67,7 @@ export const bashTool: AgentTool<typeof bashSchema> = {
}
if (output) output += "\n\n";
output += "Command aborted";
reject(new Error(output));
resolve({ content: [{ type: "text", text: `Command failed\n\n${output}` }], details: undefined });
return;
}
@ -80,7 +80,7 @@ export const bashTool: AgentTool<typeof bashSchema> = {
}
if (output) output += "\n\n";
output += "Command timed out after 30 seconds";
reject(new Error(output));
resolve({ content: [{ type: "text", text: `Command failed\n\n${output}` }], details: undefined });
return;
}
@ -93,7 +93,10 @@ export const bashTool: AgentTool<typeof bashSchema> = {
if (code !== 0 && code !== null) {
if (output) output += "\n\n";
reject(new Error(`${output}Command exited with code ${code}`));
resolve({
content: [{ type: "text", text: `Command failed\n\n${output}Command exited with code ${code}` }],
details: undefined,
});
} else {
resolve({ content: [{ type: "text", text: output || "(no output)" }], details: undefined });
}

View file

@ -66,7 +66,10 @@ export const editTool: AgentTool<typeof editSchema> = {
if (signal) {
signal.removeEventListener("abort", onAbort);
}
reject(new Error(`File not found: ${path}`));
resolve({
content: [{ type: "text", text: `Error: File not found: ${path}` }],
details: undefined,
});
return;
}
@ -88,11 +91,15 @@ export const editTool: AgentTool<typeof editSchema> = {
if (signal) {
signal.removeEventListener("abort", onAbort);
}
reject(
new Error(
`Could not find the exact text in ${path}. The old text must match exactly including all whitespace and newlines.`,
),
);
resolve({
content: [
{
type: "text",
text: `Error: Could not find the exact text in ${path}. The old text must match exactly including all whitespace and newlines.`,
},
],
details: undefined,
});
return;
}
@ -103,11 +110,15 @@ export const editTool: AgentTool<typeof editSchema> = {
if (signal) {
signal.removeEventListener("abort", onAbort);
}
reject(
new Error(
`Found ${occurrences} occurrences of the text in ${path}. The text must be unique. Please provide more context to make it unique.`,
),
);
resolve({
content: [
{
type: "text",
text: `Error: Found ${occurrences} occurrences of the text in ${path}. The text must be unique. Please provide more context to make it unique.`,
},
],
details: undefined,
});
return;
}

View file

@ -82,7 +82,10 @@ export const readTool: AgentTool<typeof readSchema> = {
if (signal) {
signal.removeEventListener("abort", onAbort);
}
reject(new Error(`File not found: ${path}`));
resolve({
content: [{ type: "text", text: `Error: File not found: ${path}` }],
details: undefined,
});
return;
}