2026-04-22 16:14:50 +08:00
|
|
|
import { getActiveProfileDir } from '../../services/hermes/hermes-profile'
|
2026-04-17 12:56:41 -04:00
|
|
|
|
2026-04-18 09:34:59 +08:00
|
|
|
const SQLITE_AVAILABLE = (() => {
|
|
|
|
|
const [major, minor] = process.versions.node.split('.').map(Number)
|
|
|
|
|
return major > 22 || (major === 22 && minor >= 5)
|
|
|
|
|
})()
|
|
|
|
|
|
2026-04-25 16:23:33 +02:00
|
|
|
const LINEAGE_TOLERANCE_SECONDS = 3
|
|
|
|
|
|
2026-04-17 12:56:41 -04:00
|
|
|
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
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-22 14:00:34 +08:00
|
|
|
export interface HermesSessionSearchRow extends HermesSessionRow {
|
|
|
|
|
matched_message_id: number | null
|
|
|
|
|
snippet: string
|
|
|
|
|
rank: number
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-25 16:23:33 +02:00
|
|
|
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
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-17 12:56:41 -04:00
|
|
|
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<string, unknown>): HermesSessionRow {
|
|
|
|
|
const startedAt = normalizeNumber(row.started_at)
|
2026-04-18 08:53:45 +08:00
|
|
|
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)
|
2026-04-17 12:56:41 -04:00
|
|
|
return {
|
|
|
|
|
id: String(row.id || ''),
|
|
|
|
|
source: String(row.source || ''),
|
|
|
|
|
user_id: normalizeNullableString(row.user_id),
|
|
|
|
|
model: String(row.model || ''),
|
2026-04-18 08:53:45 +08:00
|
|
|
title,
|
2026-04-17 12:56:41 -04:00
|
|
|
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),
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-22 14:00:34 +08:00
|
|
|
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
|
|
|
|
|
`
|
|
|
|
|
|
|
|
|
|
const SESSION_FROM = `
|
2026-04-17 12:56:41 -04:00
|
|
|
FROM sessions s
|
|
|
|
|
WHERE s.parent_session_id IS NULL
|
|
|
|
|
AND s.source != 'tool'
|
|
|
|
|
`
|
|
|
|
|
|
2026-04-22 14:00:34 +08:00
|
|
|
function buildBaseSessionSql(source?: string): { sql: string, params: any[] } {
|
|
|
|
|
const sql = source
|
|
|
|
|
? `SELECT ${SESSION_SELECT}${SESSION_FROM}\n AND s.source = ?`
|
|
|
|
|
: `SELECT ${SESSION_SELECT}${SESSION_FROM}`
|
|
|
|
|
return { sql, params: source ? [source] : [] }
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function buildListSessionSql(source?: string, limit = 2000): { sql: string, params: any[] } {
|
|
|
|
|
const base = buildBaseSessionSql(source)
|
|
|
|
|
return {
|
|
|
|
|
sql: `${base.sql}\n ORDER BY s.started_at DESC\n LIMIT ?`,
|
|
|
|
|
params: [...base.params, limit],
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-24 20:41:14 +08:00
|
|
|
function escapeLikePattern(value: string): string {
|
|
|
|
|
return value.replace(/[\\%_]/g, (match) => `\\${match}`)
|
2026-04-23 04:49:00 +02:00
|
|
|
}
|
|
|
|
|
|
2026-04-24 20:41:14 +08:00
|
|
|
function buildLikePattern(value: string): string {
|
|
|
|
|
return `%${escapeLikePattern(value)}%`
|
2026-04-23 11:18:56 +08:00
|
|
|
}
|
|
|
|
|
|
2026-04-24 20:41:14 +08:00
|
|
|
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(
|
2026-04-23 04:49:00 +02:00
|
|
|
db: { prepare: (sql: string) => { all: (...params: any[]) => Record<string, unknown>[] } },
|
|
|
|
|
source: string | undefined,
|
|
|
|
|
query: string,
|
2026-04-24 20:41:14 +08:00
|
|
|
limit: number,
|
2026-04-23 04:49:00 +02:00
|
|
|
): Record<string, unknown>[] {
|
|
|
|
|
const likeBase = buildBaseSessionSql(source)
|
2026-04-24 20:41:14 +08:00
|
|
|
const loweredQuery = query.toLowerCase()
|
|
|
|
|
const likePattern = buildLikePattern(loweredQuery)
|
2026-04-23 04:49:00 +02:00
|
|
|
const likeSql = `
|
|
|
|
|
WITH base AS (
|
|
|
|
|
${likeBase.sql}
|
|
|
|
|
)
|
|
|
|
|
SELECT
|
|
|
|
|
base.*,
|
|
|
|
|
m.id AS matched_message_id,
|
|
|
|
|
substr(
|
|
|
|
|
m.content,
|
2026-04-24 20:41:14 +08:00
|
|
|
max(1, instr(LOWER(m.content), ?) - 40),
|
2026-04-23 04:49:00 +02:00
|
|
|
120
|
|
|
|
|
) AS snippet,
|
|
|
|
|
0 AS rank
|
|
|
|
|
FROM base
|
|
|
|
|
JOIN messages m ON m.session_id = base.id
|
2026-04-24 20:41:14 +08:00
|
|
|
WHERE LOWER(m.content) LIKE ? ESCAPE '\\'
|
2026-04-23 04:49:00 +02:00
|
|
|
ORDER BY base.last_active DESC, m.timestamp DESC
|
2026-04-24 20:41:14 +08:00
|
|
|
LIMIT ?
|
2026-04-23 04:49:00 +02:00
|
|
|
`
|
2026-04-24 20:41:14 +08:00
|
|
|
return db.prepare(likeSql).all(...likeBase.params, loweredQuery, likePattern, limit * 4) as Record<string, unknown>[]
|
2026-04-23 04:49:00 +02:00
|
|
|
}
|
|
|
|
|
|
2026-04-22 14:00:34 +08:00
|
|
|
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, '')
|
2026-04-24 20:41:14 +08:00
|
|
|
sanitized = sanitized.replace(/\b([\p{L}\p{N}]+(?:[.-][\p{L}\p{N}]+)+)\b/gu, '"$1"')
|
2026-04-22 14:00:34 +08:00
|
|
|
|
|
|
|
|
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 {
|
2026-04-24 20:41:14 +08:00
|
|
|
const tokens = query.match(/"[^"]*"\*?|\S+/g)
|
2026-04-22 14:00:34 +08:00
|
|
|
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<string, unknown>): 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,
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-25 16:23:33 +02:00
|
|
|
function mapInternalSessionRow(row: Record<string, unknown>): 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<string, unknown>): 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 timingMatchesParent(parent: HermesSessionInternalRow | undefined, child: HermesSessionInternalRow | undefined): boolean {
|
|
|
|
|
if (!parent || !child || parent.ended_at == null) return false
|
|
|
|
|
return Math.abs(Number(child.started_at || 0) - Number(parent.ended_at || 0)) <= LINEAGE_TOLERANCE_SECONDS
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function continuationCandidates(
|
|
|
|
|
parent: HermesSessionInternalRow,
|
|
|
|
|
byId: Map<string, HermesSessionInternalRow>,
|
|
|
|
|
childrenByParent: Map<string | null, string[]>,
|
|
|
|
|
): HermesSessionInternalRow[] {
|
|
|
|
|
return (childrenByParent.get(parent.id) || [])
|
|
|
|
|
.map(childId => byId.get(childId))
|
|
|
|
|
.filter((child): child is HermesSessionInternalRow => !!child)
|
|
|
|
|
.filter(child => child.source !== 'tool')
|
|
|
|
|
.filter(child => child.source === parent.source)
|
|
|
|
|
.filter(child => timingMatchesParent(parent, child))
|
|
|
|
|
.sort((a, b) => {
|
|
|
|
|
const aDelta = Math.abs(Number(a.started_at || 0) - Number(parent.ended_at || 0))
|
|
|
|
|
const bDelta = Math.abs(Number(b.started_at || 0) - Number(parent.ended_at || 0))
|
|
|
|
|
if (aDelta !== bDelta) return aDelta - bDelta
|
|
|
|
|
return a.id.localeCompare(b.id)
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function normalizeComparableText(value: unknown): string {
|
|
|
|
|
return String(value || '').replace(/\s+/g, ' ').trim().toLowerCase()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function nextContinuationChild(
|
|
|
|
|
parent: HermesSessionInternalRow,
|
|
|
|
|
byId: Map<string, HermesSessionInternalRow>,
|
|
|
|
|
childrenByParent: Map<string | null, string[]>,
|
|
|
|
|
): HermesSessionInternalRow | null {
|
|
|
|
|
if (parent.end_reason !== 'compression') return null
|
|
|
|
|
const candidates = continuationCandidates(parent, byId, childrenByParent)
|
|
|
|
|
if (candidates.length === 1) return candidates[0]
|
|
|
|
|
|
|
|
|
|
const exactPreviewMatches = candidates.filter(child => {
|
|
|
|
|
const childPreview = normalizeComparableText(child.preview)
|
|
|
|
|
const parentPreview = normalizeComparableText(parent.preview)
|
|
|
|
|
return !!childPreview && childPreview === parentPreview
|
|
|
|
|
})
|
|
|
|
|
return exactPreviewMatches.length === 1 ? exactPreviewMatches[0] : null
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function collectSessionChain(
|
|
|
|
|
rootId: string,
|
|
|
|
|
byId: Map<string, HermesSessionInternalRow>,
|
|
|
|
|
childrenByParent: Map<string | null, string[]>,
|
|
|
|
|
): HermesSessionInternalRow[] {
|
|
|
|
|
const chain: HermesSessionInternalRow[] = []
|
|
|
|
|
const seen = new Set<string>()
|
|
|
|
|
let current = byId.get(rootId) || null
|
|
|
|
|
while (current && !seen.has(current.id)) {
|
|
|
|
|
chain.push(current)
|
|
|
|
|
seen.add(current.id)
|
|
|
|
|
current = nextContinuationChild(current, byId, childrenByParent)
|
|
|
|
|
}
|
|
|
|
|
return chain
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function aggregateSessionDetail(chain: HermesSessionInternalRow[], messages: HermesMessageRow[]): HermesSessionDetailRow {
|
|
|
|
|
const root = chain[0]
|
|
|
|
|
const last = chain[chain.length - 1] || root
|
|
|
|
|
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
|
|
|
|
|
|
|
|
|
|
return {
|
|
|
|
|
...root,
|
|
|
|
|
title: root.title || (firstPreview ? (firstPreview.length > 40 ? `${firstPreview.slice(0, 40)}...` : firstPreview) : null),
|
|
|
|
|
preview: root.preview || firstPreview || '',
|
|
|
|
|
model: last.model || root.model,
|
|
|
|
|
ended_at: last.ended_at,
|
|
|
|
|
end_reason: last.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: last.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')
|
|
|
|
|
return new DatabaseSync(sessionDbPath(), { open: true, readOnly: true })
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export async function getSessionDetailFromDb(sessionId: string): Promise<HermesSessionDetailRow | null> {
|
|
|
|
|
const db = await openSessionDb()
|
|
|
|
|
try {
|
|
|
|
|
const rows = db.prepare(`
|
|
|
|
|
SELECT
|
|
|
|
|
${SESSION_SELECT},
|
|
|
|
|
s.parent_session_id AS parent_session_id
|
|
|
|
|
FROM sessions s
|
|
|
|
|
WHERE s.source != 'tool'
|
|
|
|
|
`).all() as Record<string, unknown>[]
|
|
|
|
|
|
|
|
|
|
const sessions = rows.map(mapInternalSessionRow)
|
|
|
|
|
const byId = new Map(sessions.map(session => [session.id, session]))
|
|
|
|
|
const root = byId.get(sessionId)
|
|
|
|
|
if (!root) return null
|
|
|
|
|
|
|
|
|
|
const childrenByParent = new Map<string | null, string[]>()
|
|
|
|
|
for (const session of sessions) {
|
|
|
|
|
const key = session.parent_session_id ?? null
|
|
|
|
|
const siblings = childrenByParent.get(key) || []
|
|
|
|
|
siblings.push(session.id)
|
|
|
|
|
childrenByParent.set(key, siblings)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const chain = collectSessionChain(sessionId, byId, childrenByParent)
|
|
|
|
|
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<string, unknown>[]
|
|
|
|
|
|
|
|
|
|
const messages = messageRows.map(mapMessageRow)
|
|
|
|
|
return aggregateSessionDetail(chain, messages)
|
|
|
|
|
} finally {
|
|
|
|
|
db.close()
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-17 12:56:41 -04:00
|
|
|
export async function listSessionSummaries(source?: string, limit = 2000): Promise<HermesSessionRow[]> {
|
2026-04-18 09:34:59 +08:00
|
|
|
if (!SQLITE_AVAILABLE) {
|
|
|
|
|
throw new Error(`node:sqlite requires Node >= 22.5, current: ${process.versions.node}`)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const { DatabaseSync } = await import('node:sqlite')
|
2026-04-17 12:56:41 -04:00
|
|
|
const db = new DatabaseSync(sessionDbPath(), { open: true, readOnly: true })
|
|
|
|
|
|
|
|
|
|
try {
|
2026-04-22 14:00:34 +08:00
|
|
|
const { sql, params } = buildListSessionSql(source, limit)
|
2026-04-17 12:56:41 -04:00
|
|
|
const statement = db.prepare(sql)
|
2026-04-22 14:00:34 +08:00
|
|
|
const rows = statement.all(...params) as Record<string, unknown>[]
|
2026-04-17 12:56:41 -04:00
|
|
|
|
|
|
|
|
return rows.map(mapRow)
|
|
|
|
|
} finally {
|
|
|
|
|
db.close()
|
|
|
|
|
}
|
|
|
|
|
}
|
2026-04-22 14:00:34 +08:00
|
|
|
|
|
|
|
|
export async function searchSessionSummaries(
|
|
|
|
|
query: string,
|
|
|
|
|
source?: string,
|
|
|
|
|
limit = 20,
|
|
|
|
|
): Promise<HermesSessionSearchRow[]> {
|
|
|
|
|
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)
|
2026-04-24 20:41:14 +08:00
|
|
|
const titlePattern = buildLikePattern(normalizeTitleLikeQuery(trimmed).toLowerCase())
|
|
|
|
|
const useLiteralContentSearch = containsCjk(trimmed) || shouldUseLiteralContentSearch(trimmed)
|
2026-04-23 04:49:00 +02:00
|
|
|
let titleRows: Record<string, unknown>[] = []
|
2026-04-22 14:00:34 +08:00
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
const titleBase = buildBaseSessionSql(source)
|
|
|
|
|
const contentBase = buildBaseSessionSql(source)
|
|
|
|
|
|
|
|
|
|
const titleSql = `
|
|
|
|
|
WITH base AS (
|
|
|
|
|
${titleBase.sql}
|
|
|
|
|
)
|
|
|
|
|
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
|
2026-04-24 20:41:14 +08:00
|
|
|
WHERE LOWER(COALESCE(base.title, '')) LIKE ? ESCAPE '\\'
|
2026-04-22 14:00:34 +08:00
|
|
|
ORDER BY base.last_active DESC
|
|
|
|
|
LIMIT ?
|
|
|
|
|
`
|
|
|
|
|
|
|
|
|
|
const titleStatement = db.prepare(titleSql)
|
2026-04-24 20:41:14 +08:00
|
|
|
titleRows = titleStatement.all(...titleBase.params, titlePattern, limit) as Record<string, unknown>[]
|
2026-04-22 14:00:34 +08:00
|
|
|
|
|
|
|
|
const contentSql = `
|
|
|
|
|
WITH base AS (
|
|
|
|
|
${contentBase.sql}
|
|
|
|
|
)
|
|
|
|
|
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 ?
|
|
|
|
|
`
|
|
|
|
|
|
2026-04-24 20:41:14 +08:00
|
|
|
const contentRows = useLiteralContentSearch
|
|
|
|
|
? runLiteralContentSearch(db, source, trimmed, limit)
|
|
|
|
|
: prefixQuery
|
|
|
|
|
? (db.prepare(contentSql).all(...contentBase.params, prefixQuery, limit * 4) as Record<string, unknown>[])
|
|
|
|
|
: []
|
2026-04-22 14:00:34 +08:00
|
|
|
|
|
|
|
|
const merged = new Map<string, HermesSessionSearchRow>()
|
|
|
|
|
for (const row of titleRows) {
|
|
|
|
|
const mapped = mapSearchRow(row)
|
|
|
|
|
merged.set(mapped.id, mapped)
|
|
|
|
|
}
|
|
|
|
|
for (const row of contentRows) {
|
|
|
|
|
const mapped = mapSearchRow(row)
|
|
|
|
|
if (!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) {
|
2026-04-23 04:49:00 +02:00
|
|
|
const message = err instanceof Error ? err.message : String(err)
|
2026-04-22 14:00:34 +08:00
|
|
|
if (containsCjk(normalized)) {
|
2026-04-24 20:41:14 +08:00
|
|
|
const likeRows = runLiteralContentSearch(db, source, trimmed, limit)
|
2026-04-23 04:49:00 +02:00
|
|
|
const merged = new Map<string, HermesSessionSearchRow>()
|
|
|
|
|
for (const row of titleRows) {
|
|
|
|
|
const mapped = mapSearchRow(row)
|
|
|
|
|
merged.set(mapped.id, mapped)
|
|
|
|
|
}
|
|
|
|
|
for (const row of likeRows) {
|
|
|
|
|
const mapped = mapSearchRow(row)
|
|
|
|
|
if (!merged.has(mapped.id)) {
|
|
|
|
|
merged.set(mapped.id, mapped)
|
|
|
|
|
}
|
|
|
|
|
}
|
2026-04-24 20:41:14 +08:00
|
|
|
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)
|
2026-04-23 04:49:00 +02:00
|
|
|
}
|
|
|
|
|
|
2026-04-22 14:00:34 +08:00
|
|
|
throw new Error(`Failed to search sessions: ${message}`)
|
|
|
|
|
} finally {
|
|
|
|
|
db.close()
|
|
|
|
|
}
|
|
|
|
|
}
|