mirror of
https://github.com/NousResearch/hermes-agent.git
synced 2026-04-25 00:51:20 +00:00
Additional TUI fixes discovered in the same audit: 1. /plan slash command was silently lost — process_command() queues the plan skill invocation onto _pending_input which nobody reads in the slash worker subprocess. Now intercepted in slash.exec and routed through command.dispatch with a new 'send' dispatch type. Same interception added for /retry, /queue, /steer as safety nets (these already have correct TUI-local handlers in core.ts, but the server-side guard prevents regressions if the local handler is bypassed). 2. Tool results were stripping ANSI escape codes — the messageLine component used stripAnsi() + plain <Text> for tool role messages, losing all color/styling from terminal, search_files, etc. Now uses <Ansi> component (already imported) when ANSI is detected. 3. Terminal tab title now shows model + busy status via useTerminalTitle hook from @hermes/ink (was never used). Users can identify Hermes tabs and see at a glance whether the agent is busy or ready. 4. Added 'send' variant to CommandDispatchResponse type + asCommandDispatch parser + createSlashHandler handler for commands that need to inject a message into the conversation (plan, queue fallback, steer fallback).
37 lines
1.2 KiB
TypeScript
37 lines
1.2 KiB
TypeScript
import type { CommandDispatchResponse } from '../gatewayTypes.js'
|
|
|
|
export type RpcResult = Record<string, any>
|
|
|
|
export const asRpcResult = <T extends RpcResult = RpcResult>(value: unknown): T | null =>
|
|
!value || typeof value !== 'object' || Array.isArray(value) ? null : (value as T)
|
|
|
|
export const asCommandDispatch = (value: unknown): CommandDispatchResponse | null => {
|
|
const o = asRpcResult(value)
|
|
|
|
if (!o || typeof o.type !== 'string') {
|
|
return null
|
|
}
|
|
|
|
const t = o.type
|
|
|
|
if (t === 'exec' || t === 'plugin') {
|
|
return { type: t, output: typeof o.output === 'string' ? o.output : undefined }
|
|
}
|
|
|
|
if (t === 'alias' && typeof o.target === 'string') {
|
|
return { type: 'alias', target: o.target }
|
|
}
|
|
|
|
if (t === 'skill' && typeof o.name === 'string') {
|
|
return { type: 'skill', name: o.name, message: typeof o.message === 'string' ? o.message : undefined }
|
|
}
|
|
|
|
if (t === 'send' && typeof o.message === 'string') {
|
|
return { type: 'send', message: o.message }
|
|
}
|
|
|
|
return null
|
|
}
|
|
|
|
export const rpcErrorMessage = (err: unknown) =>
|
|
err instanceof Error && err.message ? err.message : typeof err === 'string' && err.trim() ? err : 'request failed'
|