summaryrefslogtreecommitdiff
path: root/packages/core/src
diff options
context:
space:
mode:
Diffstat (limited to 'packages/core/src')
-rw-r--r--packages/core/src/utils/memoryDiscovery.test.ts707
1 files changed, 233 insertions, 474 deletions
diff --git a/packages/core/src/utils/memoryDiscovery.test.ts b/packages/core/src/utils/memoryDiscovery.test.ts
index 1e38d972..3051da0e 100644
--- a/packages/core/src/utils/memoryDiscovery.test.ts
+++ b/packages/core/src/utils/memoryDiscovery.test.ts
@@ -4,604 +4,363 @@
* SPDX-License-Identifier: Apache-2.0
*/
-import { vi, describe, it, expect, beforeEach, Mocked } from 'vitest';
+import { vi, describe, it, expect, beforeEach, afterEach } from 'vitest';
import * as fsPromises from 'fs/promises';
-import * as fsSync from 'fs';
-import { Stats, Dirent } from 'fs';
import * as os from 'os';
import * as path from 'path';
import { loadServerHierarchicalMemory } from './memoryDiscovery.js';
import {
GEMINI_CONFIG_DIR,
setGeminiMdFilename,
- getCurrentGeminiMdFilename,
DEFAULT_CONTEXT_FILENAME,
} from '../tools/memoryTool.js';
import { FileDiscoveryService } from '../services/fileDiscoveryService.js';
-const ORIGINAL_GEMINI_MD_FILENAME_CONST_FOR_TEST = DEFAULT_CONTEXT_FILENAME;
-
-// Mock the entire fs/promises module
-vi.mock('fs/promises');
-// Mock the parts of fsSync we might use (like constants or existsSync if needed)
-vi.mock('fs', async (importOriginal) => {
- const actual = await importOriginal<typeof fsSync>();
+vi.mock('os', async (importOriginal) => {
+ const actualOs = await importOriginal<typeof os>();
return {
- ...actual, // Spread actual to get all exports, including Stats and Dirent if they are classes/constructors
- constants: { ...actual.constants }, // Preserve constants
+ ...actualOs,
+ homedir: vi.fn(),
};
});
-vi.mock('os');
describe('loadServerHierarchicalMemory', () => {
- const mockFs = fsPromises as Mocked<typeof fsPromises>;
- const mockOs = os as Mocked<typeof os>;
+ let testRootDir: string;
+ let cwd: string;
+ let projectRoot: string;
+ let homedir: string;
+
+ async function createEmptyDir(fullPath: string) {
+ await fsPromises.mkdir(fullPath, { recursive: true });
+ return fullPath;
+ }
- const CWD = '/test/project/src';
- const PROJECT_ROOT = '/test/project';
- const USER_HOME = '/test/userhome';
+ async function createTestFile(fullPath: string, fileContents: string) {
+ await fsPromises.mkdir(path.dirname(fullPath), { recursive: true });
+ await fsPromises.writeFile(fullPath, fileContents);
+ return path.resolve(testRootDir, fullPath);
+ }
- let GLOBAL_GEMINI_DIR: string;
- let GLOBAL_GEMINI_FILE: string; // Defined in beforeEach
+ beforeEach(async () => {
+ testRootDir = await fsPromises.mkdtemp(
+ path.join(os.tmpdir(), 'folder-structure-test-'),
+ );
- const fileService = new FileDiscoveryService(PROJECT_ROOT);
- beforeEach(() => {
vi.resetAllMocks();
// Set environment variables to indicate test environment
process.env.NODE_ENV = 'test';
process.env.VITEST = 'true';
- setGeminiMdFilename(DEFAULT_CONTEXT_FILENAME); // Use defined const
- mockOs.homedir.mockReturnValue(USER_HOME);
-
- // Define these here to use potentially reset/updated values from imports
- GLOBAL_GEMINI_DIR = path.join(USER_HOME, GEMINI_CONFIG_DIR);
- GLOBAL_GEMINI_FILE = path.join(
- GLOBAL_GEMINI_DIR,
- getCurrentGeminiMdFilename(), // Use current filename
- );
+ projectRoot = await createEmptyDir(path.join(testRootDir, 'project'));
+ cwd = await createEmptyDir(path.join(projectRoot, 'src'));
+ homedir = await createEmptyDir(path.join(testRootDir, 'userhome'));
+ vi.mocked(os.homedir).mockReturnValue(homedir);
+ });
- mockFs.stat.mockRejectedValue(new Error('File not found'));
- mockFs.readdir.mockResolvedValue([]);
- mockFs.readFile.mockRejectedValue(new Error('File not found'));
- mockFs.access.mockRejectedValue(new Error('File not found'));
+ afterEach(async () => {
+ // Some tests set this to a different value.
+ setGeminiMdFilename(DEFAULT_CONTEXT_FILENAME);
+ // Clean up the temporary directory to prevent resource leaks.
+ await fsPromises.rm(testRootDir, { recursive: true, force: true });
});
it('should return empty memory and count if no context files are found', async () => {
- const { memoryContent, fileCount } = await loadServerHierarchicalMemory(
- CWD,
+ const result = await loadServerHierarchicalMemory(
+ cwd,
false,
- fileService,
+ new FileDiscoveryService(projectRoot),
);
- expect(memoryContent).toBe('');
- expect(fileCount).toBe(0);
+
+ expect(result).toEqual({
+ memoryContent: '',
+ fileCount: 0,
+ });
});
it('should load only the global context file if present and others are not (default filename)', async () => {
- const globalDefaultFile = path.join(
- GLOBAL_GEMINI_DIR,
- DEFAULT_CONTEXT_FILENAME,
+ const defaultContextFile = await createTestFile(
+ path.join(homedir, GEMINI_CONFIG_DIR, DEFAULT_CONTEXT_FILENAME),
+ 'default context content',
);
- mockFs.access.mockImplementation(async (p) => {
- if (p === globalDefaultFile) {
- return undefined;
- }
- throw new Error('File not found');
- });
- mockFs.readFile.mockImplementation(async (p) => {
- if (p === globalDefaultFile) {
- return 'Global memory content';
- }
- throw new Error('File not found');
- });
- const { memoryContent, fileCount } = await loadServerHierarchicalMemory(
- CWD,
+ const result = await loadServerHierarchicalMemory(
+ cwd,
false,
- fileService,
+ new FileDiscoveryService(projectRoot),
);
- expect(memoryContent).toBe(
- `--- Context from: ${path.relative(CWD, globalDefaultFile)} ---\nGlobal memory content\n--- End of Context from: ${path.relative(CWD, globalDefaultFile)} ---`,
- );
- expect(fileCount).toBe(1);
- expect(mockFs.readFile).toHaveBeenCalledWith(globalDefaultFile, 'utf-8');
+ expect(result).toEqual({
+ memoryContent: `--- Context from: ${path.relative(cwd, defaultContextFile)} ---
+default context content
+--- End of Context from: ${path.relative(cwd, defaultContextFile)} ---`,
+ fileCount: 1,
+ });
});
it('should load only the global custom context file if present and filename is changed', async () => {
const customFilename = 'CUSTOM_AGENTS.md';
setGeminiMdFilename(customFilename);
- const globalCustomFile = path.join(GLOBAL_GEMINI_DIR, customFilename);
- mockFs.access.mockImplementation(async (p) => {
- if (p === globalCustomFile) {
- return undefined;
- }
- throw new Error('File not found');
- });
- mockFs.readFile.mockImplementation(async (p) => {
- if (p === globalCustomFile) {
- return 'Global custom memory';
- }
- throw new Error('File not found');
- });
+ const customContextFile = await createTestFile(
+ path.join(homedir, GEMINI_CONFIG_DIR, customFilename),
+ 'custom context content',
+ );
- const { memoryContent, fileCount } = await loadServerHierarchicalMemory(
- CWD,
+ const result = await loadServerHierarchicalMemory(
+ cwd,
false,
- fileService,
+ new FileDiscoveryService(projectRoot),
);
- expect(memoryContent).toBe(
- `--- Context from: ${path.relative(CWD, globalCustomFile)} ---\nGlobal custom memory\n--- End of Context from: ${path.relative(CWD, globalCustomFile)} ---`,
- );
- expect(fileCount).toBe(1);
- expect(mockFs.readFile).toHaveBeenCalledWith(globalCustomFile, 'utf-8');
+ expect(result).toEqual({
+ memoryContent: `--- Context from: ${path.relative(cwd, customContextFile)} ---
+custom context content
+--- End of Context from: ${path.relative(cwd, customContextFile)} ---`,
+ fileCount: 1,
+ });
});
it('should load context files by upward traversal with custom filename', async () => {
const customFilename = 'PROJECT_CONTEXT.md';
setGeminiMdFilename(customFilename);
- const projectRootCustomFile = path.join(PROJECT_ROOT, customFilename);
- const srcCustomFile = path.join(CWD, customFilename);
-
- mockFs.stat.mockImplementation(async (p) => {
- if (p === path.join(PROJECT_ROOT, '.git')) {
- return { isDirectory: () => true } as Stats;
- }
- throw new Error('File not found');
- });
-
- mockFs.access.mockImplementation(async (p) => {
- if (p === projectRootCustomFile || p === srcCustomFile) {
- return undefined;
- }
- throw new Error('File not found');
- });
- mockFs.readFile.mockImplementation(async (p) => {
- if (p === projectRootCustomFile) {
- return 'Project root custom memory';
- }
- if (p === srcCustomFile) {
- return 'Src directory custom memory';
- }
- throw new Error('File not found');
- });
+ const projectContextFile = await createTestFile(
+ path.join(projectRoot, customFilename),
+ 'project context content',
+ );
+ const cwdContextFile = await createTestFile(
+ path.join(cwd, customFilename),
+ 'cwd context content',
+ );
- const { memoryContent, fileCount } = await loadServerHierarchicalMemory(
- CWD,
+ const result = await loadServerHierarchicalMemory(
+ cwd,
false,
- fileService,
+ new FileDiscoveryService(projectRoot),
);
- const expectedContent =
- `--- Context from: ${path.relative(CWD, projectRootCustomFile)} ---\nProject root custom memory\n--- End of Context from: ${path.relative(CWD, projectRootCustomFile)} ---\n\n` +
- `--- Context from: ${customFilename} ---\nSrc directory custom memory\n--- End of Context from: ${customFilename} ---`;
- expect(memoryContent).toBe(expectedContent);
- expect(fileCount).toBe(2);
- expect(mockFs.readFile).toHaveBeenCalledWith(
- projectRootCustomFile,
- 'utf-8',
- );
- expect(mockFs.readFile).toHaveBeenCalledWith(srcCustomFile, 'utf-8');
+ expect(result).toEqual({
+ memoryContent: `--- Context from: ${path.relative(cwd, projectContextFile)} ---
+project context content
+--- End of Context from: ${path.relative(cwd, projectContextFile)} ---
+
+--- Context from: ${path.relative(cwd, cwdContextFile)} ---
+cwd context content
+--- End of Context from: ${path.relative(cwd, cwdContextFile)} ---`,
+ fileCount: 2,
+ });
});
it('should load context files by downward traversal with custom filename', async () => {
const customFilename = 'LOCAL_CONTEXT.md';
setGeminiMdFilename(customFilename);
- const subDir = path.join(CWD, 'subdir');
- const subDirCustomFile = path.join(subDir, customFilename);
- const cwdCustomFile = path.join(CWD, customFilename);
-
- mockFs.access.mockImplementation(async (p) => {
- if (p === cwdCustomFile || p === subDirCustomFile) return undefined;
- throw new Error('File not found');
- });
-
- mockFs.readFile.mockImplementation(async (p) => {
- if (p === cwdCustomFile) return 'CWD custom memory';
- if (p === subDirCustomFile) return 'Subdir custom memory';
- throw new Error('File not found');
- });
- mockFs.readdir.mockImplementation((async (
- p: fsSync.PathLike,
- ): Promise<Dirent[]> => {
- if (p === CWD) {
- return [
- {
- name: customFilename,
- isFile: () => true,
- isDirectory: () => false,
- } as Dirent,
- {
- name: 'subdir',
- isFile: () => false,
- isDirectory: () => true,
- } as Dirent,
- ] as Dirent[];
- }
- if (p === subDir) {
- return [
- {
- name: customFilename,
- isFile: () => true,
- isDirectory: () => false,
- } as Dirent,
- ] as Dirent[];
- }
- return [] as Dirent[];
- }) as unknown as typeof fsPromises.readdir);
+ await createTestFile(
+ path.join(cwd, 'subdir', customFilename),
+ 'Subdir custom memory',
+ );
+ await createTestFile(path.join(cwd, customFilename), 'CWD custom memory');
- const { memoryContent, fileCount } = await loadServerHierarchicalMemory(
- CWD,
+ const result = await loadServerHierarchicalMemory(
+ cwd,
false,
- fileService,
+ new FileDiscoveryService(projectRoot),
);
- const expectedContent =
- `--- Context from: ${customFilename} ---\nCWD custom memory\n--- End of Context from: ${customFilename} ---\n\n` +
- `--- Context from: ${path.join('subdir', customFilename)} ---\nSubdir custom memory\n--- End of Context from: ${path.join('subdir', customFilename)} ---`;
- expect(memoryContent).toBe(expectedContent);
- expect(fileCount).toBe(2);
+ expect(result).toEqual({
+ memoryContent: `--- Context from: ${customFilename} ---
+CWD custom memory
+--- End of Context from: ${customFilename} ---
+
+--- Context from: ${path.join('subdir', customFilename)} ---
+Subdir custom memory
+--- End of Context from: ${path.join('subdir', customFilename)} ---`,
+ fileCount: 2,
+ });
});
it('should load ORIGINAL_GEMINI_MD_FILENAME files by upward traversal from CWD to project root', async () => {
- const projectRootGeminiFile = path.join(
- PROJECT_ROOT,
- ORIGINAL_GEMINI_MD_FILENAME_CONST_FOR_TEST,
+ const projectRootGeminiFile = await createTestFile(
+ path.join(projectRoot, DEFAULT_CONTEXT_FILENAME),
+ 'Project root memory',
);
- const srcGeminiFile = path.join(
- CWD,
- ORIGINAL_GEMINI_MD_FILENAME_CONST_FOR_TEST,
+ const srcGeminiFile = await createTestFile(
+ path.join(cwd, DEFAULT_CONTEXT_FILENAME),
+ 'Src directory memory',
);
- mockFs.stat.mockImplementation(async (p) => {
- if (p === path.join(PROJECT_ROOT, '.git')) {
- return { isDirectory: () => true } as Stats;
- }
- throw new Error('File not found');
- });
-
- mockFs.access.mockImplementation(async (p) => {
- if (p === projectRootGeminiFile || p === srcGeminiFile) {
- return undefined;
- }
- throw new Error('File not found');
- });
-
- mockFs.readFile.mockImplementation(async (p) => {
- if (p === projectRootGeminiFile) {
- return 'Project root memory';
- }
- if (p === srcGeminiFile) {
- return 'Src directory memory';
- }
- throw new Error('File not found');
- });
-
- const { memoryContent, fileCount } = await loadServerHierarchicalMemory(
- CWD,
+ const result = await loadServerHierarchicalMemory(
+ cwd,
false,
- fileService,
+ new FileDiscoveryService(projectRoot),
);
- const expectedContent =
- `--- Context from: ${path.relative(CWD, projectRootGeminiFile)} ---\nProject root memory\n--- End of Context from: ${path.relative(CWD, projectRootGeminiFile)} ---\n\n` +
- `--- Context from: ${ORIGINAL_GEMINI_MD_FILENAME_CONST_FOR_TEST} ---\nSrc directory memory\n--- End of Context from: ${ORIGINAL_GEMINI_MD_FILENAME_CONST_FOR_TEST} ---`;
- expect(memoryContent).toBe(expectedContent);
- expect(fileCount).toBe(2);
- expect(mockFs.readFile).toHaveBeenCalledWith(
- projectRootGeminiFile,
- 'utf-8',
- );
- expect(mockFs.readFile).toHaveBeenCalledWith(srcGeminiFile, 'utf-8');
+ expect(result).toEqual({
+ memoryContent: `--- Context from: ${path.relative(cwd, projectRootGeminiFile)} ---
+Project root memory
+--- End of Context from: ${path.relative(cwd, projectRootGeminiFile)} ---
+
+--- Context from: ${path.relative(cwd, srcGeminiFile)} ---
+Src directory memory
+--- End of Context from: ${path.relative(cwd, srcGeminiFile)} ---`,
+ fileCount: 2,
+ });
});
it('should load ORIGINAL_GEMINI_MD_FILENAME files by downward traversal from CWD', async () => {
- const subDir = path.join(CWD, 'subdir');
- const subDirGeminiFile = path.join(
- subDir,
- ORIGINAL_GEMINI_MD_FILENAME_CONST_FOR_TEST,
+ await createTestFile(
+ path.join(cwd, 'subdir', DEFAULT_CONTEXT_FILENAME),
+ 'Subdir memory',
);
- const cwdGeminiFile = path.join(
- CWD,
- ORIGINAL_GEMINI_MD_FILENAME_CONST_FOR_TEST,
+ await createTestFile(
+ path.join(cwd, DEFAULT_CONTEXT_FILENAME),
+ 'CWD memory',
);
- mockFs.access.mockImplementation(async (p) => {
- if (p === cwdGeminiFile || p === subDirGeminiFile) return undefined;
- throw new Error('File not found');
- });
-
- mockFs.readFile.mockImplementation(async (p) => {
- if (p === cwdGeminiFile) return 'CWD memory';
- if (p === subDirGeminiFile) return 'Subdir memory';
- throw new Error('File not found');
- });
-
- mockFs.readdir.mockImplementation((async (
- p: fsSync.PathLike,
- ): Promise<Dirent[]> => {
- if (p === CWD) {
- return [
- {
- name: ORIGINAL_GEMINI_MD_FILENAME_CONST_FOR_TEST,
- isFile: () => true,
- isDirectory: () => false,
- } as Dirent,
- {
- name: 'subdir',
- isFile: () => false,
- isDirectory: () => true,
- } as Dirent,
- ] as Dirent[];
- }
- if (p === subDir) {
- return [
- {
- name: ORIGINAL_GEMINI_MD_FILENAME_CONST_FOR_TEST,
- isFile: () => true,
- isDirectory: () => false,
- } as Dirent,
- ] as Dirent[];
- }
- return [] as Dirent[];
- }) as unknown as typeof fsPromises.readdir);
-
- const { memoryContent, fileCount } = await loadServerHierarchicalMemory(
- CWD,
+ const result = await loadServerHierarchicalMemory(
+ cwd,
false,
- fileService,
+ new FileDiscoveryService(projectRoot),
);
- const expectedContent =
- `--- Context from: ${ORIGINAL_GEMINI_MD_FILENAME_CONST_FOR_TEST} ---\nCWD memory\n--- End of Context from: ${ORIGINAL_GEMINI_MD_FILENAME_CONST_FOR_TEST} ---\n\n` +
- `--- Context from: ${path.join('subdir', ORIGINAL_GEMINI_MD_FILENAME_CONST_FOR_TEST)} ---\nSubdir memory\n--- End of Context from: ${path.join('subdir', ORIGINAL_GEMINI_MD_FILENAME_CONST_FOR_TEST)} ---`;
- expect(memoryContent).toBe(expectedContent);
- expect(fileCount).toBe(2);
+ expect(result).toEqual({
+ memoryContent: `--- Context from: ${DEFAULT_CONTEXT_FILENAME} ---
+CWD memory
+--- End of Context from: ${DEFAULT_CONTEXT_FILENAME} ---
+
+--- Context from: ${path.join('subdir', DEFAULT_CONTEXT_FILENAME)} ---
+Subdir memory
+--- End of Context from: ${path.join('subdir', DEFAULT_CONTEXT_FILENAME)} ---`,
+ fileCount: 2,
+ });
});
it('should load and correctly order global, upward, and downward ORIGINAL_GEMINI_MD_FILENAME files', async () => {
- setGeminiMdFilename(ORIGINAL_GEMINI_MD_FILENAME_CONST_FOR_TEST); // Explicitly set for this test
-
- const globalFileToUse = path.join(
- GLOBAL_GEMINI_DIR,
- ORIGINAL_GEMINI_MD_FILENAME_CONST_FOR_TEST,
+ const defaultContextFile = await createTestFile(
+ path.join(homedir, GEMINI_CONFIG_DIR, DEFAULT_CONTEXT_FILENAME),
+ 'default context content',
);
- const projectParentDir = path.dirname(PROJECT_ROOT);
- const projectParentGeminiFile = path.join(
- projectParentDir,
- ORIGINAL_GEMINI_MD_FILENAME_CONST_FOR_TEST,
+ const rootGeminiFile = await createTestFile(
+ path.join(testRootDir, DEFAULT_CONTEXT_FILENAME),
+ 'Project parent memory',
);
- const projectRootGeminiFile = path.join(
- PROJECT_ROOT,
- ORIGINAL_GEMINI_MD_FILENAME_CONST_FOR_TEST,
+ const projectRootGeminiFile = await createTestFile(
+ path.join(projectRoot, DEFAULT_CONTEXT_FILENAME),
+ 'Project root memory',
);
- const cwdGeminiFile = path.join(
- CWD,
- ORIGINAL_GEMINI_MD_FILENAME_CONST_FOR_TEST,
+ const cwdGeminiFile = await createTestFile(
+ path.join(cwd, DEFAULT_CONTEXT_FILENAME),
+ 'CWD memory',
);
- const subDir = path.join(CWD, 'sub');
- const subDirGeminiFile = path.join(
- subDir,
- ORIGINAL_GEMINI_MD_FILENAME_CONST_FOR_TEST,
+ const subDirGeminiFile = await createTestFile(
+ path.join(cwd, 'sub', DEFAULT_CONTEXT_FILENAME),
+ 'Subdir memory',
);
- mockFs.stat.mockImplementation(async (p) => {
- if (p === path.join(PROJECT_ROOT, '.git')) {
- return { isDirectory: () => true } as Stats;
- } else if (p === path.join(PROJECT_ROOT, '.gemini')) {
- return { isDirectory: () => true } as Stats;
- }
- throw new Error('File not found');
- });
-
- mockFs.access.mockImplementation(async (p) => {
- if (
- p === globalFileToUse || // Use the dynamically set global file path
- p === projectParentGeminiFile ||
- p === projectRootGeminiFile ||
- p === cwdGeminiFile ||
- p === subDirGeminiFile
- ) {
- return undefined;
- }
- throw new Error('File not found');
- });
-
- mockFs.readFile.mockImplementation(async (p) => {
- if (p === globalFileToUse) return 'Global memory'; // Use the dynamically set global file path
- if (p === projectParentGeminiFile) return 'Project parent memory';
- if (p === projectRootGeminiFile) return 'Project root memory';
- if (p === cwdGeminiFile) return 'CWD memory';
- if (p === subDirGeminiFile) return 'Subdir memory';
- throw new Error('File not found');
- });
-
- mockFs.readdir.mockImplementation((async (
- p: fsSync.PathLike,
- ): Promise<Dirent[]> => {
- if (p === CWD) {
- return [
- {
- name: 'sub',
- isFile: () => false,
- isDirectory: () => true,
- } as Dirent,
- ] as Dirent[];
- }
- if (p === subDir) {
- return [
- {
- name: ORIGINAL_GEMINI_MD_FILENAME_CONST_FOR_TEST,
- isFile: () => true,
- isDirectory: () => false,
- } as Dirent,
- ] as Dirent[];
- }
- return [] as Dirent[];
- }) as unknown as typeof fsPromises.readdir);
-
- const { memoryContent, fileCount } = await loadServerHierarchicalMemory(
- CWD,
+ const result = await loadServerHierarchicalMemory(
+ cwd,
false,
- fileService,
+ new FileDiscoveryService(projectRoot),
);
- const relPathGlobal = path.relative(CWD, GLOBAL_GEMINI_FILE);
- const relPathProjectParent = path.relative(CWD, projectParentGeminiFile);
- const relPathProjectRoot = path.relative(CWD, projectRootGeminiFile);
- const relPathCwd = ORIGINAL_GEMINI_MD_FILENAME_CONST_FOR_TEST;
- const relPathSubDir = path.join(
- 'sub',
- ORIGINAL_GEMINI_MD_FILENAME_CONST_FOR_TEST,
- );
+ expect(result).toEqual({
+ memoryContent: `--- Context from: ${path.relative(cwd, defaultContextFile)} ---
+default context content
+--- End of Context from: ${path.relative(cwd, defaultContextFile)} ---
- const expectedContent = [
- `--- Context from: ${relPathGlobal} ---\nGlobal memory\n--- End of Context from: ${relPathGlobal} ---`,
- `--- Context from: ${relPathProjectParent} ---\nProject parent memory\n--- End of Context from: ${relPathProjectParent} ---`,
- `--- Context from: ${relPathProjectRoot} ---\nProject root memory\n--- End of Context from: ${relPathProjectRoot} ---`,
- `--- Context from: ${relPathCwd} ---\nCWD memory\n--- End of Context from: ${relPathCwd} ---`,
- `--- Context from: ${relPathSubDir} ---\nSubdir memory\n--- End of Context from: ${relPathSubDir} ---`,
- ].join('\n\n');
+--- Context from: ${path.relative(cwd, rootGeminiFile)} ---
+Project parent memory
+--- End of Context from: ${path.relative(cwd, rootGeminiFile)} ---
- expect(memoryContent).toBe(expectedContent);
- expect(fileCount).toBe(5);
- });
+--- Context from: ${path.relative(cwd, projectRootGeminiFile)} ---
+Project root memory
+--- End of Context from: ${path.relative(cwd, projectRootGeminiFile)} ---
- it('should ignore specified directories during downward scan', async () => {
- const ignoredDir = path.join(CWD, 'node_modules');
- const ignoredDirGeminiFile = path.join(
- ignoredDir,
- ORIGINAL_GEMINI_MD_FILENAME_CONST_FOR_TEST,
- ); // Corrected
- const regularSubDir = path.join(CWD, 'my_code');
- const regularSubDirGeminiFile = path.join(
- regularSubDir,
- ORIGINAL_GEMINI_MD_FILENAME_CONST_FOR_TEST,
- );
+--- Context from: ${path.relative(cwd, cwdGeminiFile)} ---
+CWD memory
+--- End of Context from: ${path.relative(cwd, cwdGeminiFile)} ---
- mockFs.access.mockImplementation(async (p) => {
- if (p === regularSubDirGeminiFile) return undefined;
- if (p === ignoredDirGeminiFile)
- throw new Error('Should not access ignored file');
- throw new Error('File not found');
+--- Context from: ${path.relative(cwd, subDirGeminiFile)} ---
+Subdir memory
+--- End of Context from: ${path.relative(cwd, subDirGeminiFile)} ---`,
+ fileCount: 5,
});
+ });
- mockFs.readFile.mockImplementation(async (p) => {
- if (p === regularSubDirGeminiFile) return 'My code memory';
- throw new Error('File not found');
- });
+ it('should ignore specified directories during downward scan', async () => {
+ await createEmptyDir(path.join(projectRoot, '.git'));
+ await createTestFile(path.join(projectRoot, '.gitignore'), 'node_modules');
- mockFs.readdir.mockImplementation((async (
- p: fsSync.PathLike,
- ): Promise<Dirent[]> => {
- if (p === CWD) {
- return [
- {
- name: 'node_modules',
- isFile: () => false,
- isDirectory: () => true,
- } as Dirent,
- {
- name: 'my_code',
- isFile: () => false,
- isDirectory: () => true,
- } as Dirent,
- ] as Dirent[];
- }
- if (p === regularSubDir) {
- return [
- {
- name: ORIGINAL_GEMINI_MD_FILENAME_CONST_FOR_TEST,
- isFile: () => true,
- isDirectory: () => false,
- } as Dirent,
- ] as Dirent[];
- }
- if (p === ignoredDir) {
- return [] as Dirent[];
- }
- return [] as Dirent[];
- }) as unknown as typeof fsPromises.readdir);
+ await createTestFile(
+ path.join(cwd, 'node_modules', DEFAULT_CONTEXT_FILENAME),
+ 'Ignored memory',
+ );
+ const regularSubDirGeminiFile = await createTestFile(
+ path.join(cwd, 'my_code', DEFAULT_CONTEXT_FILENAME),
+ 'My code memory',
+ );
- const { memoryContent, fileCount } = await loadServerHierarchicalMemory(
- CWD,
+ const result = await loadServerHierarchicalMemory(
+ cwd,
false,
- fileService,
+ new FileDiscoveryService(projectRoot),
+ [],
+ {
+ respectGitIgnore: true,
+ respectGeminiIgnore: true,
+ },
);
- const expectedContent = `--- Context from: ${path.join('my_code', ORIGINAL_GEMINI_MD_FILENAME_CONST_FOR_TEST)} ---\nMy code memory\n--- End of Context from: ${path.join('my_code', ORIGINAL_GEMINI_MD_FILENAME_CONST_FOR_TEST)} ---`;
-
- expect(memoryContent).toBe(expectedContent);
- expect(fileCount).toBe(1);
- expect(mockFs.readFile).not.toHaveBeenCalledWith(
- ignoredDirGeminiFile,
- 'utf-8',
- );
+ expect(result).toEqual({
+ memoryContent: `--- Context from: ${path.relative(cwd, regularSubDirGeminiFile)} ---
+My code memory
+--- End of Context from: ${path.relative(cwd, regularSubDirGeminiFile)} ---`,
+ fileCount: 1,
+ });
});
it('should respect MAX_DIRECTORIES_TO_SCAN_FOR_MEMORY during downward scan', async () => {
- const consoleDebugSpy = vi
- .spyOn(console, 'debug')
- .mockImplementation(() => {});
-
- const dirNames: Dirent[] = [];
+ // the max depth is 200 so it will give up before searching all these.
for (let i = 0; i < 250; i++) {
- dirNames.push({
- name: `deep_dir_${i}`,
- isFile: () => false,
- isDirectory: () => true,
- } as Dirent);
+ await createEmptyDir(path.join(cwd, `deep_dir_${i}`));
}
- mockFs.readdir.mockImplementation((async (
- p: fsSync.PathLike,
- ): Promise<Dirent[]> => {
- if (p === CWD) return dirNames;
- if (p.toString().startsWith(path.join(CWD, 'deep_dir_')))
- return [] as Dirent[];
- return [] as Dirent[];
- }) as unknown as typeof fsPromises.readdir);
- mockFs.access.mockRejectedValue(new Error('not found'));
-
- await loadServerHierarchicalMemory(CWD, true, fileService);
+ // "much_deeper" is alphabetically after "deep_dir_*" so it won't be loaded
+ await createTestFile(
+ path.join(cwd, 'much_deeper', DEFAULT_CONTEXT_FILENAME),
+ 'Ignored memory',
+ );
- expect(consoleDebugSpy).toHaveBeenCalledWith(
- expect.stringContaining('[DEBUG] [BfsFileSearch]'),
- expect.stringContaining('Scanning [200/200]:'),
+ const result = await loadServerHierarchicalMemory(
+ cwd,
+ false,
+ new FileDiscoveryService(projectRoot),
);
- consoleDebugSpy.mockRestore();
+
+ expect(result).toEqual({
+ memoryContent: '',
+ fileCount: 0,
+ });
});
it('should load extension context file paths', async () => {
- const extensionFilePath = '/test/extensions/ext1/GEMINI.md';
- mockFs.access.mockImplementation(async (p) => {
- if (p === extensionFilePath) {
- return undefined;
- }
- throw new Error('File not found');
- });
- mockFs.readFile.mockImplementation(async (p) => {
- if (p === extensionFilePath) {
- return 'Extension memory content';
- }
- throw new Error('File not found');
- });
+ const extensionFilePath = await createTestFile(
+ path.join(testRootDir, 'extensions/ext1/GEMINI.md'),
+ 'Extension memory content',
+ );
- const { memoryContent, fileCount } = await loadServerHierarchicalMemory(
- CWD,
+ const result = await loadServerHierarchicalMemory(
+ cwd,
false,
- fileService,
+ new FileDiscoveryService(projectRoot),
[extensionFilePath],
);
- expect(memoryContent).toBe(
- `--- Context from: ${path.relative(CWD, extensionFilePath)} ---\nExtension memory content\n--- End of Context from: ${path.relative(CWD, extensionFilePath)} ---`,
- );
- expect(fileCount).toBe(1);
- expect(mockFs.readFile).toHaveBeenCalledWith(extensionFilePath, 'utf-8');
+ expect(result).toEqual({
+ memoryContent: `--- Context from: ${path.relative(cwd, extensionFilePath)} ---
+Extension memory content
+--- End of Context from: ${path.relative(cwd, extensionFilePath)} ---`,
+ fileCount: 1,
+ });
});
});