mirror of
https://github.com/google-gemini/gemini-cli.git
synced 2026-03-12 07:01:09 -07:00
101 lines
2.9 KiB
TypeScript
101 lines
2.9 KiB
TypeScript
/**
|
|
* @license
|
|
* Copyright 2025 Google LLC
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
|
|
import type { GitIgnoreFilter } from '../utils/gitIgnoreParser.js';
|
|
import type { GeminiIgnoreFilter } from '../utils/geminiIgnoreParser.js';
|
|
import { GitIgnoreParser } from '../utils/gitIgnoreParser.js';
|
|
import { GeminiIgnoreParser } from '../utils/geminiIgnoreParser.js';
|
|
import { isGitRepository } from '../utils/gitUtils.js';
|
|
import * as path from 'node:path';
|
|
|
|
export interface FilterFilesOptions {
|
|
respectGitIgnore?: boolean;
|
|
respectGeminiIgnore?: boolean;
|
|
}
|
|
|
|
export interface FilterReport {
|
|
filteredPaths: string[];
|
|
ignoredCount: number;
|
|
}
|
|
|
|
export class FileDiscoveryService {
|
|
private gitIgnoreFilter: GitIgnoreFilter | null = null;
|
|
private geminiIgnoreFilter: GeminiIgnoreFilter | null = null;
|
|
private combinedIgnoreFilter: GitIgnoreFilter | null = null;
|
|
private projectRoot: string;
|
|
|
|
constructor(projectRoot: string) {
|
|
this.projectRoot = path.resolve(projectRoot);
|
|
if (isGitRepository(this.projectRoot)) {
|
|
this.gitIgnoreFilter = new GitIgnoreParser(this.projectRoot);
|
|
}
|
|
this.geminiIgnoreFilter = new GeminiIgnoreParser(this.projectRoot);
|
|
|
|
if (this.gitIgnoreFilter) {
|
|
const geminiPatterns = this.geminiIgnoreFilter.getPatterns();
|
|
// Create combined parser: .gitignore + .geminiignore
|
|
this.combinedIgnoreFilter = new GitIgnoreParser(
|
|
this.projectRoot,
|
|
geminiPatterns,
|
|
);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Filters a list of file paths based on git ignore rules
|
|
*/
|
|
filterFiles(filePaths: string[], options: FilterFilesOptions = {}): string[] {
|
|
const { respectGitIgnore = true, respectGeminiIgnore = true } = options;
|
|
return filePaths.filter((filePath) => {
|
|
if (
|
|
respectGitIgnore &&
|
|
respectGeminiIgnore &&
|
|
this.combinedIgnoreFilter
|
|
) {
|
|
return !this.combinedIgnoreFilter.isIgnored(filePath);
|
|
}
|
|
|
|
if (respectGitIgnore && this.gitIgnoreFilter?.isIgnored(filePath)) {
|
|
return false;
|
|
}
|
|
if (respectGeminiIgnore && this.geminiIgnoreFilter?.isIgnored(filePath)) {
|
|
return false;
|
|
}
|
|
return true;
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Filters a list of file paths based on git ignore rules and returns a report
|
|
* with counts of ignored files.
|
|
*/
|
|
filterFilesWithReport(
|
|
filePaths: string[],
|
|
opts: FilterFilesOptions = {
|
|
respectGitIgnore: true,
|
|
respectGeminiIgnore: true,
|
|
},
|
|
): FilterReport {
|
|
const filteredPaths = this.filterFiles(filePaths, opts);
|
|
const ignoredCount = filePaths.length - filteredPaths.length;
|
|
|
|
return {
|
|
filteredPaths,
|
|
ignoredCount,
|
|
};
|
|
}
|
|
|
|
/**
|
|
* Unified method to check if a file should be ignored based on filtering options
|
|
*/
|
|
shouldIgnoreFile(
|
|
filePath: string,
|
|
options: FilterFilesOptions = {},
|
|
): boolean {
|
|
return this.filterFiles([filePath], options).length === 0;
|
|
}
|
|
}
|