import { getActiveProfileDir, getProfileDir } from '../../services/hermes/hermes-profile' import type { LocalUsageStats } from './usage-store' const SQLITE_AVAILABLE = (() => { const [major, minor] = process.versions.node.split('.').map(Number) return major > 22 || (major === 22 && minor >= 5) })() const COMPRESSION_END_REASONS = new Set(['compression', 'compressed']) const SEARCH_CANDIDATE_MULTIPLIER = 20 const SEARCH_CANDIDATE_MIN = 100 export interface HermesSessionRow { id: string source: string user_id: string | null model: string title: string | null started_at: number ended_at: number | null end_reason: string | null message_count: number tool_call_count: number input_tokens: number output_tokens: number cache_read_tokens: number cache_write_tokens: number reasoning_tokens: number billing_provider: string | null estimated_cost_usd: number actual_cost_usd: number | null cost_status: string preview: string last_active: number } export interface HermesSessionSearchRow extends HermesSessionRow { matched_message_id: number | null snippet: string rank: number } export interface HermesMessageRow { id: number | string session_id: string role: string content: string tool_call_id: string | null tool_calls: any[] | null tool_name: string | null timestamp: number token_count: number | null finish_reason: string | null reasoning: string | null reasoning_details?: string | null codex_reasoning_items?: string | null reasoning_content?: string | null } export interface HermesSessionDetailRow extends HermesSessionRow { messages: HermesMessageRow[] thread_session_count: number } interface HermesSessionInternalRow extends HermesSessionRow { parent_session_id: string | null } function sessionDbPath(): string { return `${getActiveProfileDir()}/state.db` } function normalizeNumber(value: unknown, fallback = 0): number { if (value == null || value === '') return fallback const num = Number(value) return Number.isFinite(num) ? num : fallback } function normalizeNullableNumber(value: unknown): number | null { if (value == null || value === '') return null const num = Number(value) return Number.isFinite(num) ? num : null } function normalizeNullableString(value: unknown): string | null { if (value == null || value === '') return null return String(value) } function mapRow(row: Record): HermesSessionRow { const startedAt = normalizeNumber(row.started_at) const rawTitle = normalizeNullableString(row.title) const preview = String(row.preview || '') // Fallback: when no explicit title, use first user message as title (same as CLI path) const title = rawTitle || (preview ? (preview.length > 40 ? preview.slice(0, 40) + '...' : preview) : null) return { id: String(row.id || ''), source: String(row.source || ''), user_id: normalizeNullableString(row.user_id), model: String(row.model || ''), title, started_at: startedAt, ended_at: normalizeNullableNumber(row.ended_at), end_reason: normalizeNullableString(row.end_reason), message_count: normalizeNumber(row.message_count), tool_call_count: normalizeNumber(row.tool_call_count), input_tokens: normalizeNumber(row.input_tokens), output_tokens: normalizeNumber(row.output_tokens), cache_read_tokens: normalizeNumber(row.cache_read_tokens), cache_write_tokens: normalizeNumber(row.cache_write_tokens), reasoning_tokens: normalizeNumber(row.reasoning_tokens), billing_provider: normalizeNullableString(row.billing_provider), estimated_cost_usd: normalizeNumber(row.estimated_cost_usd), actual_cost_usd: normalizeNullableNumber(row.actual_cost_usd), cost_status: String(row.cost_status || ''), preview: String(row.preview || ''), last_active: normalizeNumber(row.last_active, startedAt), } } const SESSION_SELECT = ` s.id, s.source, COALESCE(s.user_id, '') AS user_id, COALESCE(s.model, '') AS model, COALESCE(s.title, '') AS title, COALESCE(s.started_at, 0) AS started_at, s.ended_at AS ended_at, COALESCE(s.end_reason, '') AS end_reason, COALESCE(s.message_count, 0) AS message_count, COALESCE(s.tool_call_count, 0) AS tool_call_count, COALESCE(s.input_tokens, 0) AS input_tokens, COALESCE(s.output_tokens, 0) AS output_tokens, COALESCE(s.cache_read_tokens, 0) AS cache_read_tokens, COALESCE(s.cache_write_tokens, 0) AS cache_write_tokens, COALESCE(s.reasoning_tokens, 0) AS reasoning_tokens, COALESCE(s.billing_provider, '') AS billing_provider, COALESCE(s.estimated_cost_usd, 0) AS estimated_cost_usd, s.actual_cost_usd AS actual_cost_usd, COALESCE(s.cost_status, '') AS cost_status, COALESCE( ( SELECT SUBSTR(REPLACE(REPLACE(m.content, CHAR(10), ' '), CHAR(13), ' '), 1, 63) FROM messages m WHERE m.session_id = s.id AND m.role = 'user' AND m.content IS NOT NULL ORDER BY m.timestamp, m.id LIMIT 1 ), '' ) AS preview, COALESCE((SELECT MAX(m2.timestamp) FROM messages m2 WHERE m2.session_id = s.id), s.started_at) AS last_active ` function containsCjk(text: string): boolean { for (const ch of text) { const cp = ch.codePointAt(0) ?? 0 if ( (cp >= 0x4E00 && cp <= 0x9FFF) || (cp >= 0x3400 && cp <= 0x4DBF) || (cp >= 0x20000 && cp <= 0x2A6DF) || (cp >= 0x3000 && cp <= 0x303F) || (cp >= 0x3040 && cp <= 0x309F) || (cp >= 0x30A0 && cp <= 0x30FF) || (cp >= 0xAC00 && cp <= 0xD7AF) ) { return true } } return false } function escapeLikePattern(value: string): string { return value.replace(/[\\%_]/g, (match) => `\\${match}`) } function buildLikePattern(value: string): string { return `%${escapeLikePattern(value)}%` } function normalizeTitleLikeQuery(query: string): string { const tokens = query.match(/"[^"]*"\*?|\S+/g) if (!tokens) return query const normalizedTokens = tokens .map((token) => { let value = token.endsWith('*') ? token.slice(0, -1) : token if (value.startsWith('"') && value.endsWith('"')) { value = value.slice(1, -1) } return value }) .filter(Boolean) return normalizedTokens.join(' ').trim() || query } function shouldUseLiteralContentSearch(query: string): boolean { const trimmed = query.trim() if (!trimmed) return false if (/[^\p{L}\p{N}\s"*.-]/u.test(trimmed)) return true const tokens = trimmed.match(/"[^"]*"\*?|\S+/g) if (!tokens) return true for (const token of tokens) { if (/^(AND|OR|NOT)$/i.test(token)) continue const raw = token.endsWith('*') ? token.slice(0, -1) : token if (!raw) return true if (raw.startsWith('"') && raw.endsWith('"')) { const inner = raw.slice(1, -1) if (!inner.trim()) return true if (!/^[\p{L}\p{N}\s.-]+$/u.test(inner)) return true if ((inner.includes('.') || inner.includes('-')) && !/^[\p{L}\p{N}]+(?:[.-][\p{L}\p{N}]+)*(?:\s+[\p{L}\p{N}]+(?:[.-][\p{L}\p{N}]+)*)*$/u.test(inner)) return true continue } if (raw.includes('.') || raw.includes('-')) { if (!/^[\p{L}\p{N}]+(?:[.-][\p{L}\p{N}]+)*$/u.test(raw)) return true continue } if (!/^[\p{L}\p{N}]+$/u.test(raw)) return true } return false } function runLiteralContentSearch( db: { prepare: (sql: string) => { all: (...params: any[]) => Record[] } }, source: string | undefined, query: string, limit: number, ): Record[] { const loweredQuery = query.toLowerCase() const likePattern = buildLikePattern(loweredQuery) const sourceClause = source ? 'AND s.source = ?' : '' const sourceParams = source ? [source] : [] const likeSql = ` WITH base AS ( SELECT ${SESSION_SELECT}, s.parent_session_id AS parent_session_id FROM sessions s WHERE s.source != 'tool' AND s.id NOT LIKE 'compress_%' ${sourceClause} ) SELECT base.*, m.id AS matched_message_id, substr( m.content, max(1, instr(LOWER(m.content), ?) - 40), 120 ) AS snippet, 0 AS rank FROM base JOIN messages m ON m.session_id = base.id WHERE LOWER(m.content) LIKE ? ESCAPE '\\' ORDER BY base.last_active DESC, m.timestamp DESC LIMIT ? ` return db.prepare(likeSql).all(...sourceParams, loweredQuery, likePattern, limit) as Record[] } function sanitizeFtsQuery(query: string): string { const quotedParts: string[] = [] const preserved = query.replace(/"[^"]*"/g, (match) => { quotedParts.push(match) return `\u0000Q${quotedParts.length - 1}\u0000` }) let sanitized = preserved.replace(/[+{}()"^]/g, ' ') sanitized = sanitized.replace(/\*+/g, '*') sanitized = sanitized.replace(/(^|\s)\*/g, '$1') sanitized = sanitized.trim().replace(/^(AND|OR|NOT)\b\s*/i, '') sanitized = sanitized.trim().replace(/\s+(AND|OR|NOT)\s*$/i, '') sanitized = sanitized.replace(/\b([\p{L}\p{N}]+(?:[.-][\p{L}\p{N}]+)+)\b/gu, '"$1"') for (let i = 0; i < quotedParts.length; i += 1) { sanitized = sanitized.replace(`\u0000Q${i}\u0000`, quotedParts[i]) } return sanitized.trim() } function toPrefixQuery(query: string): string { const tokens = query.match(/"[^"]*"\*?|\S+/g) if (!tokens) return '' return tokens .map((token) => { if (token === 'AND' || token === 'OR' || token === 'NOT') return token if (token.startsWith('"') && token.endsWith('"')) return token if (token.endsWith('*')) return token return `${token}*` }) .join(' ') } function mapSearchRow(row: Record): HermesSessionSearchRow { return { ...mapRow(row), matched_message_id: normalizeNullableNumber(row.matched_message_id), snippet: String(row.snippet || row.preview || ''), rank: Number.isFinite(Number(row.rank)) ? Number(row.rank) : 0, } } function mapInternalSessionRow(row: Record): HermesSessionInternalRow { return { ...mapRow(row), parent_session_id: normalizeNullableString(row.parent_session_id), } } function parseToolCalls(value: unknown): any[] | null { if (value == null || value === '') return null if (Array.isArray(value)) return value if (typeof value !== 'string') return null try { const parsed = JSON.parse(value) return Array.isArray(parsed) ? parsed : null } catch { return null } } function normalizeMessageId(value: unknown): number | string { if (typeof value === 'number' && Number.isFinite(value)) return value if (typeof value === 'bigint') return Number(value) const asNumber = Number(value) if (Number.isInteger(asNumber)) return asNumber return String(value || '') } function mapMessageRow(row: Record): HermesMessageRow { const reasoning = normalizeNullableString(row.reasoning) || normalizeNullableString(row.reasoning_content) return { id: normalizeMessageId(row.id), session_id: String(row.session_id || ''), role: String(row.role || ''), content: row.content == null ? '' : String(row.content), tool_call_id: normalizeNullableString(row.tool_call_id), tool_calls: parseToolCalls(row.tool_calls), tool_name: normalizeNullableString(row.tool_name), timestamp: normalizeNumber(row.timestamp), token_count: normalizeNullableNumber(row.token_count), finish_reason: normalizeNullableString(row.finish_reason), reasoning, reasoning_details: normalizeNullableString(row.reasoning_details), codex_reasoning_items: normalizeNullableString(row.codex_reasoning_items), reasoning_content: normalizeNullableString(row.reasoning_content), } } function isCompressionEnded(session: HermesSessionInternalRow | undefined): boolean { return !!session && COMPRESSION_END_REASONS.has(String(session.end_reason || '')) } function isCompressionContinuation(parent: HermesSessionInternalRow | undefined, child: HermesSessionInternalRow | undefined): boolean { if (!parent || !child || !isCompressionEnded(parent) || parent.ended_at == null) return false return child.source !== 'tool' && Number(child.started_at || 0) >= Number(parent.ended_at || 0) } function latestSessionInChain(chain: HermesSessionInternalRow[]): HermesSessionInternalRow { return chain.reduce((latest, session) => { const latestStarted = Number(latest.started_at || 0) const sessionStarted = Number(session.started_at || 0) if (sessionStarted !== latestStarted) return sessionStarted > latestStarted ? session : latest return session.id.localeCompare(latest.id) > 0 ? session : latest }, chain[0]) } function projectSessionSummary(root: HermesSessionInternalRow, chain: HermesSessionInternalRow[]): HermesSessionRow { const latest = latestSessionInChain(chain) const { parent_session_id: _parentSessionId, ...rootRow } = root return { ...rootRow, id: latest.id, model: latest.model || root.model, title: latest.title || root.title, ended_at: latest.ended_at, end_reason: latest.end_reason, message_count: latest.message_count, tool_call_count: latest.tool_call_count, input_tokens: latest.input_tokens, output_tokens: latest.output_tokens, cache_read_tokens: latest.cache_read_tokens, cache_write_tokens: latest.cache_write_tokens, reasoning_tokens: latest.reasoning_tokens, billing_provider: latest.billing_provider ?? root.billing_provider, estimated_cost_usd: latest.estimated_cost_usd, actual_cost_usd: latest.actual_cost_usd, cost_status: latest.cost_status, preview: latest.preview || root.preview, last_active: latest.last_active || root.last_active, } } // --- In-memory session index for chain traversal --- interface SessionIndex { byId: Map childrenByParent: Map } function loadAllSessions(db: { prepare: (sql: string) => { all: (...params: any[]) => Record[] } }): SessionIndex { const rows = db.prepare(` SELECT ${SESSION_SELECT}, s.parent_session_id AS parent_session_id FROM sessions s WHERE s.source != 'tool' AND s.id NOT LIKE 'compress_%' `).all() as Record[] const sessions = rows.map(mapInternalSessionRow) const byId = new Map(sessions.map(s => [s.id, s])) const childrenByParent = new Map() for (const s of sessions) { const key = s.parent_session_id ?? '' const list = childrenByParent.get(key) || [] list.push(s.id) childrenByParent.set(key, list) } return { byId, childrenByParent } } function getLatestContinuationChild( parent: HermesSessionInternalRow, idx: SessionIndex, ): HermesSessionInternalRow | null { if (!isCompressionEnded(parent) || parent.ended_at == null) return null const candidates = (idx.childrenByParent.get(parent.id) || []) .map(id => idx.byId.get(id)) .filter((c): c is HermesSessionInternalRow => !!c) .filter(c => Number(c.started_at || 0) >= Number(parent.ended_at || 0)) .sort((a, b) => { const aDelta = Number(a.started_at || 0) - Number(parent.ended_at || 0) const bDelta = Number(b.started_at || 0) - Number(parent.ended_at || 0) if (aDelta !== bDelta) return aDelta - bDelta return b.id.localeCompare(a.id) }) return candidates[0] || null } function collectCompressionPath( session: HermesSessionInternalRow, idx: SessionIndex, ): HermesSessionInternalRow[] { const reversed: HermesSessionInternalRow[] = [session] const seen = new Set() let current: HermesSessionInternalRow | null = session for (let depth = 0; current && current.parent_session_id && depth < 100 && !seen.has(current.id); depth += 1) { seen.add(current.id) const parent = idx.byId.get(current.parent_session_id) if (!parent || !isCompressionContinuation(parent, current)) break reversed.push(parent) current = parent } return reversed.reverse() } function extendCompressionChain( chain: HermesSessionInternalRow[], idx: SessionIndex, ): HermesSessionInternalRow[] { const result = [...chain] const seen = new Set(result.map(s => s.id)) let current: HermesSessionInternalRow | null = result[result.length - 1] || null for (let depth = 0; current && depth < 100; depth += 1) { const next = getLatestContinuationChild(current, idx) if (!next || seen.has(next.id)) break result.push(next) seen.add(next.id) current = next } return result } function collectSessionChain( root: HermesSessionInternalRow, idx: SessionIndex, ): HermesSessionInternalRow[] { return extendCompressionChain([root], idx) } function collectSessionChainForMatchedSession( session: HermesSessionInternalRow, idx: SessionIndex, ): HermesSessionInternalRow[] { return extendCompressionChain(collectCompressionPath(session, idx), idx) } type SessionDbLike = { prepare: (sql: string) => { all: (...params: any[]) => Record[] } } function searchCandidateLimit(limit: number): number { return Math.max(limit * SEARCH_CANDIDATE_MULTIPLIER, SEARCH_CANDIDATE_MIN) } function projectSearchRow( row: Record, idx: SessionIndex, source?: string, ): HermesSessionSearchRow | null { const matchedSession = mapInternalSessionRow(row) if (!matchedSession.id) return null const chain = collectSessionChainForMatchedSession(matchedSession, idx) const root = chain[0] if (!root) return null if (source && matchedSession.source !== source) return null const projected = projectSessionSummary(root, chain) return { ...projected, matched_message_id: normalizeNullableNumber(row.matched_message_id), snippet: String(row.snippet || row.preview || ''), rank: Number.isFinite(Number(row.rank)) ? Number(row.rank) : 0, } } function aggregateSessionDetail( chain: HermesSessionInternalRow[], messages: HermesMessageRow[], requestedSessionId: string, ): HermesSessionDetailRow { const root = chain[0] const latest = latestSessionInChain(chain) const costStatuses = Array.from(new Set(chain.map(session => String(session.cost_status || '')).filter(Boolean))) const actualCosts = chain .map(session => session.actual_cost_usd) .filter((value): value is number => value != null) const firstPreview = chain.map(session => session.preview).find(Boolean) || root.preview const { parent_session_id: _parentSessionId, ...rootRow } = root return { ...rootRow, id: requestedSessionId, source: latest.source || root.source, title: latest.title || root.title || (firstPreview ? (firstPreview.length > 40 ? `${firstPreview.slice(0, 40)}...` : firstPreview) : null), preview: latest.preview || root.preview || firstPreview || '', model: latest.model || root.model, ended_at: latest.ended_at, end_reason: latest.end_reason, last_active: Math.max(...chain.map(session => session.last_active || session.started_at || 0)), message_count: chain.reduce((sum, session) => sum + Number(session.message_count || 0), 0), tool_call_count: chain.reduce((sum, session) => sum + Number(session.tool_call_count || 0), 0), input_tokens: chain.reduce((sum, session) => sum + Number(session.input_tokens || 0), 0), output_tokens: chain.reduce((sum, session) => sum + Number(session.output_tokens || 0), 0), cache_read_tokens: chain.reduce((sum, session) => sum + Number(session.cache_read_tokens || 0), 0), cache_write_tokens: chain.reduce((sum, session) => sum + Number(session.cache_write_tokens || 0), 0), reasoning_tokens: chain.reduce((sum, session) => sum + Number(session.reasoning_tokens || 0), 0), billing_provider: latest.billing_provider ?? root.billing_provider, estimated_cost_usd: chain.reduce((sum, session) => sum + Number(session.estimated_cost_usd || 0), 0), actual_cost_usd: actualCosts.length ? actualCosts.reduce((sum, value) => sum + Number(value || 0), 0) : null, cost_status: costStatuses.length === 1 ? costStatuses[0] : (costStatuses.length > 1 ? 'mixed' : ''), messages, thread_session_count: chain.length, } } async function openSessionDb() { if (!SQLITE_AVAILABLE) { throw new Error(`node:sqlite requires Node >= 22.5, current: ${process.versions.node}`) } const { DatabaseSync } = await import('node:sqlite') const dbPath = sessionDbPath() console.log(`[sessions-db] Opening session db: ${dbPath}`) try { return new DatabaseSync(dbPath, { open: true, readOnly: true }) } catch (err: any) { console.error(`[sessions-db] Failed to open session db at ${dbPath}:`, err.message) throw err } } /** * Lightweight alternative: get messages + session row for a single session ID * without chain traversal. Used by syncFromHermes for ephemeral sessions. */ export async function getSessionMessagesFromDb(sessionId: string): Promise<{ messages: HermesMessageRow[] session: HermesSessionRow | null } | null> { const db = await openSessionDb() try { const sessionRow = db.prepare(` SELECT ${SESSION_SELECT} FROM sessions s WHERE s.id = ? `).get(sessionId) as Record | undefined const messageRows = db.prepare(` SELECT id, session_id, role, content, tool_call_id, tool_calls, tool_name, timestamp, token_count, finish_reason, reasoning, reasoning_details, codex_reasoning_items, reasoning_content FROM messages WHERE session_id = ? ORDER BY timestamp, id `).all(sessionId) as Record[] return { messages: messageRows.map(mapMessageRow), session: sessionRow ? mapRow(sessionRow) : null, } } finally { db.close() } } export async function getSessionDetailFromDb(sessionId: string): Promise { const db = await openSessionDb() try { const idx = loadAllSessions(db) const requested = idx.byId.get(sessionId) || null if (!requested) return null const chain = collectSessionChainForMatchedSession(requested, idx) if (!chain.length) return null const ids = chain.map(session => session.id) const placeholders = ids.map(() => '?').join(', ') const messageRows = db.prepare(` SELECT id, session_id, role, content, tool_call_id, tool_calls, tool_name, timestamp, token_count, finish_reason, reasoning, reasoning_details, codex_reasoning_items, reasoning_content FROM messages WHERE session_id IN (${placeholders}) ORDER BY timestamp, id `).all(...ids) as Record[] const messages = messageRows.map(mapMessageRow) return aggregateSessionDetail(chain, messages, sessionId) } finally { db.close() } } export async function getSessionDetailFromDbWithProfile(sessionId: string, profile: string): Promise { const { DatabaseSync } = await import('node:sqlite') const dbPath = `${getProfileDir(profile)}/state.db` const db = new DatabaseSync(dbPath, { open: true, readOnly: true }) try { const idx = loadAllSessions(db) const requested = idx.byId.get(sessionId) || null if (!requested) return null const chain = collectSessionChainForMatchedSession(requested, idx) if (!chain.length) return null const ids = chain.map(session => session.id) const placeholders = ids.map(() => '?').join(', ') const messageRows = db.prepare(` SELECT id, session_id, role, content, tool_call_id, tool_calls, tool_name, timestamp, token_count, finish_reason, reasoning, reasoning_details, codex_reasoning_items, reasoning_content FROM messages WHERE session_id IN (${placeholders}) ORDER BY timestamp, id `).all(...ids) as Record[] const messages = messageRows.map(mapMessageRow) return aggregateSessionDetail(chain, messages, sessionId) } finally { db.close() } } export interface HermesUsageStats extends LocalUsageStats { cost: number total_api_calls: number } function tableHasColumn( db: { prepare: (sql: string) => { all: (...params: any[]) => Record[] } }, tableName: string, columnName: string, ): boolean { const columns = db.prepare(`PRAGMA table_info(${tableName})`).all() return columns.some(column => String(column.name || '') === columnName) } export async function getUsageStatsFromDb( days = 30, nowSeconds = Math.floor(Date.now() / 1000), ): Promise { const empty: HermesUsageStats = { input_tokens: 0, output_tokens: 0, cache_read_tokens: 0, cache_write_tokens: 0, reasoning_tokens: 0, sessions: 0, by_model: [], by_day: [], cost: 0, total_api_calls: 0, } const normalizedDays = Number.isFinite(days) ? days : 30 const safeDays = Math.max(1, Math.floor(normalizedDays)) const since = nowSeconds - safeDays * 24 * 60 * 60 const db = await openSessionDb() try { const apiCallsExpr = tableHasColumn(db, 'sessions', 'api_call_count') ? 'COALESCE(SUM(api_call_count), 0)' : '0' const sourceFilter = tableHasColumn(db, 'sessions', 'source') ? " AND COALESCE(source, '') != 'api_server'" : '' const totals = db.prepare(` SELECT COALESCE(SUM(input_tokens), 0) AS input_tokens, COALESCE(SUM(output_tokens), 0) AS output_tokens, COALESCE(SUM(cache_read_tokens), 0) AS cache_read_tokens, COALESCE(SUM(cache_write_tokens), 0) AS cache_write_tokens, COALESCE(SUM(reasoning_tokens), 0) AS reasoning_tokens, COALESCE(SUM(COALESCE(actual_cost_usd, estimated_cost_usd, 0)), 0) AS cost, COUNT(*) AS sessions, ${apiCallsExpr} AS total_api_calls FROM sessions WHERE started_at > ?${sourceFilter} `).get(since) as Record | undefined if (!totals) return empty const byModel = db.prepare(` SELECT COALESCE(model, '') AS model, COALESCE(SUM(input_tokens), 0) AS input_tokens, COALESCE(SUM(output_tokens), 0) AS output_tokens, COALESCE(SUM(cache_read_tokens), 0) AS cache_read_tokens, COALESCE(SUM(cache_write_tokens), 0) AS cache_write_tokens, COALESCE(SUM(reasoning_tokens), 0) AS reasoning_tokens, COUNT(*) AS sessions FROM sessions WHERE started_at > ?${sourceFilter} AND model IS NOT NULL GROUP BY model ORDER BY COALESCE(SUM(input_tokens), 0) + COALESCE(SUM(output_tokens), 0) DESC `).all(since).map(row => ({ model: String(row.model || ''), input_tokens: normalizeNumber(row.input_tokens), output_tokens: normalizeNumber(row.output_tokens), cache_read_tokens: normalizeNumber(row.cache_read_tokens), cache_write_tokens: normalizeNumber(row.cache_write_tokens), reasoning_tokens: normalizeNumber(row.reasoning_tokens), sessions: normalizeNumber(row.sessions), })) const byDay = db.prepare(` SELECT date(started_at, 'unixepoch') AS date, COALESCE(SUM(input_tokens), 0) AS input_tokens, COALESCE(SUM(output_tokens), 0) AS output_tokens, COALESCE(SUM(cache_read_tokens), 0) AS cache_read_tokens, COALESCE(SUM(cache_write_tokens), 0) AS cache_write_tokens, COUNT(*) AS sessions, COALESCE(SUM(COALESCE(actual_cost_usd, estimated_cost_usd, 0)), 0) AS cost FROM sessions WHERE started_at > ?${sourceFilter} GROUP BY date ORDER BY date ASC `).all(since).map(row => ({ date: String(row.date || ''), input_tokens: normalizeNumber(row.input_tokens), output_tokens: normalizeNumber(row.output_tokens), cache_read_tokens: normalizeNumber(row.cache_read_tokens), cache_write_tokens: normalizeNumber(row.cache_write_tokens), sessions: normalizeNumber(row.sessions), errors: 0, cost: normalizeNumber(row.cost), })) return { input_tokens: normalizeNumber(totals.input_tokens), output_tokens: normalizeNumber(totals.output_tokens), cache_read_tokens: normalizeNumber(totals.cache_read_tokens), cache_write_tokens: normalizeNumber(totals.cache_write_tokens), reasoning_tokens: normalizeNumber(totals.reasoning_tokens), sessions: normalizeNumber(totals.sessions), by_model: byModel, by_day: byDay, cost: normalizeNumber(totals.cost), total_api_calls: normalizeNumber(totals.total_api_calls), } } finally { db.close() } } export async function listSessionSummaries(source?: string, limit = 2000, profile?: string): Promise { if (!SQLITE_AVAILABLE) { throw new Error(`node:sqlite requires Node >= 22.5, current: ${process.versions.node}`) } const { DatabaseSync } = await import('node:sqlite') const dbPath = profile ? `${getProfileDir(profile)}/state.db` : sessionDbPath() const db = new DatabaseSync(dbPath, { open: true, readOnly: true }) try { const clauses = ["s.parent_session_id IS NULL", "s.source != 'tool'", "s.id NOT LIKE 'compress_%'"] const params: any[] = [] if (source) { clauses.push('s.source = ?') params.push(source) } params.push(Math.max(limit * 4, limit)) const rawRows = db.prepare(` SELECT ${SESSION_SELECT}, s.parent_session_id AS parent_session_id FROM sessions s WHERE ${clauses.join(' AND ')} ORDER BY s.started_at DESC LIMIT ? `).all(...params) as Record[] | undefined const roots = (Array.isArray(rawRows) ? rawRows : []).map(mapInternalSessionRow) const idx = loadAllSessions(db) return roots .map(root => projectSessionSummary(root, collectSessionChain(root, idx))) .sort((a, b) => Number(b.last_active || b.started_at || 0) - Number(a.last_active || a.started_at || 0)) .slice(0, limit) } finally { db.close() } } export async function searchSessionSummaries( query: string, source?: string, limit = 20, ): Promise { if (!SQLITE_AVAILABLE) { throw new Error(`node:sqlite requires Node >= 22.5, current: ${process.versions.node}`) } const trimmed = query.trim() if (!trimmed) { const recent = await listSessionSummaries(source, limit) return recent.map(row => ({ ...row, matched_message_id: null, snippet: row.preview, rank: 0, })) } const { DatabaseSync } = await import('node:sqlite') const db = new DatabaseSync(sessionDbPath(), { open: true, readOnly: true }) const normalized = sanitizeFtsQuery(trimmed) const prefixQuery = toPrefixQuery(normalized) const titlePattern = buildLikePattern(normalizeTitleLikeQuery(trimmed).toLowerCase()) const useLiteralContentSearch = containsCjk(trimmed) || shouldUseLiteralContentSearch(trimmed) const candidateLimit = searchCandidateLimit(limit) let titleRows: Record[] = [] try { const sourceClause = source ? 'AND s.source = ?' : '' const sourceParams = source ? [source] : [] const allSessionsBaseSql = ` SELECT ${SESSION_SELECT}, s.parent_session_id AS parent_session_id FROM sessions s WHERE s.source != 'tool' AND s.id NOT LIKE 'compress_%' ${sourceClause} ` const titleSql = ` WITH base AS ( ${allSessionsBaseSql} ) SELECT base.*, NULL AS matched_message_id, CASE WHEN base.title IS NOT NULL AND base.title != '' THEN base.title ELSE base.preview END AS snippet, 0 AS rank FROM base WHERE LOWER(COALESCE(base.title, '')) LIKE ? ESCAPE '\\' ORDER BY base.last_active DESC LIMIT ? ` const titleStatement = db.prepare(titleSql) titleRows = titleStatement.all(...sourceParams, titlePattern, candidateLimit) as Record[] const contentSql = ` WITH base AS ( ${allSessionsBaseSql} ) SELECT base.*, m.id AS matched_message_id, snippet(messages_fts, 0, '>>>', '<<<', '...', 40) AS snippet, bm25(messages_fts) AS rank FROM messages_fts JOIN messages m ON m.id = messages_fts.rowid JOIN base ON base.id = m.session_id WHERE messages_fts MATCH ? ORDER BY rank, base.last_active DESC LIMIT ? ` const contentRows = useLiteralContentSearch ? runLiteralContentSearch(db, source, trimmed, candidateLimit) : prefixQuery ? (db.prepare(contentSql).all(...sourceParams, prefixQuery, candidateLimit) as Record[]) : [] const idx = loadAllSessions(db) const merged = new Map() for (const row of titleRows) { const mapped = projectSearchRow(row, idx, source) if (mapped) merged.set(mapped.id, mapped) } for (const row of contentRows) { const mapped = projectSearchRow(row, idx, source) if (mapped && !merged.has(mapped.id)) { merged.set(mapped.id, mapped) } } const items = [...merged.values()] items.sort((a, b) => { if (a.rank !== b.rank) return a.rank - b.rank return b.last_active - a.last_active }) return items.slice(0, limit) } catch (_err) { // FTS queries can fail for various inputs (pure numbers, special syntax, etc.) // Fall back to title-only LIKE results + literal content search for CJK const likeRows = containsCjk(normalized) ? runLiteralContentSearch(db, source, trimmed, candidateLimit) : [] const idx2 = loadAllSessions(db) const merged = new Map() for (const row of titleRows) { const mapped = projectSearchRow(row, idx2, source) if (mapped) merged.set(mapped.id, mapped) } for (const row of likeRows) { const mapped = projectSearchRow(row, idx2, source) if (mapped && !merged.has(mapped.id)) { merged.set(mapped.id, mapped) } } const items = [...merged.values()] items.sort((a, b) => { if (a.rank !== b.rank) return a.rank - b.rank return b.last_active - a.last_active }) return items.slice(0, limit) } finally { db.close() } }