mirror of
https://github.com/google-gemini/gemini-cli.git
synced 2026-02-01 22:48:03 +00:00
286 lines
9.9 KiB
TypeScript
286 lines
9.9 KiB
TypeScript
/**
|
|
* @license
|
|
* Copyright 2025 Google LLC
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
|
|
import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest';
|
|
import * as fs from 'node:fs/promises';
|
|
import * as path from 'node:path';
|
|
import * as os from 'node:os';
|
|
import { debugLogger, TOOL_OUTPUT_DIR } from '@google/gemini-cli-core';
|
|
import type { Settings } from '../config/settings.js';
|
|
import { cleanupToolOutputFiles } from './sessionCleanup.js';
|
|
|
|
describe('Tool Output Cleanup', () => {
|
|
let testTempDir: string;
|
|
|
|
beforeEach(async () => {
|
|
// Create a unique temp directory for each test
|
|
testTempDir = await fs.mkdtemp(path.join(os.tmpdir(), 'tool-output-test-'));
|
|
vi.spyOn(debugLogger, 'error').mockImplementation(() => {});
|
|
vi.spyOn(debugLogger, 'warn').mockImplementation(() => {});
|
|
vi.spyOn(debugLogger, 'debug').mockImplementation(() => {});
|
|
});
|
|
|
|
afterEach(async () => {
|
|
vi.restoreAllMocks();
|
|
// Clean up the temp directory
|
|
try {
|
|
await fs.rm(testTempDir, { recursive: true, force: true });
|
|
} catch {
|
|
// Ignore cleanup errors
|
|
}
|
|
});
|
|
|
|
describe('cleanupToolOutputFiles', () => {
|
|
it('should return early when cleanup is disabled', async () => {
|
|
const settings: Settings = {
|
|
general: { sessionRetention: { enabled: false } },
|
|
};
|
|
|
|
const result = await cleanupToolOutputFiles(settings, false, testTempDir);
|
|
|
|
expect(result.disabled).toBe(true);
|
|
expect(result.scanned).toBe(0);
|
|
expect(result.deleted).toBe(0);
|
|
expect(result.failed).toBe(0);
|
|
});
|
|
|
|
it('should return early when sessionRetention is not configured', async () => {
|
|
const settings: Settings = {};
|
|
|
|
const result = await cleanupToolOutputFiles(settings, false, testTempDir);
|
|
|
|
expect(result.disabled).toBe(true);
|
|
expect(result.scanned).toBe(0);
|
|
expect(result.deleted).toBe(0);
|
|
});
|
|
|
|
it('should return early when tool_output directory does not exist', async () => {
|
|
const settings: Settings = {
|
|
general: {
|
|
sessionRetention: {
|
|
enabled: true,
|
|
maxAge: '7d',
|
|
},
|
|
},
|
|
};
|
|
|
|
// Don't create the tool_output directory
|
|
const result = await cleanupToolOutputFiles(settings, false, testTempDir);
|
|
|
|
expect(result.disabled).toBe(false);
|
|
expect(result.scanned).toBe(0);
|
|
expect(result.deleted).toBe(0);
|
|
expect(result.failed).toBe(0);
|
|
});
|
|
|
|
it('should delete files older than maxAge', async () => {
|
|
const settings: Settings = {
|
|
general: {
|
|
sessionRetention: {
|
|
enabled: true,
|
|
maxAge: '7d',
|
|
},
|
|
},
|
|
};
|
|
|
|
// Create tool_output directory and files
|
|
const toolOutputDir = path.join(testTempDir, TOOL_OUTPUT_DIR);
|
|
await fs.mkdir(toolOutputDir, { recursive: true });
|
|
|
|
const now = Date.now();
|
|
const fiveDaysAgo = now - 5 * 24 * 60 * 60 * 1000;
|
|
const tenDaysAgo = now - 10 * 24 * 60 * 60 * 1000;
|
|
|
|
// Create files with different ages
|
|
const recentFile = path.join(toolOutputDir, 'shell_recent.txt');
|
|
const oldFile = path.join(toolOutputDir, 'shell_old.txt');
|
|
|
|
await fs.writeFile(recentFile, 'recent content');
|
|
await fs.writeFile(oldFile, 'old content');
|
|
|
|
// Set file modification times
|
|
await fs.utimes(recentFile, fiveDaysAgo / 1000, fiveDaysAgo / 1000);
|
|
await fs.utimes(oldFile, tenDaysAgo / 1000, tenDaysAgo / 1000);
|
|
|
|
const result = await cleanupToolOutputFiles(settings, false, testTempDir);
|
|
|
|
expect(result.disabled).toBe(false);
|
|
expect(result.scanned).toBe(2);
|
|
expect(result.deleted).toBe(1); // Only the 10-day-old file should be deleted
|
|
expect(result.failed).toBe(0);
|
|
|
|
// Verify the old file was deleted and recent file remains
|
|
const remainingFiles = await fs.readdir(toolOutputDir);
|
|
expect(remainingFiles).toContain('shell_recent.txt');
|
|
expect(remainingFiles).not.toContain('shell_old.txt');
|
|
});
|
|
|
|
it('should delete oldest files when exceeding maxCount', async () => {
|
|
const settings: Settings = {
|
|
general: {
|
|
sessionRetention: {
|
|
enabled: true,
|
|
maxCount: 2,
|
|
},
|
|
},
|
|
};
|
|
|
|
// Create tool_output directory and files
|
|
const toolOutputDir = path.join(testTempDir, TOOL_OUTPUT_DIR);
|
|
await fs.mkdir(toolOutputDir, { recursive: true });
|
|
|
|
const now = Date.now();
|
|
const oneDayAgo = now - 1 * 24 * 60 * 60 * 1000;
|
|
const twoDaysAgo = now - 2 * 24 * 60 * 60 * 1000;
|
|
const threeDaysAgo = now - 3 * 24 * 60 * 60 * 1000;
|
|
|
|
// Create 3 files with different ages
|
|
const file1 = path.join(toolOutputDir, 'shell_1.txt');
|
|
const file2 = path.join(toolOutputDir, 'shell_2.txt');
|
|
const file3 = path.join(toolOutputDir, 'shell_3.txt');
|
|
|
|
await fs.writeFile(file1, 'content 1');
|
|
await fs.writeFile(file2, 'content 2');
|
|
await fs.writeFile(file3, 'content 3');
|
|
|
|
// Set file modification times (file3 is oldest)
|
|
await fs.utimes(file1, oneDayAgo / 1000, oneDayAgo / 1000);
|
|
await fs.utimes(file2, twoDaysAgo / 1000, twoDaysAgo / 1000);
|
|
await fs.utimes(file3, threeDaysAgo / 1000, threeDaysAgo / 1000);
|
|
|
|
const result = await cleanupToolOutputFiles(settings, false, testTempDir);
|
|
|
|
expect(result.disabled).toBe(false);
|
|
expect(result.scanned).toBe(3);
|
|
expect(result.deleted).toBe(1); // Should delete 1 file to get down to maxCount of 2
|
|
expect(result.failed).toBe(0);
|
|
|
|
// Verify the oldest file was deleted
|
|
const remainingFiles = await fs.readdir(toolOutputDir);
|
|
expect(remainingFiles).toHaveLength(2);
|
|
expect(remainingFiles).not.toContain('shell_3.txt');
|
|
});
|
|
|
|
it('should handle empty directory', async () => {
|
|
const settings: Settings = {
|
|
general: {
|
|
sessionRetention: {
|
|
enabled: true,
|
|
maxAge: '7d',
|
|
},
|
|
},
|
|
};
|
|
|
|
// Create empty tool_output directory
|
|
const toolOutputDir = path.join(testTempDir, TOOL_OUTPUT_DIR);
|
|
await fs.mkdir(toolOutputDir, { recursive: true });
|
|
|
|
const result = await cleanupToolOutputFiles(settings, false, testTempDir);
|
|
|
|
expect(result.disabled).toBe(false);
|
|
expect(result.scanned).toBe(0);
|
|
expect(result.deleted).toBe(0);
|
|
expect(result.failed).toBe(0);
|
|
});
|
|
|
|
it('should apply both maxAge and maxCount together', async () => {
|
|
const settings: Settings = {
|
|
general: {
|
|
sessionRetention: {
|
|
enabled: true,
|
|
maxAge: '3d',
|
|
maxCount: 2,
|
|
},
|
|
},
|
|
};
|
|
|
|
// Create tool_output directory and files
|
|
const toolOutputDir = path.join(testTempDir, TOOL_OUTPUT_DIR);
|
|
await fs.mkdir(toolOutputDir, { recursive: true });
|
|
|
|
const now = Date.now();
|
|
const oneDayAgo = now - 1 * 24 * 60 * 60 * 1000;
|
|
const twoDaysAgo = now - 2 * 24 * 60 * 60 * 1000;
|
|
const twoAndHalfDaysAgo = now - 2.5 * 24 * 60 * 60 * 1000;
|
|
const fiveDaysAgo = now - 5 * 24 * 60 * 60 * 1000;
|
|
const tenDaysAgo = now - 10 * 24 * 60 * 60 * 1000;
|
|
|
|
// Create 5 files with different ages
|
|
const file1 = path.join(toolOutputDir, 'shell_1.txt'); // 1 day old - keep
|
|
const file2 = path.join(toolOutputDir, 'shell_2.txt'); // 2 days old - keep
|
|
const file3 = path.join(toolOutputDir, 'shell_3.txt'); // 2.5 days old - delete by count
|
|
const file4 = path.join(toolOutputDir, 'shell_4.txt'); // 5 days old - delete by age
|
|
const file5 = path.join(toolOutputDir, 'shell_5.txt'); // 10 days old - delete by age
|
|
|
|
await fs.writeFile(file1, 'content 1');
|
|
await fs.writeFile(file2, 'content 2');
|
|
await fs.writeFile(file3, 'content 3');
|
|
await fs.writeFile(file4, 'content 4');
|
|
await fs.writeFile(file5, 'content 5');
|
|
|
|
// Set file modification times
|
|
await fs.utimes(file1, oneDayAgo / 1000, oneDayAgo / 1000);
|
|
await fs.utimes(file2, twoDaysAgo / 1000, twoDaysAgo / 1000);
|
|
await fs.utimes(
|
|
file3,
|
|
twoAndHalfDaysAgo / 1000,
|
|
twoAndHalfDaysAgo / 1000,
|
|
);
|
|
await fs.utimes(file4, fiveDaysAgo / 1000, fiveDaysAgo / 1000);
|
|
await fs.utimes(file5, tenDaysAgo / 1000, tenDaysAgo / 1000);
|
|
|
|
const result = await cleanupToolOutputFiles(settings, false, testTempDir);
|
|
|
|
expect(result.disabled).toBe(false);
|
|
expect(result.scanned).toBe(5);
|
|
// file4 and file5 deleted by maxAge, file3 deleted by maxCount
|
|
expect(result.deleted).toBe(3);
|
|
expect(result.failed).toBe(0);
|
|
|
|
// Verify only the 2 newest files remain
|
|
const remainingFiles = await fs.readdir(toolOutputDir);
|
|
expect(remainingFiles).toHaveLength(2);
|
|
expect(remainingFiles).toContain('shell_1.txt');
|
|
expect(remainingFiles).toContain('shell_2.txt');
|
|
expect(remainingFiles).not.toContain('shell_3.txt');
|
|
expect(remainingFiles).not.toContain('shell_4.txt');
|
|
expect(remainingFiles).not.toContain('shell_5.txt');
|
|
});
|
|
|
|
it('should log debug information when enabled', async () => {
|
|
const settings: Settings = {
|
|
general: {
|
|
sessionRetention: {
|
|
enabled: true,
|
|
maxAge: '1d',
|
|
},
|
|
},
|
|
};
|
|
|
|
// Create tool_output directory and an old file
|
|
const toolOutputDir = path.join(testTempDir, TOOL_OUTPUT_DIR);
|
|
await fs.mkdir(toolOutputDir, { recursive: true });
|
|
|
|
const tenDaysAgo = Date.now() - 10 * 24 * 60 * 60 * 1000;
|
|
const oldFile = path.join(toolOutputDir, 'shell_old.txt');
|
|
await fs.writeFile(oldFile, 'old content');
|
|
await fs.utimes(oldFile, tenDaysAgo / 1000, tenDaysAgo / 1000);
|
|
|
|
const debugSpy = vi
|
|
.spyOn(debugLogger, 'debug')
|
|
.mockImplementation(() => {});
|
|
|
|
await cleanupToolOutputFiles(settings, true, testTempDir);
|
|
|
|
expect(debugSpy).toHaveBeenCalledWith(
|
|
expect.stringContaining('Tool output cleanup: deleted'),
|
|
);
|
|
|
|
debugSpy.mockRestore();
|
|
});
|
|
});
|
|
});
|