Files
gemini-cli/packages/cli/src/ui/hooks/useIncludeDirsTrust.tsx
2026-02-10 00:10:15 +00:00

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]);
}