fix: stop codex transcript replay after hooks migration (#2365)
This commit is contained in:
@@ -0,0 +1,111 @@
|
||||
import { afterEach, beforeEach, describe, expect, it, mock, spyOn } from 'bun:test';
|
||||
import { appendFileSync, mkdirSync, rmSync, writeFileSync } from 'fs';
|
||||
import { tmpdir } from 'os';
|
||||
import { join } from 'path';
|
||||
import type { NormalizedHookInput } from '../../src/cli/types.js';
|
||||
import type { TranscriptSchema, WatchTarget } from '../../src/services/transcripts/types.js';
|
||||
|
||||
const sessionInitCalls: NormalizedHookInput[] = [];
|
||||
|
||||
mock.module('../../src/cli/handlers/session-init.js', () => ({
|
||||
sessionInitHandler: {
|
||||
execute: async (input: NormalizedHookInput) => {
|
||||
sessionInitCalls.push(input);
|
||||
return { continue: true, suppressOutput: true };
|
||||
},
|
||||
},
|
||||
}));
|
||||
|
||||
import { logger } from '../../src/utils/logger.js';
|
||||
import { TranscriptWatcher } from '../../src/services/transcripts/watcher.js';
|
||||
|
||||
const waitForAsyncTail = () => new Promise(resolve => setTimeout(resolve, 50));
|
||||
|
||||
describe('TranscriptWatcher startAtEnd', () => {
|
||||
let tmpRoot: string;
|
||||
let loggerSpies: ReturnType<typeof spyOn>[] = [];
|
||||
|
||||
beforeEach(() => {
|
||||
sessionInitCalls.length = 0;
|
||||
tmpRoot = join(tmpdir(), `claude-mem-transcript-watch-${Date.now()}-${Math.random().toString(16).slice(2)}`);
|
||||
mkdirSync(tmpRoot, { recursive: true });
|
||||
loggerSpies = [
|
||||
spyOn(logger, 'info').mockImplementation(() => {}),
|
||||
spyOn(logger, 'debug').mockImplementation(() => {}),
|
||||
spyOn(logger, 'warn').mockImplementation(() => {}),
|
||||
spyOn(logger, 'error').mockImplementation(() => {}),
|
||||
];
|
||||
});
|
||||
|
||||
afterEach(() => {
|
||||
loggerSpies.forEach(spy => spy.mockRestore());
|
||||
rmSync(tmpRoot, { recursive: true, force: true });
|
||||
});
|
||||
|
||||
it('does not replay history from transcript files discovered after startup', async () => {
|
||||
const sessionId = '019e050e-7ae0-71b2-b19f-6cc428e5763a';
|
||||
const filePath = join(tmpRoot, `${sessionId}.jsonl`);
|
||||
const statePath = join(tmpRoot, 'state.json');
|
||||
|
||||
writeFileSync(
|
||||
filePath,
|
||||
`${JSON.stringify({
|
||||
type: 'event',
|
||||
payload: {
|
||||
type: 'user_message',
|
||||
session_id: sessionId,
|
||||
message: 'historical prompt that must not be replayed',
|
||||
},
|
||||
})}\n`,
|
||||
'utf8',
|
||||
);
|
||||
|
||||
const schema: TranscriptSchema = {
|
||||
name: 'codex-test',
|
||||
events: [
|
||||
{
|
||||
name: 'user-message',
|
||||
match: { path: 'payload.type', equals: 'user_message' },
|
||||
action: 'session_init',
|
||||
fields: {
|
||||
sessionId: 'payload.session_id',
|
||||
prompt: 'payload.message',
|
||||
},
|
||||
},
|
||||
],
|
||||
};
|
||||
const watch: WatchTarget = {
|
||||
name: 'codex',
|
||||
path: join(tmpRoot, '*.jsonl'),
|
||||
schema,
|
||||
startAtEnd: true,
|
||||
};
|
||||
const watcher = new TranscriptWatcher({ version: 1, watches: [watch] }, statePath);
|
||||
|
||||
await (watcher as any).addTailer(filePath, watch, schema);
|
||||
await waitForAsyncTail();
|
||||
|
||||
expect(sessionInitCalls).toHaveLength(0);
|
||||
|
||||
appendFileSync(
|
||||
filePath,
|
||||
`${JSON.stringify({
|
||||
type: 'event',
|
||||
payload: {
|
||||
type: 'user_message',
|
||||
session_id: sessionId,
|
||||
message: 'live prompt',
|
||||
},
|
||||
})}\n`,
|
||||
'utf8',
|
||||
);
|
||||
|
||||
(watcher as any).tailers.get(filePath)?.poke();
|
||||
await waitForAsyncTail();
|
||||
watcher.stop();
|
||||
|
||||
const prompts = sessionInitCalls.map(call => call.prompt);
|
||||
expect(prompts).toContain('live prompt');
|
||||
expect(prompts).not.toContain('historical prompt that must not be replayed');
|
||||
});
|
||||
});
|
||||
Reference in New Issue
Block a user