mirror of
https://github.com/google-gemini/gemini-cli.git
synced 2026-03-18 10:01:29 -07:00
156 lines
4.8 KiB
TypeScript
156 lines
4.8 KiB
TypeScript
/**
|
|
* @license
|
|
* Copyright 2025 Google LLC
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
|
|
import { useEffect } from 'react';
|
|
import { type Config } from '@google/gemini-cli-core';
|
|
import { loadTrustedFolders } from '../../config/trustedFolders.js';
|
|
import { expandHomeDir, batchAddDirectories } from '../utils/directoryUtils.js';
|
|
import {
|
|
debugLogger,
|
|
refreshServerHierarchicalMemory,
|
|
} from '@google/gemini-cli-core';
|
|
import { MultiFolderTrustDialog } from '../components/MultiFolderTrustDialog.js';
|
|
import type { UseHistoryManagerReturn } from './useHistoryManager.js';
|
|
import { MessageType, type HistoryItem } from '../types.js';
|
|
|
|
async function finishAddingDirectories(
|
|
config: Config,
|
|
addItem: (
|
|
itemData: Omit<HistoryItem, 'id'>,
|
|
baseTimestamp?: number,
|
|
) => number,
|
|
added: string[],
|
|
errors: string[],
|
|
) {
|
|
if (!config) {
|
|
addItem({
|
|
type: MessageType.ERROR,
|
|
text: 'Configuration is not available.',
|
|
});
|
|
return;
|
|
}
|
|
|
|
try {
|
|
if (config.shouldLoadMemoryFromIncludeDirectories()) {
|
|
await refreshServerHierarchicalMemory(config);
|
|
}
|
|
} catch (error) {
|
|
// eslint-disable-next-line @typescript-eslint/no-unsafe-type-assertion
|
|
errors.push(`Error refreshing memory: ${(error as Error).message}`);
|
|
}
|
|
|
|
if (added.length > 0) {
|
|
const gemini = config.getGeminiClient();
|
|
if (gemini) {
|
|
await gemini.addDirectoryContext();
|
|
}
|
|
}
|
|
|
|
if (errors.length > 0) {
|
|
addItem({ type: MessageType.ERROR, text: errors.join('\n') });
|
|
}
|
|
}
|
|
|
|
export function useIncludeDirsTrust(
|
|
config: Config,
|
|
isTrustedFolder: boolean | undefined,
|
|
historyManager: UseHistoryManagerReturn,
|
|
setCustomDialog: (dialog: React.ReactNode | null) => void,
|
|
) {
|
|
const { addItem } = historyManager;
|
|
|
|
useEffect(() => {
|
|
// Don't run this until the initial trust is determined.
|
|
if (isTrustedFolder === undefined || !config) {
|
|
return;
|
|
}
|
|
|
|
const pendingDirs = config.getPendingIncludeDirectories();
|
|
if (pendingDirs.length === 0) {
|
|
return;
|
|
}
|
|
|
|
// If folder trust is disabled, isTrustedFolder will be undefined.
|
|
// In that case, or if the user decided not to trust the main folder,
|
|
// we can just add the directories without checking them.
|
|
if (config.getFolderTrust() === false || isTrustedFolder === false) {
|
|
const added: string[] = [];
|
|
const errors: string[] = [];
|
|
const workspaceContext = config.getWorkspaceContext();
|
|
|
|
const result = batchAddDirectories(workspaceContext, pendingDirs);
|
|
added.push(...result.added);
|
|
errors.push(...result.errors);
|
|
|
|
if (added.length > 0 || errors.length > 0) {
|
|
// eslint-disable-next-line @typescript-eslint/no-floating-promises
|
|
finishAddingDirectories(config, addItem, added, errors);
|
|
}
|
|
config.clearPendingIncludeDirectories();
|
|
return;
|
|
}
|
|
|
|
const trustedFolders = loadTrustedFolders();
|
|
const untrustedDirs: string[] = [];
|
|
const undefinedTrustDirs: string[] = [];
|
|
const trustedDirs: string[] = [];
|
|
const added: string[] = [];
|
|
const errors: string[] = [];
|
|
|
|
for (const pathToAdd of pendingDirs) {
|
|
const expandedPath = expandHomeDir(pathToAdd.trim());
|
|
const isTrusted = trustedFolders.isPathTrusted(expandedPath);
|
|
if (isTrusted === false) {
|
|
untrustedDirs.push(pathToAdd.trim());
|
|
} else if (isTrusted === undefined) {
|
|
undefinedTrustDirs.push(pathToAdd.trim());
|
|
} else {
|
|
trustedDirs.push(pathToAdd.trim());
|
|
}
|
|
}
|
|
|
|
if (untrustedDirs.length > 0) {
|
|
errors.push(
|
|
`The following directories are explicitly untrusted and cannot be added to a trusted workspace:\n- ${untrustedDirs.join(
|
|
'\n- ',
|
|
)}\nPlease use the permissions command to modify their trust level.`,
|
|
);
|
|
}
|
|
|
|
const workspaceContext = config.getWorkspaceContext();
|
|
if (trustedDirs.length > 0) {
|
|
const result = batchAddDirectories(workspaceContext, trustedDirs);
|
|
added.push(...result.added);
|
|
errors.push(...result.errors);
|
|
}
|
|
|
|
if (undefinedTrustDirs.length > 0) {
|
|
debugLogger.log(
|
|
'Creating custom dialog with undecidedDirs:',
|
|
undefinedTrustDirs,
|
|
);
|
|
setCustomDialog(
|
|
<MultiFolderTrustDialog
|
|
folders={undefinedTrustDirs}
|
|
onComplete={() => {
|
|
setCustomDialog(null);
|
|
config.clearPendingIncludeDirectories();
|
|
}}
|
|
trustedDirs={added}
|
|
errors={errors}
|
|
finishAddingDirectories={finishAddingDirectories}
|
|
config={config}
|
|
addItem={addItem}
|
|
/>,
|
|
);
|
|
} else if (added.length > 0 || errors.length > 0) {
|
|
// eslint-disable-next-line @typescript-eslint/no-floating-promises
|
|
finishAddingDirectories(config, addItem, added, errors);
|
|
config.clearPendingIncludeDirectories();
|
|
}
|
|
}, [isTrustedFolder, config, addItem, setCustomDialog]);
|
|
}
|