/** * @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[]; gitIgnoredCount: number; geminiIgnoredCount: number; } export class FileDiscoveryService { private gitIgnoreFilter: GitIgnoreFilter | null = null; private geminiIgnoreFilter: GeminiIgnoreFilter | 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); } /** * Filters a list of file paths based on git ignore rules */ filterFiles( filePaths: string[], options: FilterFilesOptions = { respectGitIgnore: true, respectGeminiIgnore: true, }, ): string[] { return filePaths.filter((filePath) => { if (options.respectGitIgnore && this.shouldGitIgnoreFile(filePath)) { return false; } if ( options.respectGeminiIgnore && this.shouldGeminiIgnoreFile(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: string[] = []; let gitIgnoredCount = 0; let geminiIgnoredCount = 0; for (const filePath of filePaths) { if (opts.respectGitIgnore && this.shouldGitIgnoreFile(filePath)) { gitIgnoredCount++; continue; } if (opts.respectGeminiIgnore && this.shouldGeminiIgnoreFile(filePath)) { geminiIgnoredCount++; continue; } filteredPaths.push(filePath); } return { filteredPaths, gitIgnoredCount, geminiIgnoredCount, }; } /** * Checks if a single file should be git-ignored */ shouldGitIgnoreFile(filePath: string): boolean { if (this.gitIgnoreFilter) { return this.gitIgnoreFilter.isIgnored(filePath); } return false; } /** * Checks if a single file should be gemini-ignored */ shouldGeminiIgnoreFile(filePath: string): boolean { if (this.geminiIgnoreFilter) { return this.geminiIgnoreFilter.isIgnored(filePath); } return false; } /** * Unified method to check if a file should be ignored based on filtering options */ shouldIgnoreFile( filePath: string, options: FilterFilesOptions = {}, ): boolean { const { respectGitIgnore = true, respectGeminiIgnore = true } = options; if (respectGitIgnore && this.shouldGitIgnoreFile(filePath)) { return true; } if (respectGeminiIgnore && this.shouldGeminiIgnoreFile(filePath)) { return true; } return false; } /** * Returns loaded patterns from .geminiignore */ getGeminiIgnorePatterns(): string[] { return this.geminiIgnoreFilter?.getPatterns() ?? []; } }