mirror of
https://github.com/pacnpal/Roo-Code.git
synced 2025-12-22 05:11:06 -05:00
* Chore: Pretier for consistant formatting - TODO: This PR needs to be updated by Saoud after he runs `npm install` & `npm format:fix` and commits the results of the prettier changes. * Revert prettier config * Run npm install * Fix prettier config and ignore package lock * Run format --------- Co-authored-by: Saoud Rizwan <7799382+saoudrizwan@users.noreply.github.com>
131 lines
4.9 KiB
TypeScript
131 lines
4.9 KiB
TypeScript
import { Anthropic } from "@anthropic-ai/sdk"
|
|
import * as path from "path"
|
|
import * as diff from "diff"
|
|
|
|
export const formatResponse = {
|
|
toolDenied: () => `The user denied this operation.`,
|
|
|
|
toolDeniedWithFeedback: (feedback?: string) =>
|
|
`The user denied this operation and provided the following feedback:\n<feedback>\n${feedback}\n</feedback>`,
|
|
|
|
toolError: (error?: string) => `The tool execution failed with the following error:\n<error>\n${error}\n</error>`,
|
|
|
|
noToolsUsed: () =>
|
|
`[ERROR] You did not use a tool in your previous response! Please retry with a tool use.
|
|
|
|
${toolUseInstructionsReminder}
|
|
|
|
# Next Steps
|
|
|
|
If you have completed the user's task, use the attempt_completion tool.
|
|
If you require additional information from the user, use the ask_followup_question tool.
|
|
Otherwise, if you have not completed the task and do not need additional information, then proceed with the next step of the task.
|
|
(This is an automated message, so do not respond to it conversationally.)`,
|
|
|
|
tooManyMistakes: (feedback?: string) =>
|
|
`You seem to be having trouble proceeding. The user has provided the following feedback to help guide you:\n<feedback>\n${feedback}\n</feedback>`,
|
|
|
|
missingToolParameterError: (paramName: string) =>
|
|
`Missing value for required parameter '${paramName}'. Please retry with complete response.\n\n${toolUseInstructionsReminder}`,
|
|
|
|
toolResult: (
|
|
text: string,
|
|
images?: string[],
|
|
): string | Array<Anthropic.TextBlockParam | Anthropic.ImageBlockParam> => {
|
|
if (images && images.length > 0) {
|
|
const textBlock: Anthropic.TextBlockParam = { type: "text", text }
|
|
const imageBlocks: Anthropic.ImageBlockParam[] = formatImagesIntoBlocks(images)
|
|
// Placing images after text leads to better results
|
|
return [textBlock, ...imageBlocks]
|
|
} else {
|
|
return text
|
|
}
|
|
},
|
|
|
|
imageBlocks: (images?: string[]): Anthropic.ImageBlockParam[] => {
|
|
return formatImagesIntoBlocks(images)
|
|
},
|
|
|
|
formatFilesList: (absolutePath: string, files: string[], didHitLimit: boolean): string => {
|
|
const sorted = files
|
|
.map((file) => {
|
|
// convert absolute path to relative path
|
|
const relativePath = path.relative(absolutePath, file).toPosix()
|
|
return file.endsWith("/") ? relativePath + "/" : relativePath
|
|
})
|
|
// Sort so files are listed under their respective directories to make it clear what files are children of what directories. Since we build file list top down, even if file list is truncated it will show directories that cline can then explore further.
|
|
.sort((a, b) => {
|
|
const aParts = a.split("/") // only works if we use toPosix first
|
|
const bParts = b.split("/")
|
|
for (let i = 0; i < Math.min(aParts.length, bParts.length); i++) {
|
|
if (aParts[i] !== bParts[i]) {
|
|
// If one is a directory and the other isn't at this level, sort the directory first
|
|
if (i + 1 === aParts.length && i + 1 < bParts.length) {
|
|
return -1
|
|
}
|
|
if (i + 1 === bParts.length && i + 1 < aParts.length) {
|
|
return 1
|
|
}
|
|
// Otherwise, sort alphabetically
|
|
return aParts[i].localeCompare(bParts[i], undefined, { numeric: true, sensitivity: "base" })
|
|
}
|
|
}
|
|
// If all parts are the same up to the length of the shorter path,
|
|
// the shorter one comes first
|
|
return aParts.length - bParts.length
|
|
})
|
|
if (didHitLimit) {
|
|
return `${sorted.join(
|
|
"\n",
|
|
)}\n\n(File list truncated. Use list_files on specific subdirectories if you need to explore further.)`
|
|
} else if (sorted.length === 0 || (sorted.length === 1 && sorted[0] === "")) {
|
|
return "No files found."
|
|
} else {
|
|
return sorted.join("\n")
|
|
}
|
|
},
|
|
|
|
createPrettyPatch: (filename = "file", oldStr?: string, newStr?: string) => {
|
|
// strings cannot be undefined or diff throws exception
|
|
const patch = diff.createPatch(filename.toPosix(), oldStr || "", newStr || "")
|
|
const lines = patch.split("\n")
|
|
const prettyPatchLines = lines.slice(4)
|
|
return prettyPatchLines.join("\n")
|
|
},
|
|
}
|
|
|
|
// to avoid circular dependency
|
|
const formatImagesIntoBlocks = (images?: string[]): Anthropic.ImageBlockParam[] => {
|
|
return images
|
|
? images.map((dataUrl) => {
|
|
// 
|
|
const [rest, base64] = dataUrl.split(",")
|
|
const mimeType = rest.split(":")[1].split(";")[0]
|
|
return {
|
|
type: "image",
|
|
source: { type: "base64", media_type: mimeType, data: base64 },
|
|
} as Anthropic.ImageBlockParam
|
|
})
|
|
: []
|
|
}
|
|
|
|
const toolUseInstructionsReminder = `# Reminder: Instructions for Tool Use
|
|
|
|
Tool uses are formatted using XML-style tags. The tool name is enclosed in opening and closing tags, and each parameter is similarly enclosed within its own set of tags. Here's the structure:
|
|
|
|
<tool_name>
|
|
<parameter1_name>value1</parameter1_name>
|
|
<parameter2_name>value2</parameter2_name>
|
|
...
|
|
</tool_name>
|
|
|
|
For example:
|
|
|
|
<attempt_completion>
|
|
<result>
|
|
I have completed the task...
|
|
</result>
|
|
</attempt_completion>
|
|
|
|
Always adhere to this format for all tool uses to ensure proper parsing and execution.`
|