fix(core): ensure tool output cleanup on session deletion for legacy files (#26263)

This commit is contained in:
Coco Sheng
2026-04-30 16:11:38 -04:00
committed by GitHub
parent 84616626f5
commit 7125d2cd65
4 changed files with 244 additions and 34 deletions
@@ -602,6 +602,52 @@ describe('ChatRecordingService', () => {
expect(fs.existsSync(sessionDir)).toBe(false);
});
it('should delete legacy pretty-printed session files and their artifacts', async () => {
const sessionId = 'legacy-uuid';
const shortId = 'legacy12';
const chatsDir = path.join(testTempDir, 'chats');
const toolOutputsDir = path.join(testTempDir, 'tool-outputs');
fs.mkdirSync(chatsDir, { recursive: true });
fs.mkdirSync(toolOutputsDir, { recursive: true });
const sessionFile = path.join(
chatsDir,
`session-2023-01-01T00-00-${shortId}.json`,
);
// Pretty-printed JSON (not JSONL)
fs.writeFileSync(
sessionFile,
JSON.stringify({ sessionId, messages: [] }, null, 2),
);
const toolOutputDir = path.join(toolOutputsDir, `session-${sessionId}`);
fs.mkdirSync(toolOutputDir, { recursive: true });
fs.writeFileSync(path.join(toolOutputDir, 'output.txt'), 'data');
await chatRecordingService.deleteSession(shortId);
expect(fs.existsSync(sessionFile)).toBe(false);
expect(fs.existsSync(toolOutputDir)).toBe(false);
});
it('should delete the session file even if it is corrupted (invalid JSON)', async () => {
const shortId = 'corrupt1';
const chatsDir = path.join(testTempDir, 'chats');
fs.mkdirSync(chatsDir, { recursive: true });
const sessionFile = path.join(
chatsDir,
`session-2023-01-01T00-00-${shortId}.jsonl`,
);
fs.writeFileSync(sessionFile, 'not-json');
await chatRecordingService.deleteSession(shortId);
expect(fs.existsSync(sessionFile)).toBe(false);
});
it('should delete subagent files and their logs when parent is deleted', async () => {
const parentSessionId = '12345678-session-id';
const shortId = '12345678';
@@ -744,6 +744,8 @@ export class ChatRecordingService {
tempDir: string,
): Promise<void> {
const filePath = path.join(chatsDir, file);
let fullSessionId: string | undefined;
try {
const CHUNK_SIZE = 4096;
const buffer = Buffer.alloc(CHUNK_SIZE);
@@ -752,32 +754,43 @@ export class ChatRecordingService {
try {
fd = await fs.promises.open(filePath, 'r');
const { bytesRead } = await fd.read(buffer, 0, CHUNK_SIZE, 0);
if (bytesRead === 0) {
await fd.close();
await fs.promises.unlink(filePath);
return;
if (bytesRead > 0) {
const contentChunk = buffer.toString('utf8', 0, bytesRead);
const newlineIndex = contentChunk.indexOf('\n');
firstLine =
newlineIndex !== -1
? contentChunk.substring(0, newlineIndex)
: contentChunk;
try {
const content = JSON.parse(firstLine) as unknown;
if (isSessionIdRecord(content)) {
fullSessionId = content.sessionId;
}
} catch {
// If first line parse fails, it might be a legacy pretty-printed JSON.
// We'll fall back to full file read below.
}
}
const contentChunk = buffer.toString('utf8', 0, bytesRead);
const newlineIndex = contentChunk.indexOf('\n');
firstLine =
newlineIndex !== -1
? contentChunk.substring(0, newlineIndex)
: contentChunk;
} finally {
if (fd !== undefined) {
await fd.close();
}
}
const content = JSON.parse(firstLine) as unknown;
let fullSessionId: string | undefined;
if (isSessionIdRecord(content)) {
fullSessionId = content['sessionId'];
// Fallback for legacy JSON files if we couldn't get sessionId from first line
if (!fullSessionId) {
try {
const fileContent = await fs.promises.readFile(filePath, 'utf8');
const parsed = JSON.parse(fileContent) as unknown;
if (isSessionIdRecord(parsed)) {
fullSessionId = parsed.sessionId;
}
} catch {
// Ignore parse errors, we'll still try to unlink the file
}
}
// Delete the session file
await fs.promises.unlink(filePath);
if (fullSessionId) {
// Delegate to shared utility!
await deleteSessionArtifactsAsync(fullSessionId, tempDir);
@@ -788,7 +801,19 @@ export class ChatRecordingService {
);
}
} catch (error) {
debugLogger.error(`Error deleting associated file ${file}:`, error);
debugLogger.error(
`Error deleting artifacts for session file ${file}:`,
error,
);
} finally {
// ALWAYS try to delete the session file itself
try {
await fs.promises.unlink(filePath);
} catch (error) {
if (isNodeError(error) && error.code !== 'ENOENT') {
debugLogger.error(`Error unlinking session file ${file}:`, error);
}
}
}
}