mirror of
https://github.com/siteboon/claudecodeui.git
synced 2026-05-28 23:15:33 +08:00
Fix/websocket streaming issues (#748)
This commit is contained in:
@@ -89,13 +89,8 @@ const RIPGREP_CHUNK_CONCURRENCY = 6;
|
||||
const UNKNOWN_PROJECT_KEY = '__unknown_project__';
|
||||
|
||||
const INTERNAL_CONTENT_PREFIXES = [
|
||||
'<command-name>',
|
||||
'<command-message>',
|
||||
'<command-args>',
|
||||
'<local-command-stdout>',
|
||||
'<system-reminder>',
|
||||
'Caveat:',
|
||||
'This session is being continued from a previous',
|
||||
'Invalid API key',
|
||||
'[Request interrupted',
|
||||
] as const;
|
||||
@@ -302,6 +297,135 @@ function extractClaudeText(content: unknown): string {
|
||||
.join(' ');
|
||||
}
|
||||
|
||||
function extractTaggedContent(content: string, tagName: string): string | null {
|
||||
const escapedTagName = tagName.replace(/[.*+?^${}()|[\]\\]/g, '\\$&');
|
||||
const match = new RegExp(`<${escapedTagName}>([\\s\\S]*?)<\\/${escapedTagName}>`).exec(content);
|
||||
return match ? match[1] : null;
|
||||
}
|
||||
|
||||
type ClaudeLocalCommandPayload = {
|
||||
commandName: string;
|
||||
commandMessage: string;
|
||||
commandArgs: string;
|
||||
};
|
||||
|
||||
function parseClaudeLocalCommandPayload(content: string): ClaudeLocalCommandPayload | null {
|
||||
const commandName = extractTaggedContent(content, 'command-name');
|
||||
const commandMessage = extractTaggedContent(content, 'command-message');
|
||||
const commandArgs = extractTaggedContent(content, 'command-args');
|
||||
|
||||
if (commandName === null && commandMessage === null && commandArgs === null) {
|
||||
return null;
|
||||
}
|
||||
|
||||
return {
|
||||
commandName: commandName ?? '',
|
||||
commandMessage: commandMessage ?? '',
|
||||
commandArgs: commandArgs ?? '',
|
||||
};
|
||||
}
|
||||
|
||||
function buildClaudeLocalCommandDisplayText(payload: ClaudeLocalCommandPayload): string {
|
||||
const commandName = payload.commandName.trim();
|
||||
const commandMessage = payload.commandMessage.trim();
|
||||
const commandArgs = payload.commandArgs.trim();
|
||||
const baseCommand = commandName || commandMessage;
|
||||
|
||||
if (!baseCommand) {
|
||||
return '';
|
||||
}
|
||||
|
||||
return commandArgs ? `${baseCommand} ${commandArgs}` : baseCommand;
|
||||
}
|
||||
|
||||
function stripAnsiFormatting(text: string): string {
|
||||
return text.replace(/\u001B\[[0-9;?]*[ -/]*[@-~]/g, '');
|
||||
}
|
||||
|
||||
type ClaudeSearchableMessage = {
|
||||
text: string;
|
||||
role: 'user' | 'assistant';
|
||||
};
|
||||
|
||||
/**
|
||||
* Claude mixes visible chat, compact summaries, and local command wrappers into
|
||||
* the same transcript stream. Search should operate on the user-visible meaning
|
||||
* of those rows rather than the raw wrapper syntax.
|
||||
*/
|
||||
function extractClaudeSearchableMessage(entry: AnyRecord): ClaudeSearchableMessage | null {
|
||||
if (!entry.message?.content || entry.isApiErrorMessage) {
|
||||
return null;
|
||||
}
|
||||
|
||||
const rawRole = entry.message.role;
|
||||
if (rawRole !== 'user' && rawRole !== 'assistant') {
|
||||
return null;
|
||||
}
|
||||
|
||||
if (typeof entry.message.content === 'string') {
|
||||
const content = String(entry.message.content);
|
||||
|
||||
if (entry.isCompactSummary === true && content.trim()) {
|
||||
return {
|
||||
text: content,
|
||||
role: 'assistant',
|
||||
};
|
||||
}
|
||||
|
||||
const localCommand = parseClaudeLocalCommandPayload(content);
|
||||
if (localCommand) {
|
||||
const displayText = buildClaudeLocalCommandDisplayText(localCommand);
|
||||
return displayText
|
||||
? {
|
||||
text: displayText,
|
||||
role: 'user',
|
||||
}
|
||||
: null;
|
||||
}
|
||||
|
||||
const localCommandStdout = extractTaggedContent(content, 'local-command-stdout');
|
||||
if (localCommandStdout !== null) {
|
||||
const stdoutText = stripAnsiFormatting(localCommandStdout).trim();
|
||||
return stdoutText
|
||||
? {
|
||||
text: stdoutText,
|
||||
role: 'assistant',
|
||||
}
|
||||
: null;
|
||||
}
|
||||
|
||||
if (!content || isInternalContent(content)) {
|
||||
return null;
|
||||
}
|
||||
|
||||
return {
|
||||
text: content,
|
||||
role: rawRole,
|
||||
};
|
||||
}
|
||||
|
||||
const text = extractClaudeText(entry.message.content);
|
||||
if (!text) {
|
||||
return null;
|
||||
}
|
||||
|
||||
if (entry.isCompactSummary === true) {
|
||||
return {
|
||||
text,
|
||||
role: 'assistant',
|
||||
};
|
||||
}
|
||||
|
||||
if (isInternalContent(text)) {
|
||||
return null;
|
||||
}
|
||||
|
||||
return {
|
||||
text,
|
||||
role: rawRole,
|
||||
};
|
||||
}
|
||||
|
||||
function extractCodexText(content: unknown): string {
|
||||
if (typeof content === 'string') {
|
||||
return content;
|
||||
@@ -348,6 +472,7 @@ function extractGeminiText(content: unknown): string {
|
||||
|
||||
function normalizeSearchableSessions(rows: SessionRepositoryRow[]): SearchableSessionRow[] {
|
||||
const normalizedRows: SearchableSessionRow[] = [];
|
||||
const projectArchiveStateByPath = new Map<string, boolean>();
|
||||
|
||||
for (const row of rows) {
|
||||
const provider = row.provider as SearchableProvider;
|
||||
@@ -365,6 +490,27 @@ function normalizeSearchableSessions(rows: SessionRepositoryRow[]): SearchableSe
|
||||
continue;
|
||||
}
|
||||
|
||||
/**
|
||||
* Active session rows can still belong to an archived project because
|
||||
* project archiving intentionally preserves the underlying session data.
|
||||
* Global conversation search should follow the visible workspace model,
|
||||
* which means excluding any session whose owning project is archived.
|
||||
*
|
||||
* Cache the archive lookup per normalized project path so one search pass
|
||||
* does not re-query the same project row for every session in that folder.
|
||||
*/
|
||||
const normalizedProjectPath = typeof row.project_path === 'string' ? row.project_path.trim() : '';
|
||||
if (normalizedProjectPath) {
|
||||
if (!projectArchiveStateByPath.has(normalizedProjectPath)) {
|
||||
const projectRow = projectsDb.getProjectPath(normalizedProjectPath);
|
||||
projectArchiveStateByPath.set(normalizedProjectPath, Boolean(projectRow?.isArchived));
|
||||
}
|
||||
|
||||
if (projectArchiveStateByPath.get(normalizedProjectPath) === true) {
|
||||
continue;
|
||||
}
|
||||
}
|
||||
|
||||
normalizedRows.push({
|
||||
...row,
|
||||
provider,
|
||||
@@ -733,18 +879,21 @@ async function parseClaudeSessionMatches(
|
||||
}
|
||||
}
|
||||
|
||||
if (!entry.message?.content || entry.isApiErrorMessage) {
|
||||
const searchableMessage = extractClaudeSearchableMessage(entry);
|
||||
if (!searchableMessage) {
|
||||
continue;
|
||||
}
|
||||
|
||||
const role = entry.message.role;
|
||||
if (role !== 'user' && role !== 'assistant') {
|
||||
continue;
|
||||
}
|
||||
const { text, role } = searchableMessage;
|
||||
|
||||
const text = extractClaudeText(entry.message.content);
|
||||
if (!text || isInternalContent(text)) {
|
||||
continue;
|
||||
/**
|
||||
* Claude compact summaries are the most faithful session-summary source
|
||||
* after a `/compact` because they describe the post-compaction state that
|
||||
* the resumed session actually continues from. Prefer them over generic
|
||||
* fallback user text when present.
|
||||
*/
|
||||
if (entry.isCompactSummary === true) {
|
||||
state.resolvedSummary = text;
|
||||
}
|
||||
|
||||
if (role === 'user') {
|
||||
|
||||
@@ -18,16 +18,18 @@ const PROVIDER_WATCH_PATHS: Array<{ provider: LLMProvider; rootPath: string }> =
|
||||
},
|
||||
{
|
||||
provider: 'cursor',
|
||||
rootPath: path.join(os.homedir(), '.cursor', 'chats'),
|
||||
rootPath: path.join(os.homedir(), '.cursor', 'projects'),
|
||||
},
|
||||
{
|
||||
provider: 'codex',
|
||||
rootPath: path.join(os.homedir(), '.codex', 'sessions'),
|
||||
},
|
||||
{
|
||||
provider: 'gemini',
|
||||
rootPath: path.join(os.homedir(), '.gemini', 'sessions'),
|
||||
},
|
||||
// {
|
||||
// provider: 'gemini',
|
||||
// rootPath: path.join(os.homedir(), '.gemini', 'sessions'),
|
||||
// },
|
||||
// Keep `sessions/` watcher disabled: Gemini also mirrors artifacts there,
|
||||
// which causes duplicate synchronization events.
|
||||
{
|
||||
provider: 'gemini',
|
||||
rootPath: path.join(os.homedir(), '.gemini', 'tmp'),
|
||||
|
||||
@@ -1,6 +1,7 @@
|
||||
import fsp from 'node:fs/promises';
|
||||
import path from 'node:path';
|
||||
|
||||
import { sessionsDb } from '@/modules/database/index.js';
|
||||
import { projectsDb, sessionsDb } from '@/modules/database/index.js';
|
||||
import { providerRegistry } from '@/modules/providers/provider.registry.js';
|
||||
import type {
|
||||
FetchHistoryOptions,
|
||||
@@ -10,6 +11,19 @@ import type {
|
||||
} from '@/shared/types.js';
|
||||
import { AppError } from '@/shared/utils.js';
|
||||
|
||||
type ArchivedSessionListItem = {
|
||||
sessionId: string;
|
||||
provider: LLMProvider;
|
||||
projectId: string | null;
|
||||
projectPath: string | null;
|
||||
projectDisplayName: string;
|
||||
sessionTitle: string;
|
||||
createdAt: string | null;
|
||||
updatedAt: string | null;
|
||||
lastActivity: string | null;
|
||||
isProjectArchived: boolean;
|
||||
};
|
||||
|
||||
/**
|
||||
* Removes one file if it exists.
|
||||
*/
|
||||
@@ -26,6 +40,28 @@ async function removeFileIfExists(filePath: string): Promise<boolean> {
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Archive rows need a stable project label even when the owning project is not
|
||||
* part of the active sidebar payload. This lightweight resolver keeps the
|
||||
* archive API self-contained while still matching the project's stored display
|
||||
* name when one exists.
|
||||
*/
|
||||
function resolveProjectDisplayName(
|
||||
projectPath: string | null,
|
||||
customProjectName: string | null | undefined,
|
||||
): string {
|
||||
const trimmedCustomName = typeof customProjectName === 'string' ? customProjectName.trim() : '';
|
||||
if (trimmedCustomName.length > 0) {
|
||||
return trimmedCustomName;
|
||||
}
|
||||
|
||||
if (!projectPath) {
|
||||
return 'Unknown Project';
|
||||
}
|
||||
|
||||
return path.basename(projectPath) || projectPath;
|
||||
}
|
||||
|
||||
/**
|
||||
* Application service for provider-backed session message operations.
|
||||
*
|
||||
@@ -79,15 +115,53 @@ export const sessionsService = {
|
||||
},
|
||||
|
||||
/**
|
||||
* Deletes one persisted session row by id.
|
||||
*
|
||||
* When `deletedFromDisk` is true and a session `jsonl_path` exists, the path
|
||||
* is deleted from disk before the DB row is removed.
|
||||
* Returns archived sessions with enough project metadata for the sidebar to
|
||||
* group, filter, open, and restore them without a per-row follow-up query.
|
||||
*/
|
||||
async deleteSessionById(
|
||||
listArchivedSessions(): ArchivedSessionListItem[] {
|
||||
const archivedSessions = sessionsDb.getArchivedSessions();
|
||||
const projectCache = new Map<string, ReturnType<typeof projectsDb.getProjectPath>>();
|
||||
|
||||
return archivedSessions.map((session) => {
|
||||
const projectPath = session.project_path?.trim() ? session.project_path : null;
|
||||
let project = null;
|
||||
|
||||
if (projectPath) {
|
||||
if (!projectCache.has(projectPath)) {
|
||||
projectCache.set(projectPath, projectsDb.getProjectPath(projectPath));
|
||||
}
|
||||
project = projectCache.get(projectPath) ?? null;
|
||||
}
|
||||
|
||||
return {
|
||||
sessionId: session.session_id,
|
||||
provider: session.provider as LLMProvider,
|
||||
projectId: project?.project_id ?? null,
|
||||
projectPath,
|
||||
projectDisplayName: resolveProjectDisplayName(projectPath, project?.custom_project_name),
|
||||
sessionTitle: session.custom_name?.trim() || session.session_id,
|
||||
createdAt: session.created_at ?? null,
|
||||
updatedAt: session.updated_at ?? null,
|
||||
lastActivity: session.updated_at ?? session.created_at ?? null,
|
||||
isProjectArchived: Boolean(project?.isArchived),
|
||||
};
|
||||
});
|
||||
},
|
||||
|
||||
/**
|
||||
* Archives or permanently deletes one persisted session row by id.
|
||||
*
|
||||
* Soft-delete mirrors the project behavior by toggling `isArchived` so the
|
||||
* row disappears from active lists but remains restorable. Force-delete
|
||||
* optionally removes the transcript file before deleting the database row.
|
||||
*/
|
||||
async deleteOrArchiveSessionById(
|
||||
sessionId: string,
|
||||
deletedFromDisk = false,
|
||||
): Promise<{ sessionId: string; deletedFromDisk: boolean }> {
|
||||
options: {
|
||||
force?: boolean;
|
||||
deletedFromDisk?: boolean;
|
||||
} = {},
|
||||
): Promise<{ sessionId: string; action: 'archived' | 'deleted'; deletedFromDisk: boolean }> {
|
||||
const session = sessionsDb.getSessionById(sessionId);
|
||||
if (!session) {
|
||||
throw new AppError(`Session "${sessionId}" was not found.`, {
|
||||
@@ -96,8 +170,17 @@ export const sessionsService = {
|
||||
});
|
||||
}
|
||||
|
||||
if (!options.force) {
|
||||
sessionsDb.updateSessionIsArchived(sessionId, true);
|
||||
return {
|
||||
sessionId,
|
||||
action: 'archived',
|
||||
deletedFromDisk: false,
|
||||
};
|
||||
}
|
||||
|
||||
let removedFromDisk = false;
|
||||
if (deletedFromDisk && session.jsonl_path) {
|
||||
if (options.deletedFromDisk && session.jsonl_path) {
|
||||
removedFromDisk = await removeFileIfExists(session.jsonl_path);
|
||||
}
|
||||
|
||||
@@ -109,7 +192,27 @@ export const sessionsService = {
|
||||
});
|
||||
}
|
||||
|
||||
return { sessionId, deletedFromDisk: removedFromDisk };
|
||||
return {
|
||||
sessionId,
|
||||
action: 'deleted',
|
||||
deletedFromDisk: removedFromDisk,
|
||||
};
|
||||
},
|
||||
|
||||
/**
|
||||
* Restores one archived session back into the active sidebar lists.
|
||||
*/
|
||||
restoreSessionById(sessionId: string): { sessionId: string; isArchived: false } {
|
||||
const session = sessionsDb.getSessionById(sessionId);
|
||||
if (!session) {
|
||||
throw new AppError(`Session "${sessionId}" was not found.`, {
|
||||
code: 'SESSION_NOT_FOUND',
|
||||
statusCode: 404,
|
||||
});
|
||||
}
|
||||
|
||||
sessionsDb.updateSessionIsArchived(sessionId, false);
|
||||
return { sessionId, isArchived: false };
|
||||
},
|
||||
|
||||
/**
|
||||
|
||||
Reference in New Issue
Block a user