mirror of
https://github.com/siteboon/claudecodeui.git
synced 2026-05-07 21:25:46 +00:00
180 lines
5.3 KiB
TypeScript
180 lines
5.3 KiB
TypeScript
import os from 'node:os';
|
|
import path from 'node:path';
|
|
import { readFile } from 'node:fs/promises';
|
|
|
|
import { sessionsDb } from '@/modules/database/index.js';
|
|
import {
|
|
buildLookupMap,
|
|
extractFirstValidJsonlData,
|
|
findFilesRecursivelyCreatedAfter,
|
|
normalizeSessionName,
|
|
readFileTimestamps,
|
|
} from '@/shared/utils.js';
|
|
import type { IProviderSessionSynchronizer } from '@/shared/interfaces.js';
|
|
|
|
type ParsedSession = {
|
|
sessionId: string;
|
|
projectPath: string;
|
|
sessionName?: string;
|
|
};
|
|
|
|
/**
|
|
* Session indexer for Codex transcript artifacts.
|
|
*/
|
|
export class CodexSessionSynchronizer implements IProviderSessionSynchronizer {
|
|
private readonly provider = 'codex' as const;
|
|
private readonly codexHome = path.join(os.homedir(), '.codex');
|
|
|
|
/**
|
|
* Scans ~/.codex/sessions and upserts discovered sessions into DB.
|
|
*/
|
|
async synchronize(since?: Date): Promise<number> {
|
|
const nameMap = await buildLookupMap(path.join(this.codexHome, 'session_index.jsonl'), 'id', 'thread_name');
|
|
const files = await findFilesRecursivelyCreatedAfter(
|
|
path.join(this.codexHome, 'sessions'),
|
|
'.jsonl',
|
|
since ?? null
|
|
);
|
|
|
|
let processed = 0;
|
|
for (const filePath of files) {
|
|
const parsed = await this.processSessionFile(filePath, nameMap);
|
|
if (!parsed) {
|
|
continue;
|
|
}
|
|
|
|
const existingSession = sessionsDb.getSessionById(parsed.sessionId);
|
|
if (existingSession) {
|
|
// If session name is untitled and we now have a name, update it
|
|
if (existingSession.custom_name === 'Untitled Codex Session' && parsed.sessionName && parsed.sessionName !== 'Untitled Codex Session') {
|
|
sessionsDb.updateSessionCustomName(parsed.sessionId, parsed.sessionName);
|
|
}
|
|
}
|
|
|
|
const timestamps = await readFileTimestamps(filePath);
|
|
sessionsDb.createSession(
|
|
parsed.sessionId,
|
|
this.provider,
|
|
parsed.projectPath,
|
|
parsed.sessionName,
|
|
timestamps.createdAt,
|
|
timestamps.updatedAt,
|
|
filePath
|
|
);
|
|
processed += 1;
|
|
}
|
|
|
|
return processed;
|
|
}
|
|
|
|
/**
|
|
* Parses and upserts one Codex session JSONL file.
|
|
*/
|
|
async synchronizeFile(filePath: string): Promise<string | null> {
|
|
if (!filePath.endsWith('.jsonl')) {
|
|
return null;
|
|
}
|
|
|
|
const nameMap = await buildLookupMap(path.join(this.codexHome, 'session_index.jsonl'), 'id', 'thread_name');
|
|
const parsed = await this.processSessionFile(filePath, nameMap);
|
|
if (!parsed) {
|
|
return null;
|
|
}
|
|
|
|
const timestamps = await readFileTimestamps(filePath);
|
|
return sessionsDb.createSession(
|
|
parsed.sessionId,
|
|
this.provider,
|
|
parsed.projectPath,
|
|
parsed.sessionName,
|
|
timestamps.createdAt,
|
|
timestamps.updatedAt,
|
|
filePath
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Extracts session metadata from one Codex JSONL session file.
|
|
*/
|
|
private async processSessionFile(
|
|
filePath: string,
|
|
nameMap: Map<string, string>
|
|
): Promise<ParsedSession | null> {
|
|
const parsed = await extractFirstValidJsonlData(filePath, (rawData) => {
|
|
const data = rawData as Record<string, unknown>;
|
|
const payload = data.payload as Record<string, unknown> | undefined;
|
|
const sessionId = typeof payload?.id === 'string' ? payload.id : undefined;
|
|
const projectPath = typeof payload?.cwd === 'string' ? payload.cwd : undefined;
|
|
|
|
if (!sessionId || !projectPath) {
|
|
return null;
|
|
}
|
|
|
|
return {
|
|
sessionId,
|
|
projectPath,
|
|
};
|
|
});
|
|
|
|
if (!parsed) {
|
|
return null;
|
|
}
|
|
|
|
const existingSession = sessionsDb.getSessionById(parsed.sessionId);
|
|
const existingSessionName = existingSession?.custom_name;
|
|
if (existingSessionName && existingSessionName !== 'Untitled Codex Session') {
|
|
return {
|
|
...parsed,
|
|
sessionName: normalizeSessionName(existingSessionName, 'Untitled Codex Session'),
|
|
};
|
|
}
|
|
|
|
let sessionName = nameMap.get(parsed.sessionId);
|
|
if (!sessionName) {
|
|
sessionName = await this.extractLastAgentMessageFromEnd(filePath);
|
|
}
|
|
|
|
return {
|
|
...parsed,
|
|
sessionName: normalizeSessionName(sessionName, 'Untitled Codex Session'),
|
|
};
|
|
}
|
|
|
|
private async extractLastAgentMessageFromEnd(filePath: string): Promise<string | undefined> {
|
|
try {
|
|
const content = await readFile(filePath, 'utf8');
|
|
const lines = content.split(/\r?\n/);
|
|
|
|
for (let index = lines.length - 1; index >= 0; index -= 1) {
|
|
const line = lines[index]?.trim();
|
|
if (!line) {
|
|
continue;
|
|
}
|
|
|
|
let parsed: unknown;
|
|
try {
|
|
parsed = JSON.parse(line);
|
|
} catch {
|
|
continue;
|
|
}
|
|
|
|
const data = parsed as Record<string, unknown>;
|
|
const eventType = typeof data.type === 'string' ? data.type : undefined;
|
|
const payload = data.payload as Record<string, unknown> | undefined;
|
|
const payloadType = typeof payload?.type === 'string' ? payload.type : undefined;
|
|
const lastAgentMessage = typeof payload?.last_agent_message === 'string'
|
|
? payload.last_agent_message
|
|
: undefined;
|
|
|
|
if (eventType === 'event_msg' && payloadType === 'task_complete' && lastAgentMessage?.trim()) {
|
|
return lastAgentMessage;
|
|
}
|
|
}
|
|
} catch {
|
|
// Ignore missing/unreadable files so sync can continue.
|
|
}
|
|
|
|
return undefined;
|
|
}
|
|
}
|