File size: 4,271 Bytes
40e575e |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 |
/**
* @license
* Copyright 2025 Google LLC
* SPDX-License-Identifier: Apache-2.0
*/
import { EditorType } from '../utils/editor.js';
import os from 'os';
import path from 'path';
import fs from 'fs';
import * as Diff from 'diff';
import { openDiff } from '../utils/editor.js';
import { DEFAULT_DIFF_OPTIONS } from './diffOptions.js';
import { isNodeError } from '../utils/errors.js';
import { Tool } from './tools.js';
/**
* A tool that supports a modify operation.
*/
export interface ModifiableTool<ToolParams> extends Tool<ToolParams> {
getModifyContext(abortSignal: AbortSignal): ModifyContext<ToolParams>;
}
export interface ModifyContext<ToolParams> {
getFilePath: (params: ToolParams) => string;
getCurrentContent: (params: ToolParams) => Promise<string>;
getProposedContent: (params: ToolParams) => Promise<string>;
createUpdatedParams: (
oldContent: string,
modifiedProposedContent: string,
originalParams: ToolParams,
) => ToolParams;
}
export interface ModifyResult<ToolParams> {
updatedParams: ToolParams;
updatedDiff: string;
}
export function isModifiableTool<TParams>(
tool: Tool<TParams>,
): tool is ModifiableTool<TParams> {
return 'getModifyContext' in tool;
}
function createTempFilesForModify(
currentContent: string,
proposedContent: string,
file_path: string,
): { oldPath: string; newPath: string } {
const tempDir = os.tmpdir();
const diffDir = path.join(tempDir, 'gemini-cli-tool-modify-diffs');
if (!fs.existsSync(diffDir)) {
fs.mkdirSync(diffDir, { recursive: true });
}
const fileName = path.basename(file_path);
const timestamp = Date.now();
const tempOldPath = path.join(
diffDir,
`gemini-cli-modify-${fileName}-old-${timestamp}`,
);
const tempNewPath = path.join(
diffDir,
`gemini-cli-modify-${fileName}-new-${timestamp}`,
);
fs.writeFileSync(tempOldPath, currentContent, 'utf8');
fs.writeFileSync(tempNewPath, proposedContent, 'utf8');
return { oldPath: tempOldPath, newPath: tempNewPath };
}
function getUpdatedParams<ToolParams>(
tmpOldPath: string,
tempNewPath: string,
originalParams: ToolParams,
modifyContext: ModifyContext<ToolParams>,
): { updatedParams: ToolParams; updatedDiff: string } {
let oldContent = '';
let newContent = '';
try {
oldContent = fs.readFileSync(tmpOldPath, 'utf8');
} catch (err) {
if (!isNodeError(err) || err.code !== 'ENOENT') throw err;
oldContent = '';
}
try {
newContent = fs.readFileSync(tempNewPath, 'utf8');
} catch (err) {
if (!isNodeError(err) || err.code !== 'ENOENT') throw err;
newContent = '';
}
const updatedParams = modifyContext.createUpdatedParams(
oldContent,
newContent,
originalParams,
);
const updatedDiff = Diff.createPatch(
path.basename(modifyContext.getFilePath(originalParams)),
oldContent,
newContent,
'Current',
'Proposed',
DEFAULT_DIFF_OPTIONS,
);
return { updatedParams, updatedDiff };
}
function deleteTempFiles(oldPath: string, newPath: string): void {
try {
fs.unlinkSync(oldPath);
} catch {
console.error(`Error deleting temp diff file: ${oldPath}`);
}
try {
fs.unlinkSync(newPath);
} catch {
console.error(`Error deleting temp diff file: ${newPath}`);
}
}
/**
* Triggers an external editor for the user to modify the proposed content,
* and returns the updated tool parameters and the diff after the user has modified the proposed content.
*/
export async function modifyWithEditor<ToolParams>(
originalParams: ToolParams,
modifyContext: ModifyContext<ToolParams>,
editorType: EditorType,
_abortSignal: AbortSignal,
): Promise<ModifyResult<ToolParams>> {
const currentContent = await modifyContext.getCurrentContent(originalParams);
const proposedContent =
await modifyContext.getProposedContent(originalParams);
const { oldPath, newPath } = createTempFilesForModify(
currentContent,
proposedContent,
modifyContext.getFilePath(originalParams),
);
try {
await openDiff(oldPath, newPath, editorType);
const result = getUpdatedParams(
oldPath,
newPath,
originalParams,
modifyContext,
);
return result;
} finally {
deleteTempFiles(oldPath, newPath);
}
}
|