@oflow-ai/core 0.1.0 → 0.1.2
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.
- package/dist/conversation.js +21 -2
- package/dist/tools/base.d.ts +7 -0
- package/dist/tools/base.js +28 -2
- package/dist/tools/glob.d.ts +6 -2
- package/dist/tools/glob.js +28 -87
- package/dist/tools/list-directory.d.ts +1 -0
- package/dist/tools/list-directory.js +29 -24
- package/dist/tools/read-file.d.ts +1 -0
- package/dist/tools/read-file.js +7 -6
- package/dist/tools/replace.d.ts +1 -0
- package/dist/tools/replace.js +24 -19
- package/dist/tools/run-shell-command.d.ts +2 -5
- package/dist/tools/run-shell-command.js +61 -37
- package/dist/tools/search-file-content.d.ts +2 -2
- package/dist/tools/search-file-content.js +94 -72
- package/dist/tools/write-file.d.ts +1 -0
- package/dist/tools/write-file.js +7 -6
- package/dist/types/index.d.ts +6 -0
- package/dist/types/index.js +1 -1
- package/package.json +1 -1
package/dist/tools/replace.js
CHANGED
|
@@ -39,50 +39,55 @@ const fs = __importStar(require("fs/promises"));
|
|
|
39
39
|
const path = __importStar(require("path"));
|
|
40
40
|
class ReplaceTool extends base_1.BaseTool {
|
|
41
41
|
name = 'replace';
|
|
42
|
-
description = 'Replaces text within a file.
|
|
42
|
+
description = 'Replaces text within a file. Replaces a single occurrence. This tool requires providing significant context around the change to ensure precise targeting. Always use the read_file tool to examine the file current content before attempting a text replacement.';
|
|
43
43
|
parameters = {
|
|
44
44
|
type: 'object',
|
|
45
45
|
properties: {
|
|
46
46
|
file_path: {
|
|
47
47
|
type: 'string',
|
|
48
|
-
description: 'The absolute path to the file to modify'
|
|
48
|
+
description: 'The absolute path to the file to modify. Must start with /'
|
|
49
49
|
},
|
|
50
50
|
old_string: {
|
|
51
51
|
type: 'string',
|
|
52
|
-
description: 'The exact literal text to replace'
|
|
52
|
+
description: 'The exact literal text to replace. Include at least 3 lines of context BEFORE and AFTER the target text, matching whitespace and indentation precisely.'
|
|
53
53
|
},
|
|
54
54
|
new_string: {
|
|
55
55
|
type: 'string',
|
|
56
|
-
description: 'The exact literal text to replace with'
|
|
56
|
+
description: 'The exact literal text to replace old_string with. Also including all whitespace, indentation, newlines, and surrounding code etc.'
|
|
57
57
|
},
|
|
58
58
|
instruction: {
|
|
59
59
|
type: 'string',
|
|
60
|
-
description: '
|
|
60
|
+
description: 'A clear, semantic instruction for the code change, acting as a high-quality prompt for an expert LLM assistant.'
|
|
61
61
|
}
|
|
62
62
|
},
|
|
63
|
-
required: ['file_path', 'old_string', 'new_string']
|
|
63
|
+
required: ['file_path', 'old_string', 'new_string'],
|
|
64
|
+
additionalProperties: false
|
|
64
65
|
};
|
|
65
66
|
async execute(params, options) {
|
|
66
67
|
this.validateRequired(params, ['file_path', 'old_string', 'new_string']);
|
|
68
|
+
const filePath = params.file_path;
|
|
67
69
|
const workingDir = options?.workingDirectory || process.cwd();
|
|
68
|
-
const
|
|
69
|
-
|
|
70
|
-
: path.
|
|
70
|
+
const resolvedPath = path.isAbsolute(filePath) ? filePath : path.join(workingDir, filePath);
|
|
71
|
+
if (!path.isAbsolute(filePath)) {
|
|
72
|
+
return `Error: file_path must be an absolute path. Received: ${filePath}`;
|
|
73
|
+
}
|
|
74
|
+
if (params.old_string === params.new_string) {
|
|
75
|
+
return `Error: old_string and new_string are identical. No changes made.`;
|
|
76
|
+
}
|
|
71
77
|
try {
|
|
72
|
-
const content = await fs.readFile(
|
|
73
|
-
//
|
|
74
|
-
if (!content.includes(params.old_string)) {
|
|
75
|
-
return `Error: The old_string was not found in ${filePath}. Please read the file first to ensure exact matching.`;
|
|
76
|
-
}
|
|
77
|
-
// Check for multiple occurrences
|
|
78
|
+
const content = await fs.readFile(resolvedPath, 'utf-8');
|
|
79
|
+
// Count occurrences
|
|
78
80
|
const occurrences = content.split(params.old_string).length - 1;
|
|
81
|
+
if (occurrences === 0) {
|
|
82
|
+
return `Error: old_string not found in file ${resolvedPath}. Please read the file first to see its current content.`;
|
|
83
|
+
}
|
|
79
84
|
if (occurrences > 1) {
|
|
80
|
-
return `Error: Found ${occurrences} occurrences of old_string. The
|
|
85
|
+
return `Error: Found ${occurrences} occurrences of old_string. The string must uniquely identify a single location. Please include more context lines to make it unique.`;
|
|
81
86
|
}
|
|
82
87
|
// Perform replacement
|
|
83
88
|
const newContent = content.replace(params.old_string, params.new_string);
|
|
84
|
-
await fs.writeFile(
|
|
85
|
-
return `Successfully replaced text in ${
|
|
89
|
+
await fs.writeFile(resolvedPath, newContent, 'utf-8');
|
|
90
|
+
return `Successfully replaced text in ${resolvedPath}`;
|
|
86
91
|
}
|
|
87
92
|
catch (error) {
|
|
88
93
|
return (0, base_1.formatToolError)(error);
|
|
@@ -90,4 +95,4 @@ class ReplaceTool extends base_1.BaseTool {
|
|
|
90
95
|
}
|
|
91
96
|
}
|
|
92
97
|
exports.ReplaceTool = ReplaceTool;
|
|
93
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
98
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmVwbGFjZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy90b29scy9yZXBsYWNlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQUFBLGlDQUF1RTtBQUN2RSxnREFBa0M7QUFDbEMsMkNBQTZCO0FBUzdCLE1BQWEsV0FBWSxTQUFRLGVBQXVCO0lBQ3RELElBQUksR0FBRyxTQUFTLENBQUM7SUFDakIsV0FBVyxHQUFHLG9RQUFvUSxDQUFDO0lBQ25SLFVBQVUsR0FBRztRQUNYLElBQUksRUFBRSxRQUFRO1FBQ2QsVUFBVSxFQUFFO1lBQ1YsU0FBUyxFQUFFO2dCQUNULElBQUksRUFBRSxRQUFRO2dCQUNkLFdBQVcsRUFBRSw0REFBNEQ7YUFDMUU7WUFDRCxVQUFVLEVBQUU7Z0JBQ1YsSUFBSSxFQUFFLFFBQVE7Z0JBQ2QsV0FBVyxFQUFFLHlKQUF5SjthQUN2SztZQUNELFVBQVUsRUFBRTtnQkFDVixJQUFJLEVBQUUsUUFBUTtnQkFDZCxXQUFXLEVBQUUsb0lBQW9JO2FBQ2xKO1lBQ0QsV0FBVyxFQUFFO2dCQUNYLElBQUksRUFBRSxRQUFRO2dCQUNkLFdBQVcsRUFBRSxpSEFBaUg7YUFDL0g7U0FDRjtRQUNELFFBQVEsRUFBRSxDQUFDLFdBQVcsRUFBRSxZQUFZLEVBQUUsWUFBWSxDQUFDO1FBQ25ELG9CQUFvQixFQUFFLEtBQUs7S0FDNUIsQ0FBQztJQUVGLEtBQUssQ0FBQyxPQUFPLENBQUMsTUFBcUIsRUFBRSxPQUE0QjtRQUMvRCxJQUFJLENBQUMsZ0JBQWdCLENBQUMsTUFBTSxFQUFFLENBQUMsV0FBVyxFQUFFLFlBQVksRUFBRSxZQUFZLENBQUMsQ0FBQyxDQUFDO1FBRXpFLE1BQU0sUUFBUSxHQUFHLE1BQU0sQ0FBQyxTQUFTLENBQUM7UUFDbEMsTUFBTSxVQUFVLEdBQUcsT0FBTyxFQUFFLGdCQUFnQixJQUFJLE9BQU8sQ0FBQyxHQUFHLEVBQUUsQ0FBQztRQUM5RCxNQUFNLFlBQVksR0FBRyxJQUFJLENBQUMsVUFBVSxDQUFDLFFBQVEsQ0FBQyxDQUFDLENBQUMsQ0FBQyxRQUFRLENBQUMsQ0FBQyxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsVUFBVSxFQUFFLFFBQVEsQ0FBQyxDQUFDO1FBRTVGLElBQUksQ0FBQyxJQUFJLENBQUMsVUFBVSxDQUFDLFFBQVEsQ0FBQyxFQUFFLENBQUM7WUFDL0IsT0FBTyx3REFBd0QsUUFBUSxFQUFFLENBQUM7UUFDNUUsQ0FBQztRQUVELElBQUksTUFBTSxDQUFDLFVBQVUsS0FBSyxNQUFNLENBQUMsVUFBVSxFQUFFLENBQUM7WUFDNUMsT0FBTyxrRUFBa0UsQ0FBQztRQUM1RSxDQUFDO1FBRUQsSUFBSSxDQUFDO1lBQ0gsTUFBTSxPQUFPLEdBQUcsTUFBTSxFQUFFLENBQUMsUUFBUSxDQUFDLFlBQVksRUFBRSxPQUFPLENBQUMsQ0FBQztZQUV6RCxvQkFBb0I7WUFDcEIsTUFBTSxXQUFXLEdBQUcsT0FBTyxDQUFDLEtBQUssQ0FBQyxNQUFNLENBQUMsVUFBVSxDQUFDLENBQUMsTUFBTSxHQUFHLENBQUMsQ0FBQztZQUVoRSxJQUFJLFdBQVcsS0FBSyxDQUFDLEVBQUUsQ0FBQztnQkFDdEIsT0FBTyx1Q0FBdUMsWUFBWSwwREFBMEQsQ0FBQztZQUN2SCxDQUFDO1lBRUQsSUFBSSxXQUFXLEdBQUcsQ0FBQyxFQUFFLENBQUM7Z0JBQ3BCLE9BQU8sZ0JBQWdCLFdBQVcsdUlBQXVJLENBQUM7WUFDNUssQ0FBQztZQUVELHNCQUFzQjtZQUN0QixNQUFNLFVBQVUsR0FBRyxPQUFPLENBQUMsT0FBTyxDQUFDLE1BQU0sQ0FBQyxVQUFVLEVBQUUsTUFBTSxDQUFDLFVBQVUsQ0FBQyxDQUFDO1lBQ3pFLE1BQU0sRUFBRSxDQUFDLFNBQVMsQ0FBQyxZQUFZLEVBQUUsVUFBVSxFQUFFLE9BQU8sQ0FBQyxDQUFDO1lBRXRELE9BQU8saUNBQWlDLFlBQVksRUFBRSxDQUFDO1FBQ3pELENBQUM7UUFBQyxPQUFPLEtBQUssRUFBRSxDQUFDO1lBQ2YsT0FBTyxJQUFBLHNCQUFlLEVBQUMsS0FBSyxDQUFDLENBQUM7UUFDaEMsQ0FBQztJQUNILENBQUM7Q0FDRjtBQWpFRCxrQ0FpRUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBCYXNlVG9vbCwgVG9vbEV4ZWN1dGVPcHRpb25zLCBmb3JtYXRUb29sRXJyb3IgfSBmcm9tICcuL2Jhc2UnO1xuaW1wb3J0ICogYXMgZnMgZnJvbSAnZnMvcHJvbWlzZXMnO1xuaW1wb3J0ICogYXMgcGF0aCBmcm9tICdwYXRoJztcblxuaW50ZXJmYWNlIFJlcGxhY2VQYXJhbXMge1xuICBmaWxlX3BhdGg6IHN0cmluZztcbiAgb2xkX3N0cmluZzogc3RyaW5nO1xuICBuZXdfc3RyaW5nOiBzdHJpbmc7XG4gIGluc3RydWN0aW9uPzogc3RyaW5nO1xufVxuXG5leHBvcnQgY2xhc3MgUmVwbGFjZVRvb2wgZXh0ZW5kcyBCYXNlVG9vbDxSZXBsYWNlUGFyYW1zPiB7XG4gIG5hbWUgPSAncmVwbGFjZSc7XG4gIGRlc2NyaXB0aW9uID0gJ1JlcGxhY2VzIHRleHQgd2l0aGluIGEgZmlsZS4gUmVwbGFjZXMgYSBzaW5nbGUgb2NjdXJyZW5jZS4gVGhpcyB0b29sIHJlcXVpcmVzIHByb3ZpZGluZyBzaWduaWZpY2FudCBjb250ZXh0IGFyb3VuZCB0aGUgY2hhbmdlIHRvIGVuc3VyZSBwcmVjaXNlIHRhcmdldGluZy4gQWx3YXlzIHVzZSB0aGUgcmVhZF9maWxlIHRvb2wgdG8gZXhhbWluZSB0aGUgZmlsZSBjdXJyZW50IGNvbnRlbnQgYmVmb3JlIGF0dGVtcHRpbmcgYSB0ZXh0IHJlcGxhY2VtZW50Lic7XG4gIHBhcmFtZXRlcnMgPSB7XG4gICAgdHlwZTogJ29iamVjdCcsXG4gICAgcHJvcGVydGllczoge1xuICAgICAgZmlsZV9wYXRoOiB7XG4gICAgICAgIHR5cGU6ICdzdHJpbmcnLFxuICAgICAgICBkZXNjcmlwdGlvbjogJ1RoZSBhYnNvbHV0ZSBwYXRoIHRvIHRoZSBmaWxlIHRvIG1vZGlmeS4gTXVzdCBzdGFydCB3aXRoIC8nXG4gICAgICB9LFxuICAgICAgb2xkX3N0cmluZzoge1xuICAgICAgICB0eXBlOiAnc3RyaW5nJyxcbiAgICAgICAgZGVzY3JpcHRpb246ICdUaGUgZXhhY3QgbGl0ZXJhbCB0ZXh0IHRvIHJlcGxhY2UuIEluY2x1ZGUgYXQgbGVhc3QgMyBsaW5lcyBvZiBjb250ZXh0IEJFRk9SRSBhbmQgQUZURVIgdGhlIHRhcmdldCB0ZXh0LCBtYXRjaGluZyB3aGl0ZXNwYWNlIGFuZCBpbmRlbnRhdGlvbiBwcmVjaXNlbHkuJ1xuICAgICAgfSxcbiAgICAgIG5ld19zdHJpbmc6IHtcbiAgICAgICAgdHlwZTogJ3N0cmluZycsXG4gICAgICAgIGRlc2NyaXB0aW9uOiAnVGhlIGV4YWN0IGxpdGVyYWwgdGV4dCB0byByZXBsYWNlIG9sZF9zdHJpbmcgd2l0aC4gQWxzbyBpbmNsdWRpbmcgYWxsIHdoaXRlc3BhY2UsIGluZGVudGF0aW9uLCBuZXdsaW5lcywgYW5kIHN1cnJvdW5kaW5nIGNvZGUgZXRjLidcbiAgICAgIH0sXG4gICAgICBpbnN0cnVjdGlvbjoge1xuICAgICAgICB0eXBlOiAnc3RyaW5nJyxcbiAgICAgICAgZGVzY3JpcHRpb246ICdBIGNsZWFyLCBzZW1hbnRpYyBpbnN0cnVjdGlvbiBmb3IgdGhlIGNvZGUgY2hhbmdlLCBhY3RpbmcgYXMgYSBoaWdoLXF1YWxpdHkgcHJvbXB0IGZvciBhbiBleHBlcnQgTExNIGFzc2lzdGFudC4nXG4gICAgICB9XG4gICAgfSxcbiAgICByZXF1aXJlZDogWydmaWxlX3BhdGgnLCAnb2xkX3N0cmluZycsICduZXdfc3RyaW5nJ10sXG4gICAgYWRkaXRpb25hbFByb3BlcnRpZXM6IGZhbHNlXG4gIH07XG5cbiAgYXN5bmMgZXhlY3V0ZShwYXJhbXM6IFJlcGxhY2VQYXJhbXMsIG9wdGlvbnM/OiBUb29sRXhlY3V0ZU9wdGlvbnMpOiBQcm9taXNlPHN0cmluZz4ge1xuICAgIHRoaXMudmFsaWRhdGVSZXF1aXJlZChwYXJhbXMsIFsnZmlsZV9wYXRoJywgJ29sZF9zdHJpbmcnLCAnbmV3X3N0cmluZyddKTtcblxuICAgIGNvbnN0IGZpbGVQYXRoID0gcGFyYW1zLmZpbGVfcGF0aDtcbiAgICBjb25zdCB3b3JraW5nRGlyID0gb3B0aW9ucz8ud29ya2luZ0RpcmVjdG9yeSB8fCBwcm9jZXNzLmN3ZCgpO1xuICAgIGNvbnN0IHJlc29sdmVkUGF0aCA9IHBhdGguaXNBYnNvbHV0ZShmaWxlUGF0aCkgPyBmaWxlUGF0aCA6IHBhdGguam9pbih3b3JraW5nRGlyLCBmaWxlUGF0aCk7XG5cbiAgICBpZiAoIXBhdGguaXNBYnNvbHV0ZShmaWxlUGF0aCkpIHtcbiAgICAgIHJldHVybiBgRXJyb3I6IGZpbGVfcGF0aCBtdXN0IGJlIGFuIGFic29sdXRlIHBhdGguIFJlY2VpdmVkOiAke2ZpbGVQYXRofWA7XG4gICAgfVxuXG4gICAgaWYgKHBhcmFtcy5vbGRfc3RyaW5nID09PSBwYXJhbXMubmV3X3N0cmluZykge1xuICAgICAgcmV0dXJuIGBFcnJvcjogb2xkX3N0cmluZyBhbmQgbmV3X3N0cmluZyBhcmUgaWRlbnRpY2FsLiBObyBjaGFuZ2VzIG1hZGUuYDtcbiAgICB9XG5cbiAgICB0cnkge1xuICAgICAgY29uc3QgY29udGVudCA9IGF3YWl0IGZzLnJlYWRGaWxlKHJlc29sdmVkUGF0aCwgJ3V0Zi04Jyk7XG5cbiAgICAgIC8vIENvdW50IG9jY3VycmVuY2VzXG4gICAgICBjb25zdCBvY2N1cnJlbmNlcyA9IGNvbnRlbnQuc3BsaXQocGFyYW1zLm9sZF9zdHJpbmcpLmxlbmd0aCAtIDE7XG5cbiAgICAgIGlmIChvY2N1cnJlbmNlcyA9PT0gMCkge1xuICAgICAgICByZXR1cm4gYEVycm9yOiBvbGRfc3RyaW5nIG5vdCBmb3VuZCBpbiBmaWxlICR7cmVzb2x2ZWRQYXRofS4gUGxlYXNlIHJlYWQgdGhlIGZpbGUgZmlyc3QgdG8gc2VlIGl0cyBjdXJyZW50IGNvbnRlbnQuYDtcbiAgICAgIH1cblxuICAgICAgaWYgKG9jY3VycmVuY2VzID4gMSkge1xuICAgICAgICByZXR1cm4gYEVycm9yOiBGb3VuZCAke29jY3VycmVuY2VzfSBvY2N1cnJlbmNlcyBvZiBvbGRfc3RyaW5nLiBUaGUgc3RyaW5nIG11c3QgdW5pcXVlbHkgaWRlbnRpZnkgYSBzaW5nbGUgbG9jYXRpb24uIFBsZWFzZSBpbmNsdWRlIG1vcmUgY29udGV4dCBsaW5lcyB0byBtYWtlIGl0IHVuaXF1ZS5gO1xuICAgICAgfVxuXG4gICAgICAvLyBQZXJmb3JtIHJlcGxhY2VtZW50XG4gICAgICBjb25zdCBuZXdDb250ZW50ID0gY29udGVudC5yZXBsYWNlKHBhcmFtcy5vbGRfc3RyaW5nLCBwYXJhbXMubmV3X3N0cmluZyk7XG4gICAgICBhd2FpdCBmcy53cml0ZUZpbGUocmVzb2x2ZWRQYXRoLCBuZXdDb250ZW50LCAndXRmLTgnKTtcblxuICAgICAgcmV0dXJuIGBTdWNjZXNzZnVsbHkgcmVwbGFjZWQgdGV4dCBpbiAke3Jlc29sdmVkUGF0aH1gO1xuICAgIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgICByZXR1cm4gZm9ybWF0VG9vbEVycm9yKGVycm9yKTtcbiAgICB9XG4gIH1cbn1cbiJdfQ==
|
|
@@ -3,7 +3,6 @@ interface RunShellCommandParams {
|
|
|
3
3
|
command: string;
|
|
4
4
|
description?: string;
|
|
5
5
|
timeout?: number;
|
|
6
|
-
run_in_bg?: boolean;
|
|
7
6
|
}
|
|
8
7
|
export declare class RunShellCommandTool extends BaseTool<RunShellCommandParams> {
|
|
9
8
|
name: string;
|
|
@@ -23,13 +22,11 @@ export declare class RunShellCommandTool extends BaseTool<RunShellCommandParams>
|
|
|
23
22
|
type: string;
|
|
24
23
|
description: string;
|
|
25
24
|
};
|
|
26
|
-
run_in_bg: {
|
|
27
|
-
type: string;
|
|
28
|
-
description: string;
|
|
29
|
-
};
|
|
30
25
|
};
|
|
31
26
|
required: string[];
|
|
27
|
+
additionalProperties: boolean;
|
|
32
28
|
};
|
|
33
29
|
execute(params: RunShellCommandParams, options?: ToolExecuteOptions): Promise<string>;
|
|
30
|
+
private runCommand;
|
|
34
31
|
}
|
|
35
32
|
export {};
|
|
@@ -5,77 +5,101 @@ const base_1 = require("./base");
|
|
|
5
5
|
const child_process_1 = require("child_process");
|
|
6
6
|
class RunShellCommandTool extends base_1.BaseTool {
|
|
7
7
|
name = 'run_shell_command';
|
|
8
|
-
description =
|
|
8
|
+
description = `This tool executes a given shell command as \`powershell.exe -NoProfile -Command <command>\` on Windows, or \`bash -c <command>\` on Unix. Command can start background processes using PowerShell constructs such as \`Start-Process -NoNewWindow\` or \`Start-Job\`.
|
|
9
|
+
|
|
10
|
+
Usage notes:
|
|
11
|
+
- The command argument is required.
|
|
12
|
+
- It is very helpful if you write a clear, concise description of what this command does in 5-10 words.
|
|
13
|
+
- You can use the \`run_in_bg\` parameter to run the command in the background, which allows you to continue working while the command runs. You can monitor the output using the BashOutput tool as it becomes available.
|
|
14
|
+
- The following information is returned:
|
|
15
|
+
Command: Executed command.
|
|
16
|
+
Directory: Directory where command was executed, or \`(root)\`.
|
|
17
|
+
Stdout: Output on stdout stream. Can be \`(empty)\` or partial on error and for any unwaited background processes.
|
|
18
|
+
Stderr: Output on stderr stream. Can be \`(empty)\` or partial on error and for any unwaited background processes.
|
|
19
|
+
Error: Error or \`(none)\` if no error was reported for the subprocess.
|
|
20
|
+
Exit Code: Exit code or \`(none)\` if terminated by signal.
|
|
21
|
+
Signal: Signal number or \`(none)\` if no signal was received.
|
|
22
|
+
Background PIDs: List of background processes started or \`(none)\`.
|
|
23
|
+
Process Group PGID: Process group started or \`(none)\`.`;
|
|
9
24
|
parameters = {
|
|
10
25
|
type: 'object',
|
|
11
26
|
properties: {
|
|
12
27
|
command: {
|
|
13
28
|
type: 'string',
|
|
14
|
-
description: '
|
|
29
|
+
description: 'Exact command to execute as powershell.exe -NoProfile -Command <command> on Windows or bash -c <command> on Unix'
|
|
15
30
|
},
|
|
16
31
|
description: {
|
|
17
32
|
type: 'string',
|
|
18
|
-
description: 'Brief description of
|
|
33
|
+
description: 'Brief description of the command for the user. Be specific and concise. Ideally a single sentence. Can be up to 3 sentences for clarity. No line breaks.'
|
|
19
34
|
},
|
|
20
35
|
timeout: {
|
|
21
36
|
type: 'number',
|
|
22
|
-
description: 'Timeout in seconds
|
|
23
|
-
},
|
|
24
|
-
run_in_bg: {
|
|
25
|
-
type: 'boolean',
|
|
26
|
-
description: 'If true, run command in background'
|
|
37
|
+
description: 'Optional: Timeout in seconds for the command execution. If not provided, uses the default timeout of 120s.'
|
|
27
38
|
}
|
|
28
39
|
},
|
|
29
|
-
required: ['command']
|
|
40
|
+
required: ['command'],
|
|
41
|
+
additionalProperties: false
|
|
30
42
|
};
|
|
31
43
|
async execute(params, options) {
|
|
32
44
|
this.validateRequired(params, ['command']);
|
|
33
|
-
const timeout = (params.timeout || 120) * 1000;
|
|
34
45
|
const workingDir = options?.workingDirectory || process.cwd();
|
|
46
|
+
const timeout = params.timeout || options?.timeout || 120000; // Default 120 seconds
|
|
47
|
+
try {
|
|
48
|
+
const result = await this.runCommand(params.command, workingDir, timeout);
|
|
49
|
+
return result;
|
|
50
|
+
}
|
|
51
|
+
catch (error) {
|
|
52
|
+
return (0, base_1.formatToolError)(error);
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
async runCommand(command, cwd, timeout) {
|
|
35
56
|
return new Promise((resolve) => {
|
|
36
57
|
const isWindows = process.platform === 'win32';
|
|
37
|
-
const shell = isWindows ? 'powershell.exe' : '
|
|
38
|
-
const
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
env: process.env,
|
|
44
|
-
timeout: timeout
|
|
58
|
+
const shell = isWindows ? 'powershell.exe' : 'bash';
|
|
59
|
+
const args = isWindows ? ['-NoProfile', '-Command', command] : ['-c', command];
|
|
60
|
+
const proc = (0, child_process_1.spawn)(shell, args, {
|
|
61
|
+
cwd,
|
|
62
|
+
shell: false,
|
|
63
|
+
windowsHide: true
|
|
45
64
|
});
|
|
46
65
|
let stdout = '';
|
|
47
66
|
let stderr = '';
|
|
48
|
-
let
|
|
67
|
+
let timedOut = false;
|
|
68
|
+
const timeoutId = setTimeout(() => {
|
|
69
|
+
timedOut = true;
|
|
70
|
+
proc.kill();
|
|
71
|
+
}, timeout);
|
|
49
72
|
proc.stdout.on('data', (data) => {
|
|
50
73
|
stdout += data.toString();
|
|
51
74
|
});
|
|
52
75
|
proc.stderr.on('data', (data) => {
|
|
53
76
|
stderr += data.toString();
|
|
54
77
|
});
|
|
55
|
-
proc.on('
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
clearTimeout(timeoutId);
|
|
61
|
-
let output = `Command: ${params.command}\n`;
|
|
62
|
-
output += `Directory: ${workingDir}\n`;
|
|
63
|
-
output += `Exit Code: ${code}\n\n`;
|
|
78
|
+
proc.on('close', (code) => {
|
|
79
|
+
clearTimeout(timeoutId);
|
|
80
|
+
let result = `Command: ${command}\n`;
|
|
81
|
+
result += `Directory: ${cwd}\n`;
|
|
82
|
+
result += `Exit Code: ${code ?? '(none)'}\n`;
|
|
64
83
|
if (stdout) {
|
|
65
|
-
|
|
84
|
+
result += `\nStdout:\n${stdout}`;
|
|
85
|
+
}
|
|
86
|
+
else {
|
|
87
|
+
result += `\nStdout: (empty)`;
|
|
66
88
|
}
|
|
67
89
|
if (stderr) {
|
|
68
|
-
|
|
90
|
+
result += `\n\nStderr:\n${stderr}`;
|
|
91
|
+
}
|
|
92
|
+
if (timedOut) {
|
|
93
|
+
result += `\n\n⚠️ Command timed out after ${timeout / 1000} seconds`;
|
|
69
94
|
}
|
|
70
|
-
resolve(
|
|
95
|
+
resolve(result);
|
|
96
|
+
});
|
|
97
|
+
proc.on('error', (err) => {
|
|
98
|
+
clearTimeout(timeoutId);
|
|
99
|
+
resolve(`Error executing command: ${err.message}`);
|
|
71
100
|
});
|
|
72
|
-
// Handle timeout
|
|
73
|
-
timeoutId = setTimeout(() => {
|
|
74
|
-
proc.kill();
|
|
75
|
-
resolve(`Command timed out after ${timeout / 1000} seconds`);
|
|
76
|
-
}, timeout);
|
|
77
101
|
});
|
|
78
102
|
}
|
|
79
103
|
}
|
|
80
104
|
exports.RunShellCommandTool = RunShellCommandTool;
|
|
81
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
105
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"run-shell-command.js","sourceRoot":"","sources":["../../src/tools/run-shell-command.ts"],"names":[],"mappings":"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","sourcesContent":["import { BaseTool, ToolExecuteOptions, formatToolError } from './base';\nimport { spawn } from 'child_process';\n\ninterface RunShellCommandParams {\n  command: string;\n  description?: string;\n  timeout?: number;\n}\n\nexport class RunShellCommandTool extends BaseTool<RunShellCommandParams> {\n  name = 'run_shell_command';\n  description = `This tool executes a given shell command as \\`powershell.exe -NoProfile -Command <command>\\` on Windows, or \\`bash -c <command>\\` on Unix. Command can start background processes using PowerShell constructs such as \\`Start-Process -NoNewWindow\\` or \\`Start-Job\\`.\n      \nUsage notes:\n- The command argument is required.\n- It is very helpful if you write a clear, concise description of what this command does in 5-10 words.\n- You can use the \\`run_in_bg\\` parameter to run the command in the background, which allows you to continue working while the command runs. You can monitor the output using the BashOutput tool as it becomes available.\n- The following information is returned:\n  Command: Executed command.\n  Directory: Directory where command was executed, or \\`(root)\\`.\n  Stdout: Output on stdout stream. Can be \\`(empty)\\` or partial on error and for any unwaited background processes.\n  Stderr: Output on stderr stream. Can be \\`(empty)\\` or partial on error and for any unwaited background processes.\n  Error: Error or \\`(none)\\` if no error was reported for the subprocess.\n  Exit Code: Exit code or \\`(none)\\` if terminated by signal.\n  Signal: Signal number or \\`(none)\\` if no signal was received.\n  Background PIDs: List of background processes started or \\`(none)\\`.\n  Process Group PGID: Process group started or \\`(none)\\`.`;\n  parameters = {\n    type: 'object',\n    properties: {\n      command: {\n        type: 'string',\n        description: 'Exact command to execute as powershell.exe -NoProfile -Command <command> on Windows or bash -c <command> on Unix'\n      },\n      description: {\n        type: 'string',\n        description: 'Brief description of the command for the user. Be specific and concise. Ideally a single sentence. Can be up to 3 sentences for clarity. No line breaks.'\n      },\n      timeout: {\n        type: 'number',\n        description: 'Optional: Timeout in seconds for the command execution. If not provided, uses the default timeout of 120s.'\n      }\n    },\n    required: ['command'],\n    additionalProperties: false\n  };\n\n  async execute(params: RunShellCommandParams, options?: ToolExecuteOptions): Promise<string> {\n    this.validateRequired(params, ['command']);\n\n    const workingDir = options?.workingDirectory || process.cwd();\n    const timeout = params.timeout || options?.timeout || 120000; // Default 120 seconds\n\n    try {\n      const result = await this.runCommand(params.command, workingDir, timeout);\n      return result;\n    } catch (error) {\n      return formatToolError(error);\n    }\n  }\n\n  private async runCommand(command: string, cwd: string, timeout: number): Promise<string> {\n    return new Promise((resolve) => {\n      const isWindows = process.platform === 'win32';\n      const shell = isWindows ? 'powershell.exe' : 'bash';\n      const args = isWindows ? ['-NoProfile', '-Command', command] : ['-c', command];\n\n      const proc = spawn(shell, args, {\n        cwd,\n        shell: false,\n        windowsHide: true\n      });\n\n      let stdout = '';\n      let stderr = '';\n      let timedOut = false;\n\n      const timeoutId = setTimeout(() => {\n        timedOut = true;\n        proc.kill();\n      }, timeout);\n\n      proc.stdout.on('data', (data) => {\n        stdout += data.toString();\n      });\n\n      proc.stderr.on('data', (data) => {\n        stderr += data.toString();\n      });\n\n      proc.on('close', (code) => {\n        clearTimeout(timeoutId);\n        \n        let result = `Command: ${command}\\n`;\n        result += `Directory: ${cwd}\\n`;\n        result += `Exit Code: ${code ?? '(none)'}\\n`;\n        \n        if (stdout) {\n          result += `\\nStdout:\\n${stdout}`;\n        } else {\n          result += `\\nStdout: (empty)`;\n        }\n        \n        if (stderr) {\n          result += `\\n\\nStderr:\\n${stderr}`;\n        }\n        \n        if (timedOut) {\n          result += `\\n\\n⚠️ Command timed out after ${timeout / 1000} seconds`;\n        }\n        \n        resolve(result);\n      });\n\n      proc.on('error', (err) => {\n        clearTimeout(timeoutId);\n        resolve(`Error executing command: ${err.message}`);\n      });\n    });\n  }\n}"]}
|
|
@@ -34,9 +34,9 @@ export declare class SearchFileContentTool extends BaseTool<SearchFileContentPar
|
|
|
34
34
|
};
|
|
35
35
|
};
|
|
36
36
|
required: string[];
|
|
37
|
+
additionalProperties: boolean;
|
|
37
38
|
};
|
|
38
39
|
execute(params: SearchFileContentParams, options?: ToolExecuteOptions): Promise<string>;
|
|
39
|
-
private
|
|
40
|
-
private shouldInclude;
|
|
40
|
+
private runSearch;
|
|
41
41
|
}
|
|
42
42
|
export {};
|
|
@@ -35,119 +35,141 @@ var __importStar = (this && this.__importStar) || (function () {
|
|
|
35
35
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
36
36
|
exports.SearchFileContentTool = void 0;
|
|
37
37
|
const base_1 = require("./base");
|
|
38
|
-
const fs = __importStar(require("fs
|
|
38
|
+
const fs = __importStar(require("fs"));
|
|
39
39
|
const path = __importStar(require("path"));
|
|
40
40
|
class SearchFileContentTool extends base_1.BaseTool {
|
|
41
41
|
name = 'search_file_content';
|
|
42
|
-
description = 'Searches for a pattern within files
|
|
42
|
+
description = 'Searches for a regular expression pattern within the content of files or directories. Returns the lines containing matches, along with their file paths and line numbers.';
|
|
43
43
|
parameters = {
|
|
44
44
|
type: 'object',
|
|
45
45
|
properties: {
|
|
46
46
|
pattern: {
|
|
47
47
|
type: 'string',
|
|
48
|
-
description: 'The regular expression pattern to search for'
|
|
48
|
+
description: 'The regular expression (regex) pattern to search for within file contents'
|
|
49
49
|
},
|
|
50
50
|
path: {
|
|
51
51
|
type: 'string',
|
|
52
|
-
description: 'Optional: The path to search within'
|
|
52
|
+
description: 'Optional: The absolute path to the file or directory to search within. If omitted, searches the current working directory.'
|
|
53
53
|
},
|
|
54
54
|
include: {
|
|
55
55
|
type: 'string',
|
|
56
|
-
description: 'Optional: A glob pattern to filter files'
|
|
56
|
+
description: 'Optional: A glob pattern to filter which files are searched (e.g., *.js, *.{ts,tsx}, src/**)'
|
|
57
57
|
},
|
|
58
58
|
case_sensitive: {
|
|
59
59
|
type: 'boolean',
|
|
60
|
-
description: 'If true, search is case-sensitive'
|
|
60
|
+
description: 'If true, search is case-sensitive. Defaults to false.'
|
|
61
61
|
},
|
|
62
62
|
context: {
|
|
63
63
|
type: 'number',
|
|
64
|
-
description: '
|
|
64
|
+
description: 'Show this many lines of context around each match. Defaults to 0.'
|
|
65
65
|
}
|
|
66
66
|
},
|
|
67
|
-
required: ['pattern']
|
|
67
|
+
required: ['pattern'],
|
|
68
|
+
additionalProperties: false
|
|
68
69
|
};
|
|
69
70
|
async execute(params, options) {
|
|
70
71
|
this.validateRequired(params, ['pattern']);
|
|
71
72
|
const workingDir = options?.workingDirectory || process.cwd();
|
|
72
73
|
const searchPath = params.path || workingDir;
|
|
73
|
-
const caseSensitive = params.case_sensitive ?? false;
|
|
74
74
|
try {
|
|
75
|
-
const
|
|
76
|
-
|
|
77
|
-
const maxResults = 500;
|
|
78
|
-
await this.searchInDir(searchPath, regex, params.include, results, maxResults);
|
|
79
|
-
if (results.length === 0) {
|
|
80
|
-
return `No matches found for pattern: ${params.pattern}`;
|
|
81
|
-
}
|
|
82
|
-
const output = results.map(r => {
|
|
83
|
-
return `${r.file}:${r.line}: ${r.content}`;
|
|
84
|
-
}).join('\n');
|
|
85
|
-
return `Found ${results.length} matches:\n\n${output}`;
|
|
75
|
+
const results = await this.runSearch(params, searchPath);
|
|
76
|
+
return results;
|
|
86
77
|
}
|
|
87
78
|
catch (error) {
|
|
88
79
|
return (0, base_1.formatToolError)(error);
|
|
89
80
|
}
|
|
90
81
|
}
|
|
91
|
-
async
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
82
|
+
async runSearch(params, searchPath) {
|
|
83
|
+
const isWindows = process.platform === 'win32';
|
|
84
|
+
const pattern = params.case_sensitive
|
|
85
|
+
? new RegExp(params.pattern, 'g')
|
|
86
|
+
: new RegExp(params.pattern, 'gi');
|
|
87
|
+
const results = [];
|
|
88
|
+
const contextLines = params.context || 0;
|
|
89
|
+
const searchFile = (filePath) => {
|
|
90
|
+
try {
|
|
91
|
+
const content = fs.readFileSync(filePath, 'utf-8');
|
|
92
|
+
const lines = content.split('\n');
|
|
93
|
+
for (let i = 0; i < lines.length; i++) {
|
|
94
|
+
const line = lines[i];
|
|
95
|
+
if (pattern.test(line)) {
|
|
96
|
+
// Add context before
|
|
97
|
+
if (contextLines > 0) {
|
|
98
|
+
for (let j = Math.max(0, i - contextLines); j < i; j++) {
|
|
99
|
+
results.push(` ${filePath}:${j + 1}: ${lines[j]}`);
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
// Add matching line
|
|
103
|
+
results.push(`> ${filePath}:${i + 1}: ${line}`);
|
|
104
|
+
// Add context after
|
|
105
|
+
if (contextLines > 0) {
|
|
106
|
+
for (let j = i + 1; j <= Math.min(lines.length - 1, i + contextLines); j++) {
|
|
107
|
+
results.push(` ${filePath}:${j + 1}: ${lines[j]}`);
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
results.push(''); // Empty line between matches
|
|
111
|
+
}
|
|
102
112
|
}
|
|
103
|
-
await this.searchInDir(fullPath, regex, include, results, maxResults);
|
|
104
113
|
}
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
if (
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
match: match[0]
|
|
122
|
-
});
|
|
114
|
+
catch {
|
|
115
|
+
// Skip files that can't be read as text
|
|
116
|
+
}
|
|
117
|
+
};
|
|
118
|
+
const searchDir = (dirPath) => {
|
|
119
|
+
try {
|
|
120
|
+
const entries = fs.readdirSync(dirPath, { withFileTypes: true });
|
|
121
|
+
for (const entry of entries) {
|
|
122
|
+
const fullPath = path.join(dirPath, entry.name);
|
|
123
|
+
if (entry.isDirectory()) {
|
|
124
|
+
// Skip hidden directories and common ignore patterns
|
|
125
|
+
if (!entry.name.startsWith('.') &&
|
|
126
|
+
entry.name !== 'node_modules' &&
|
|
127
|
+
entry.name !== 'dist' &&
|
|
128
|
+
entry.name !== 'build') {
|
|
129
|
+
searchDir(fullPath);
|
|
123
130
|
}
|
|
124
|
-
}
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
131
|
+
}
|
|
132
|
+
else if (entry.isFile()) {
|
|
133
|
+
// Apply include filter if provided
|
|
134
|
+
if (params.include) {
|
|
135
|
+
const minimatch = require('minimatch');
|
|
136
|
+
if (!minimatch(entry.name, params.include)) {
|
|
137
|
+
continue;
|
|
138
|
+
}
|
|
139
|
+
}
|
|
140
|
+
searchFile(fullPath);
|
|
141
|
+
}
|
|
128
142
|
}
|
|
129
143
|
}
|
|
144
|
+
catch {
|
|
145
|
+
// Skip directories that can't be accessed
|
|
146
|
+
}
|
|
147
|
+
};
|
|
148
|
+
try {
|
|
149
|
+
const stat = fs.statSync(searchPath);
|
|
150
|
+
if (stat.isDirectory()) {
|
|
151
|
+
searchDir(searchPath);
|
|
152
|
+
}
|
|
153
|
+
else {
|
|
154
|
+
searchFile(searchPath);
|
|
155
|
+
}
|
|
130
156
|
}
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
const textExts = [
|
|
137
|
-
'.ts', '.tsx', '.js', '.jsx', '.json', '.md', '.txt',
|
|
138
|
-
'.py', '.java', '.c', '.cpp', '.h', '.go', '.rs',
|
|
139
|
-
'.yaml', '.yml', '.toml', '.ini', '.env', '.sh',
|
|
140
|
-
'.html', '.css', '.scss', '.less', '.vue', '.svelte'
|
|
141
|
-
];
|
|
142
|
-
return textExts.includes(ext) || ext === '';
|
|
157
|
+
catch (error) {
|
|
158
|
+
return `Error: Could not access ${searchPath}: ${error}`;
|
|
159
|
+
}
|
|
160
|
+
if (results.length === 0) {
|
|
161
|
+
return `No matches found for pattern "${params.pattern}" in ${searchPath}`;
|
|
143
162
|
}
|
|
144
|
-
//
|
|
145
|
-
|
|
146
|
-
|
|
147
|
-
|
|
163
|
+
// Limit results to prevent overwhelming output
|
|
164
|
+
const maxResults = 500;
|
|
165
|
+
const limitedResults = results.slice(0, maxResults);
|
|
166
|
+
let output = `Found ${results.length} matches for "${params.pattern}":\n\n`;
|
|
167
|
+
output += limitedResults.join('\n');
|
|
168
|
+
if (results.length > maxResults) {
|
|
169
|
+
output += `\n\n... and ${results.length - maxResults} more matches (truncated)`;
|
|
148
170
|
}
|
|
149
|
-
return
|
|
171
|
+
return output;
|
|
150
172
|
}
|
|
151
173
|
}
|
|
152
174
|
exports.SearchFileContentTool = SearchFileContentTool;
|
|
153
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"search-file-content.js","sourceRoot":"","sources":["../../src/tools/search-file-content.ts"],"names":[],"mappings":"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","sourcesContent":["import { BaseTool, ToolExecuteOptions, formatToolError } from './base';\nimport * as fs from 'fs/promises';\nimport * as path from 'path';\n\ninterface SearchFileContentParams {\n  pattern: string;\n  path?: string;\n  include?: string;\n  case_sensitive?: boolean;\n  context?: number;\n}\n\ninterface SearchResult {\n  file: string;\n  line: number;\n  content: string;\n  match: string;\n}\n\nexport class SearchFileContentTool extends BaseTool<SearchFileContentParams> {\n  name = 'search_file_content';\n  description = 'Searches for a pattern within files in a directory using regex.';\n  parameters = {\n    type: 'object',\n    properties: {\n      pattern: {\n        type: 'string',\n        description: 'The regular expression pattern to search for'\n      },\n      path: {\n        type: 'string',\n        description: 'Optional: The path to search within'\n      },\n      include: {\n        type: 'string',\n        description: 'Optional: A glob pattern to filter files'\n      },\n      case_sensitive: {\n        type: 'boolean',\n        description: 'If true, search is case-sensitive'\n      },\n      context: {\n        type: 'number',\n        description: 'Number of lines of context around matches'\n      }\n    },\n    required: ['pattern']\n  };\n\n  async execute(params: SearchFileContentParams, options?: ToolExecuteOptions): Promise<string> {\n    this.validateRequired(params, ['pattern']);\n\n    const workingDir = options?.workingDirectory || process.cwd();\n    const searchPath = params.path || workingDir;\n    const caseSensitive = params.case_sensitive ?? false;\n\n    try {\n      const regex = new RegExp(params.pattern, caseSensitive ? 'g' : 'gi');\n      const results: SearchResult[] = [];\n      const maxResults = 500;\n\n      await this.searchInDir(searchPath, regex, params.include, results, maxResults);\n\n      if (results.length === 0) {\n        return `No matches found for pattern: ${params.pattern}`;\n      }\n\n      const output = results.map(r => {\n        return `${r.file}:${r.line}: ${r.content}`;\n      }).join('\\n');\n\n      return `Found ${results.length} matches:\\n\\n${output}`;\n    } catch (error) {\n      return formatToolError(error);\n    }\n  }\n\n  private async searchInDir(\n    dir: string,\n    regex: RegExp,\n    include: string | undefined,\n    results: SearchResult[],\n    maxResults: number\n  ): Promise<void> {\n    if (results.length >= maxResults) return;\n\n    const entries = await fs.readdir(dir, { withFileTypes: true });\n\n    for (const entry of entries) {\n      if (results.length >= maxResults) return;\n\n      const fullPath = path.join(dir, entry.name);\n\n      if (entry.isDirectory()) {\n        if (['node_modules', '.git', 'dist', 'build', '.next'].includes(entry.name)) {\n          continue;\n        }\n        await this.searchInDir(fullPath, regex, include, results, maxResults);\n      } else if (entry.isFile()) {\n        if (!this.shouldInclude(fullPath, include)) {\n          continue;\n        }\n\n        try {\n          const content = await fs.readFile(fullPath, 'utf-8');\n          const lines = content.split('\\n');\n\n          lines.forEach((line, index) => {\n            if (results.length >= maxResults) return;\n\n            const match = line.match(regex);\n            if (match) {\n              results.push({\n                file: fullPath,\n                line: index + 1,\n                content: line.trim(),\n                match: match[0]\n              });\n            }\n          });\n        } catch {\n          // Skip files that can't be read as text\n        }\n      }\n    }\n  }\n\n  private shouldInclude(filePath: string, include: string | undefined): boolean {\n    if (!include) {\n      // Default: only text files\n      const ext = path.extname(filePath).toLowerCase();\n      const textExts = [\n        '.ts', '.tsx', '.js', '.jsx', '.json', '.md', '.txt',\n        '.py', '.java', '.c', '.cpp', '.h', '.go', '.rs',\n        '.yaml', '.yml', '.toml', '.ini', '.env', '.sh',\n        '.html', '.css', '.scss', '.less', '.vue', '.svelte'\n      ];\n      return textExts.includes(ext) || ext === '';\n    }\n\n    // Simple include pattern matching\n    if (include.startsWith('*.')) {\n      const ext = include.slice(1);\n      return filePath.endsWith(ext);\n    }\n    return filePath.includes(include);\n  }\n}\n"]}
|
|
175
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"search-file-content.js","sourceRoot":"","sources":["../../src/tools/search-file-content.ts"],"names":[],"mappings":"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","sourcesContent":["import { BaseTool, ToolExecuteOptions, formatToolError } from './base';\nimport { spawn } from 'child_process';\nimport * as fs from 'fs';\nimport * as path from 'path';\n\ninterface SearchFileContentParams {\n  pattern: string;\n  path?: string;\n  include?: string;\n  case_sensitive?: boolean;\n  context?: number;\n}\n\nexport class SearchFileContentTool extends BaseTool<SearchFileContentParams> {\n  name = 'search_file_content';\n  description = 'Searches for a regular expression pattern within the content of files or directories. Returns the lines containing matches, along with their file paths and line numbers.';\n  parameters = {\n    type: 'object',\n    properties: {\n      pattern: {\n        type: 'string',\n        description: 'The regular expression (regex) pattern to search for within file contents'\n      },\n      path: {\n        type: 'string',\n        description: 'Optional: The absolute path to the file or directory to search within. If omitted, searches the current working directory.'\n      },\n      include: {\n        type: 'string',\n        description: 'Optional: A glob pattern to filter which files are searched (e.g., *.js, *.{ts,tsx}, src/**)'\n      },\n      case_sensitive: {\n        type: 'boolean',\n        description: 'If true, search is case-sensitive. Defaults to false.'\n      },\n      context: {\n        type: 'number',\n        description: 'Show this many lines of context around each match. Defaults to 0.'\n      }\n    },\n    required: ['pattern'],\n    additionalProperties: false\n  };\n\n  async execute(params: SearchFileContentParams, options?: ToolExecuteOptions): Promise<string> {\n    this.validateRequired(params, ['pattern']);\n\n    const workingDir = options?.workingDirectory || process.cwd();\n    const searchPath = params.path || workingDir;\n\n    try {\n      const results = await this.runSearch(params, searchPath);\n      return results;\n    } catch (error) {\n      return formatToolError(error);\n    }\n  }\n\n  private async runSearch(params: SearchFileContentParams, searchPath: string): Promise<string> {\n    const isWindows = process.platform === 'win32';\n    const pattern = params.case_sensitive \n      ? new RegExp(params.pattern, 'g')\n      : new RegExp(params.pattern, 'gi');\n\n    const results: string[] = [];\n    const contextLines = params.context || 0;\n\n    const searchFile = (filePath: string): void => {\n      try {\n        const content = fs.readFileSync(filePath, 'utf-8');\n        const lines = content.split('\\n');\n\n        for (let i = 0; i < lines.length; i++) {\n          const line = lines[i];\n          if (pattern.test(line)) {\n            // Add context before\n            if (contextLines > 0) {\n              for (let j = Math.max(0, i - contextLines); j < i; j++) {\n                results.push(`  ${filePath}:${j + 1}: ${lines[j]}`);\n              }\n            }\n            // Add matching line\n            results.push(`> ${filePath}:${i + 1}: ${line}`);\n            // Add context after\n            if (contextLines > 0) {\n              for (let j = i + 1; j <= Math.min(lines.length - 1, i + contextLines); j++) {\n                results.push(`  ${filePath}:${j + 1}: ${lines[j]}`);\n              }\n            }\n            results.push(''); // Empty line between matches\n          }\n        }\n      } catch {\n        // Skip files that can't be read as text\n      }\n    };\n\n    const searchDir = (dirPath: string): void => {\n      try {\n        const entries = fs.readdirSync(dirPath, { withFileTypes: true });\n\n        for (const entry of entries) {\n          const fullPath = path.join(dirPath, entry.name);\n\n          if (entry.isDirectory()) {\n            // Skip hidden directories and common ignore patterns\n            if (!entry.name.startsWith('.') && \n                entry.name !== 'node_modules' && \n                entry.name !== 'dist' &&\n                entry.name !== 'build') {\n              searchDir(fullPath);\n            }\n          } else if (entry.isFile()) {\n            // Apply include filter if provided\n            if (params.include) {\n              const minimatch = require('minimatch');\n              if (!minimatch(entry.name, params.include)) {\n                continue;\n              }\n            }\n            searchFile(fullPath);\n          }\n        }\n      } catch {\n        // Skip directories that can't be accessed\n      }\n    };\n\n    try {\n      const stat = fs.statSync(searchPath);\n      if (stat.isDirectory()) {\n        searchDir(searchPath);\n      } else {\n        searchFile(searchPath);\n      }\n    } catch (error) {\n      return `Error: Could not access ${searchPath}: ${error}`;\n    }\n\n    if (results.length === 0) {\n      return `No matches found for pattern \"${params.pattern}\" in ${searchPath}`;\n    }\n\n    // Limit results to prevent overwhelming output\n    const maxResults = 500;\n    const limitedResults = results.slice(0, maxResults);\n\n    let output = `Found ${results.length} matches for \"${params.pattern}\":\\n\\n`;\n    output += limitedResults.join('\\n');\n\n    if (results.length > maxResults) {\n      output += `\\n\\n... and ${results.length - maxResults} more matches (truncated)`;\n    }\n\n    return output;\n  }\n}\n"]}
|