mirror of
https://github.com/NousResearch/hermes-agent.git
synced 2026-06-02 07:11:49 +00:00
Add a first-class active-session orchestrator for the Ink TUI: - list, activate, close, and launch live process-local TUI sessions - hydrate committed and in-flight output when switching sessions - dispatch a new prompt session from the +new row with session-scoped model picks - expose a clickable live-session count in the status chrome - preserve stable row order while initially focusing the current session - support mouse hit-testing for floating orchestrator overlays - add backend and frontend regression coverage for the lifecycle and UI helpers
366 lines
11 KiB
TypeScript
366 lines
11 KiB
TypeScript
import { writeFileSync } from 'node:fs'
|
|
|
|
import type { ScrollBoxHandle } from '@hermes/ink'
|
|
import { evictInkCaches } from '@hermes/ink'
|
|
import { type RefObject, useCallback } from 'react'
|
|
|
|
import { buildSetupRequiredSections, SETUP_REQUIRED_TITLE } from '../content/setup.js'
|
|
import { introMsg, toTranscriptMessages } from '../domain/messages.js'
|
|
import { ZERO } from '../domain/usage.js'
|
|
import { type GatewayClient } from '../gatewayClient.js'
|
|
import type {
|
|
SessionActivateResponse,
|
|
SessionCloseResponse,
|
|
SessionCreateResponse,
|
|
SessionInflightTurn,
|
|
SessionResumeResponse,
|
|
SessionTitleResponse,
|
|
SetupStatusResponse
|
|
} from '../gatewayTypes.js'
|
|
import { asRpcResult } from '../lib/rpc.js'
|
|
import type { Msg, PanelSection, SessionInfo, Usage } from '../types.js'
|
|
|
|
import type { ComposerActions, GatewayRpc, StateSetter } from './interfaces.js'
|
|
import { patchOverlayState } from './overlayStore.js'
|
|
import { turnController } from './turnController.js'
|
|
import { patchTurnState } from './turnStore.js'
|
|
import { getUiState, patchUiState } from './uiStore.js'
|
|
|
|
const usageFrom = (info: null | SessionInfo): Usage => (info?.usage ? { ...ZERO, ...info.usage } : ZERO)
|
|
|
|
const statusFromLiveSession = (status?: string, running = false) => {
|
|
if (status === 'waiting') {
|
|
return 'waiting for input…'
|
|
}
|
|
|
|
if (status === 'starting') {
|
|
return 'starting agent…'
|
|
}
|
|
|
|
return running || status === 'working' ? 'running…' : 'ready'
|
|
}
|
|
|
|
export const writeActiveSessionFile = (sessionId: null | string, file = process.env.HERMES_TUI_ACTIVE_SESSION_FILE) => {
|
|
if (!file || !sessionId) {
|
|
return
|
|
}
|
|
|
|
try {
|
|
writeFileSync(file, JSON.stringify({ session_id: sessionId }), { mode: 0o600 })
|
|
} catch {
|
|
// Best-effort shell epilogue hint only; never break live session changes.
|
|
}
|
|
}
|
|
|
|
export const liveSessionInflightMessages = (inflight?: null | SessionInflightTurn): Msg[] => {
|
|
const user = String(inflight?.user ?? '').trim()
|
|
|
|
return user ? [{ role: 'user', text: user }] : []
|
|
}
|
|
|
|
export const hydrateLiveSessionInflight = (inflight?: null | SessionInflightTurn) => {
|
|
const assistant = String(inflight?.assistant ?? '')
|
|
|
|
if (!assistant && !inflight?.streaming) {
|
|
return
|
|
}
|
|
|
|
turnController.hydrateStreamingText(assistant)
|
|
}
|
|
|
|
const trimTail = (items: Msg[]) => {
|
|
const q = [...items]
|
|
|
|
while (q.at(-1)?.role === 'assistant' || q.at(-1)?.role === 'tool') {
|
|
q.pop()
|
|
}
|
|
|
|
if (q.at(-1)?.role === 'user') {
|
|
q.pop()
|
|
}
|
|
|
|
return q
|
|
}
|
|
|
|
export interface UseSessionLifecycleOptions {
|
|
colsRef: { current: number }
|
|
composerActions: ComposerActions
|
|
gw: GatewayClient
|
|
panel: (title: string, sections: PanelSection[]) => void
|
|
rpc: GatewayRpc
|
|
scrollRef: RefObject<null | ScrollBoxHandle>
|
|
setHistoryItems: StateSetter<Msg[]>
|
|
setLastUserMsg: StateSetter<string>
|
|
setSessionStartedAt: StateSetter<number>
|
|
setStickyPrompt: StateSetter<string>
|
|
setVoiceProcessing: StateSetter<boolean>
|
|
setVoiceRecording: StateSetter<boolean>
|
|
sys: (text: string) => void
|
|
}
|
|
|
|
export function useSessionLifecycle(opts: UseSessionLifecycleOptions) {
|
|
const {
|
|
colsRef,
|
|
composerActions,
|
|
gw,
|
|
panel,
|
|
rpc,
|
|
scrollRef,
|
|
setHistoryItems,
|
|
setLastUserMsg,
|
|
setSessionStartedAt,
|
|
setStickyPrompt,
|
|
setVoiceProcessing,
|
|
setVoiceRecording,
|
|
sys
|
|
} = opts
|
|
|
|
const closeSession = useCallback(
|
|
(targetSid?: null | string) =>
|
|
targetSid ? rpc<SessionCloseResponse>('session.close', { session_id: targetSid }) : Promise.resolve(null),
|
|
[rpc]
|
|
)
|
|
|
|
const resetSession = useCallback(() => {
|
|
turnController.fullReset()
|
|
setVoiceRecording(false)
|
|
setVoiceProcessing(false)
|
|
patchUiState({ bgTasks: new Set(), info: null, sid: null, usage: ZERO })
|
|
setHistoryItems([])
|
|
setLastUserMsg('')
|
|
setStickyPrompt('')
|
|
composerActions.setPasteSnips([])
|
|
// Half-prune: new session has new keys, but keep a warm pool in case
|
|
// the user resumes back to the prior session.
|
|
evictInkCaches('half')
|
|
}, [composerActions, setHistoryItems, setLastUserMsg, setStickyPrompt, setVoiceProcessing, setVoiceRecording])
|
|
|
|
const resetVisibleHistory = useCallback(
|
|
(info: null | SessionInfo = null) => {
|
|
turnController.idle()
|
|
turnController.clearReasoning()
|
|
turnController.turnTools = []
|
|
turnController.persistedToolLabels.clear()
|
|
|
|
setHistoryItems(info ? [introMsg(info)] : [])
|
|
setStickyPrompt('')
|
|
setLastUserMsg('')
|
|
composerActions.setPasteSnips([])
|
|
patchTurnState({ activity: [] })
|
|
patchUiState({ info, usage: usageFrom(info) })
|
|
},
|
|
[composerActions, setHistoryItems, setLastUserMsg, setStickyPrompt]
|
|
)
|
|
|
|
const startNewSession = useCallback(
|
|
async (msg?: string, title?: string, keepCurrent = false) => {
|
|
const setup = await rpc<SetupStatusResponse>('setup.status', {})
|
|
|
|
if (setup?.provider_configured === false) {
|
|
panel(SETUP_REQUIRED_TITLE, buildSetupRequiredSections())
|
|
patchUiState({ status: 'setup required' })
|
|
|
|
return null
|
|
}
|
|
|
|
if (!keepCurrent) {
|
|
await closeSession(getUiState().sid)
|
|
}
|
|
|
|
const r = await rpc<SessionCreateResponse>('session.create', { cols: colsRef.current })
|
|
|
|
if (!r) {
|
|
patchUiState({ status: 'ready' })
|
|
|
|
return null
|
|
}
|
|
|
|
const info = r.info ?? null
|
|
const requestedTitle = title?.trim() ?? ''
|
|
|
|
resetSession()
|
|
setSessionStartedAt(Date.now())
|
|
|
|
writeActiveSessionFile(r.session_id)
|
|
patchUiState({
|
|
info,
|
|
sid: r.session_id,
|
|
status: info?.version ? 'ready' : 'starting agent…',
|
|
usage: usageFrom(info)
|
|
})
|
|
|
|
if (info) {
|
|
setHistoryItems([introMsg(info)])
|
|
}
|
|
|
|
if (info?.credential_warning) {
|
|
sys(`warning: ${info.credential_warning}`)
|
|
}
|
|
|
|
if (info?.config_warning) {
|
|
sys(`warning: ${info.config_warning}`)
|
|
}
|
|
|
|
if (msg) {
|
|
sys(msg)
|
|
}
|
|
|
|
if (requestedTitle) {
|
|
rpc<SessionTitleResponse>('session.title', {
|
|
session_id: r.session_id,
|
|
title: requestedTitle
|
|
})
|
|
.then(result => {
|
|
if (!result || getUiState().sid !== r.session_id) {
|
|
return
|
|
}
|
|
|
|
const nextTitle = (result.title ?? requestedTitle).trim()
|
|
const suffix = result.pending ? ' (queued while session initializes)' : ''
|
|
sys(`session title set: ${nextTitle}${suffix}`)
|
|
})
|
|
.catch((err: unknown) => {
|
|
if (getUiState().sid !== r.session_id) {
|
|
return
|
|
}
|
|
|
|
const message = err instanceof Error ? err.message : String(err)
|
|
sys(`warning: failed to set session title: ${message}`)
|
|
})
|
|
}
|
|
|
|
return r.session_id
|
|
},
|
|
[closeSession, colsRef, panel, resetSession, rpc, setHistoryItems, setSessionStartedAt, sys]
|
|
)
|
|
|
|
const newSession = useCallback(
|
|
(msg?: string, title?: string) => startNewSession(msg, title, false),
|
|
[startNewSession]
|
|
)
|
|
|
|
const newLiveSession = useCallback(
|
|
(msg = 'new live session started', title?: string) => {
|
|
patchOverlayState({ sessions: false })
|
|
|
|
return startNewSession(msg, title, true)
|
|
},
|
|
[startNewSession]
|
|
)
|
|
|
|
const activateLiveSession = useCallback(
|
|
(id: string) => {
|
|
patchOverlayState({ sessions: false })
|
|
patchUiState({ status: 'switching session…' })
|
|
|
|
gw.request<SessionActivateResponse>('session.activate', { session_id: id })
|
|
.then(raw => {
|
|
const r = asRpcResult<SessionActivateResponse>(raw)
|
|
|
|
if (!r) {
|
|
sys('error: invalid response: session.activate')
|
|
|
|
return patchUiState({ status: 'ready' })
|
|
}
|
|
|
|
const info = r.info ?? null
|
|
const running = Boolean(r.running || r.status === 'working' || r.status === 'waiting')
|
|
|
|
resetSession()
|
|
setSessionStartedAt(r.started_at ? r.started_at * 1000 : Date.now())
|
|
const transcript = [...toTranscriptMessages(r.messages), ...liveSessionInflightMessages(r.inflight)]
|
|
setHistoryItems(info ? [introMsg(info), ...transcript] : transcript)
|
|
writeActiveSessionFile(r.session_key ?? r.session_id)
|
|
patchUiState({
|
|
busy: running,
|
|
info,
|
|
sid: r.session_id,
|
|
status: statusFromLiveSession(r.status, running),
|
|
usage: usageFrom(info)
|
|
})
|
|
hydrateLiveSessionInflight(r.inflight)
|
|
setTimeout(() => scrollRef.current?.scrollToBottom(), 0)
|
|
})
|
|
.catch((e: Error) => {
|
|
sys(`error: ${e.message}`)
|
|
patchUiState({ status: 'ready' })
|
|
})
|
|
},
|
|
[gw, resetSession, scrollRef, setHistoryItems, setSessionStartedAt, sys]
|
|
)
|
|
|
|
const resumeById = useCallback(
|
|
(id: string) => {
|
|
patchOverlayState({ picker: false })
|
|
patchUiState({ status: 'resuming…' })
|
|
|
|
rpc<SetupStatusResponse>('setup.status', {}).then(setup => {
|
|
if (setup?.provider_configured === false) {
|
|
panel(SETUP_REQUIRED_TITLE, buildSetupRequiredSections())
|
|
patchUiState({ status: 'setup required' })
|
|
|
|
return
|
|
}
|
|
|
|
closeSession(getUiState().sid === id ? null : getUiState().sid).then(() =>
|
|
gw
|
|
.request<SessionResumeResponse>('session.resume', { cols: colsRef.current, session_id: id })
|
|
.then(raw => {
|
|
const r = asRpcResult<SessionResumeResponse>(raw)
|
|
|
|
if (!r) {
|
|
sys('error: invalid response: session.resume')
|
|
|
|
return patchUiState({ status: 'ready' })
|
|
}
|
|
|
|
resetSession()
|
|
setSessionStartedAt(Date.now())
|
|
|
|
const resumed = toTranscriptMessages(r.messages)
|
|
|
|
setHistoryItems(r.info ? [introMsg(r.info), ...resumed] : resumed)
|
|
writeActiveSessionFile(r.resumed ?? r.session_id)
|
|
patchUiState({
|
|
info: r.info ?? null,
|
|
sid: r.session_id,
|
|
status: 'ready',
|
|
usage: usageFrom(r.info ?? null)
|
|
})
|
|
setTimeout(() => scrollRef.current?.scrollToBottom(), 0)
|
|
})
|
|
.catch((e: Error) => {
|
|
sys(`error: ${e.message}`)
|
|
patchUiState({ status: 'ready' })
|
|
})
|
|
)
|
|
})
|
|
},
|
|
[closeSession, colsRef, gw, panel, resetSession, rpc, scrollRef, setHistoryItems, setSessionStartedAt, sys]
|
|
)
|
|
|
|
const guardBusySessionSwitch = useCallback(
|
|
(what = 'switch sessions') => {
|
|
if (!getUiState().busy) {
|
|
return false
|
|
}
|
|
|
|
sys(`interrupt the current turn before trying to ${what}`)
|
|
|
|
return true
|
|
},
|
|
[sys]
|
|
)
|
|
|
|
return {
|
|
activateLiveSession,
|
|
closeSession,
|
|
guardBusySessionSwitch,
|
|
newLiveSession,
|
|
newSession,
|
|
resetSession,
|
|
resetVisibleHistory,
|
|
resumeById,
|
|
trimLastExchange: trimTail
|
|
}
|
|
}
|