mirror of
https://github.com/siteboon/claudecodeui.git
synced 2026-06-12 09:02:08 +08:00
The frontend previously juggled placeholder IDs, provider-native IDs, and session_created handoffs, which caused race conditions and provider-specific branching. This introduces app-allocated session IDs, a chat run registry with event replay, delta sidebar updates, and one kind-based websocket contract so the UI can treat every provider the same while JSONL remains the source of truth.
184 lines
5.6 KiB
TypeScript
184 lines
5.6 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.getSessionByProviderSessionId(parsed.sessionId)
|
|
?? 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(existingSession.session_id, 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;
|
|
}
|
|
|
|
// App-created sessions are keyed by an app id, so disk-discovered provider
|
|
// ids must be resolved through the provider-id mapping first.
|
|
const existingSession = sessionsDb.getSessionByProviderSessionId(parsed.sessionId)
|
|
?? 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;
|
|
}
|
|
}
|