mirror of
https://github.com/NousResearch/hermes-agent.git
synced 2026-05-07 02:51:50 +00:00
Today there's no way to remove a queued message — ↑ loads it for edit, ctrl-K dispatches the head, but a draft you no longer want stays put forever. ctrl-C just clears the composer and exits edit mode without touching the queue. Two new bindings, both gated on queueEditIdx !== null so they're inert when the user isn't pointing at a queue item: - ctrl-X — delete the queue item being edited, clear composer, exit edit mode. "cut" matches the mental model and doesn't collide with any existing binding. - esc — cancel the edit (composer clears, item stays in queue). Mirrors ctrl-C's existing behavior so muscle memory has two paths. Header line now reads `queued (3) · editing 2 · ⌃X delete · esc cancel` when in edit mode, so the affordance is discoverable without /help. The /help hotkey table also gets a Ctrl+X entry. ctrl-C is intentionally unchanged: it should never destroy queued content. Cancel is non-destructive (esc / ctrl-C); only ctrl-X removes the item.
364 lines
9.8 KiB
TypeScript
364 lines
9.8 KiB
TypeScript
import { spawnSync } from 'node:child_process'
|
|
import { mkdtempSync, readFileSync, rmSync, writeFileSync } from 'node:fs'
|
|
import { tmpdir } from 'node:os'
|
|
import { join } from 'node:path'
|
|
|
|
import { useStdin, withInkSuspended } from '@hermes/ink'
|
|
import { useStore } from '@nanostores/react'
|
|
import { useCallback, useMemo, useState } from 'react'
|
|
|
|
import type { PasteEvent } from '../components/textInput.js'
|
|
import { LARGE_PASTE } from '../config/limits.js'
|
|
import type { ImageAttachResponse, InputDetectDropResponse } from '../gatewayTypes.js'
|
|
import { useCompletion } from '../hooks/useCompletion.js'
|
|
import { useInputHistory } from '../hooks/useInputHistory.js'
|
|
import { useQueue } from '../hooks/useQueue.js'
|
|
import { isUsableClipboardText, readClipboardText } from '../lib/clipboard.js'
|
|
import { resolveEditor } from '../lib/editor.js'
|
|
import { readOsc52Clipboard } from '../lib/osc52.js'
|
|
import { isRemoteShellSession } from '../lib/terminalSetup.js'
|
|
import { pasteTokenLabel, stripTrailingPasteNewlines } from '../lib/text.js'
|
|
|
|
import type { MaybePromise, PasteSnippet, UseComposerStateOptions, UseComposerStateResult } from './interfaces.js'
|
|
import { $isBlocked } from './overlayStore.js'
|
|
import { getUiState } from './uiStore.js'
|
|
|
|
const PASTE_SNIP_MAX_COUNT = 32
|
|
const PASTE_SNIP_MAX_TOTAL_BYTES = 4 * 1024 * 1024
|
|
|
|
const trimSnips = (snips: PasteSnippet[]): PasteSnippet[] => {
|
|
let total = 0
|
|
const out: PasteSnippet[] = []
|
|
|
|
for (let i = snips.length - 1; i >= 0; i--) {
|
|
const snip = snips[i]!
|
|
const size = snip.text.length
|
|
|
|
if (out.length >= PASTE_SNIP_MAX_COUNT || total + size > PASTE_SNIP_MAX_TOTAL_BYTES) {
|
|
break
|
|
}
|
|
|
|
total += size
|
|
out.unshift(snip)
|
|
}
|
|
|
|
return out.length === snips.length ? snips : out
|
|
}
|
|
|
|
/** Insert text at the cursor position, adding spacing to separate from adjacent non-whitespace. */
|
|
function insertAtCursor(value: string, cursor: number, text: string): { cursor: number; value: string } {
|
|
const lead = cursor > 0 && !/\s/.test(value[cursor - 1] ?? '') ? ' ' : ''
|
|
const tail = cursor < value.length && !/\s/.test(value[cursor] ?? '') ? ' ' : ''
|
|
const insert = `${lead}${text}${tail}`
|
|
|
|
return {
|
|
cursor: cursor + insert.length,
|
|
value: value.slice(0, cursor) + insert + value.slice(cursor)
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Quick client-side heuristic to detect text that looks like a dropped file path.
|
|
* When this returns true the composer sends RPC calls to the server for actual
|
|
* validation. Keep in sync with _detect_file_drop() in cli.py — see that
|
|
* function for the canonical prefix list.
|
|
*/
|
|
export function looksLikeDroppedPath(text: string): boolean {
|
|
const trimmed = text.trim()
|
|
|
|
if (!trimmed || trimmed.includes('\n')) {
|
|
return false
|
|
}
|
|
|
|
// file:// URIs, relative, home-relative, quoted, and Windows drive paths
|
|
if (
|
|
trimmed.startsWith('file://') ||
|
|
trimmed.startsWith('~/') ||
|
|
trimmed.startsWith('./') ||
|
|
trimmed.startsWith('../') ||
|
|
trimmed.startsWith('"/') ||
|
|
trimmed.startsWith("'/") ||
|
|
trimmed.startsWith('"~') ||
|
|
trimmed.startsWith("'~") ||
|
|
/^[A-Za-z]:[/\\]/.test(trimmed) ||
|
|
/^["'][A-Za-z]:[/\\]/.test(trimmed)
|
|
) {
|
|
return true
|
|
}
|
|
|
|
// Bare absolute paths (start with /) — require a second '/' or a '.' to avoid
|
|
// false positives on short strings like "/api" or "/help" which would trigger
|
|
// unnecessary RPC round-trips.
|
|
if (trimmed.startsWith('/')) {
|
|
const rest = trimmed.slice(1)
|
|
|
|
return rest.includes('/') || rest.includes('.')
|
|
}
|
|
|
|
return false
|
|
}
|
|
|
|
export function useComposerState({
|
|
gw,
|
|
onClipboardPaste,
|
|
onImageAttached,
|
|
submitRef
|
|
}: UseComposerStateOptions): UseComposerStateResult {
|
|
const [input, setInput] = useState('')
|
|
const [inputBuf, setInputBuf] = useState<string[]>([])
|
|
const [pasteSnips, setPasteSnips] = useState<PasteSnippet[]>([])
|
|
const isBlocked = useStore($isBlocked)
|
|
const { querier } = useStdin() as { querier: Parameters<typeof readOsc52Clipboard>[0] }
|
|
|
|
const {
|
|
queueRef,
|
|
queueEditRef,
|
|
queuedDisplay,
|
|
queueEditIdx,
|
|
enqueue,
|
|
dequeue,
|
|
removeQ,
|
|
replaceQ,
|
|
setQueueEdit,
|
|
syncQueue
|
|
} = useQueue()
|
|
|
|
const { historyRef, historyIdx, setHistoryIdx, historyDraftRef, pushHistory } = useInputHistory()
|
|
const { completions, compIdx, setCompIdx, compReplace } = useCompletion(input, isBlocked, gw)
|
|
|
|
const clearIn = useCallback(() => {
|
|
setInput('')
|
|
setInputBuf([])
|
|
setPasteSnips([])
|
|
setQueueEdit(null)
|
|
setHistoryIdx(null)
|
|
historyDraftRef.current = ''
|
|
}, [historyDraftRef, setQueueEdit, setHistoryIdx])
|
|
|
|
const handleResolvedPaste = useCallback(
|
|
async ({
|
|
bracketed,
|
|
cursor,
|
|
text,
|
|
value
|
|
}: Omit<PasteEvent, 'hotkey'>): Promise<null | { cursor: number; value: string }> => {
|
|
const cleanedText = stripTrailingPasteNewlines(text)
|
|
|
|
if (!cleanedText || !/[^\n]/.test(cleanedText)) {
|
|
if (bracketed) {
|
|
void onClipboardPaste(true)
|
|
}
|
|
|
|
return null
|
|
}
|
|
|
|
const sid = getUiState().sid
|
|
|
|
if (sid && looksLikeDroppedPath(cleanedText)) {
|
|
try {
|
|
const attached = await gw.request<ImageAttachResponse>('image.attach', {
|
|
path: cleanedText,
|
|
session_id: sid
|
|
})
|
|
|
|
if (attached?.name) {
|
|
onImageAttached?.(attached)
|
|
const remainder = attached.remainder?.trim() ?? ''
|
|
|
|
if (!remainder) {
|
|
return { cursor, value }
|
|
}
|
|
|
|
return insertAtCursor(value, cursor, remainder)
|
|
}
|
|
} catch {
|
|
// Fall back to generic file-drop detection below.
|
|
}
|
|
|
|
try {
|
|
const dropped = await gw.request<InputDetectDropResponse>('input.detect_drop', {
|
|
session_id: sid,
|
|
text: cleanedText
|
|
})
|
|
|
|
if (dropped?.matched && dropped.text) {
|
|
return insertAtCursor(value, cursor, dropped.text)
|
|
}
|
|
} catch {
|
|
// Fall through to normal text paste behavior.
|
|
}
|
|
}
|
|
|
|
const lineCount = cleanedText.split('\n').length
|
|
|
|
if (cleanedText.length < LARGE_PASTE.chars && lineCount < LARGE_PASTE.lines) {
|
|
return {
|
|
cursor: cursor + cleanedText.length,
|
|
value: value.slice(0, cursor) + cleanedText + value.slice(cursor)
|
|
}
|
|
}
|
|
|
|
const label = pasteTokenLabel(cleanedText, lineCount)
|
|
const inserted = insertAtCursor(value, cursor, label)
|
|
|
|
setPasteSnips(prev => trimSnips([...prev, { label, text: cleanedText }]))
|
|
|
|
void gw
|
|
.request<{ path?: string }>('paste.collapse', { text: cleanedText })
|
|
.then(r => {
|
|
const path = r?.path
|
|
|
|
if (!path) {
|
|
return
|
|
}
|
|
|
|
setPasteSnips(prev => prev.map(s => (s.label === label ? { ...s, path } : s)))
|
|
})
|
|
.catch(() => {})
|
|
|
|
return inserted
|
|
},
|
|
[gw, onClipboardPaste, onImageAttached]
|
|
)
|
|
|
|
const handleTextPaste = useCallback(
|
|
({
|
|
bracketed,
|
|
cursor,
|
|
hotkey,
|
|
text,
|
|
value
|
|
}: PasteEvent): MaybePromise<null | { cursor: number; value: string }> => {
|
|
if (hotkey) {
|
|
const preferOsc52 = isRemoteShellSession(process.env)
|
|
|
|
const readPreferredText = preferOsc52
|
|
? readOsc52Clipboard(querier).then(async osc52Text => {
|
|
if (isUsableClipboardText(osc52Text)) {
|
|
return osc52Text
|
|
}
|
|
|
|
return readClipboardText()
|
|
})
|
|
: readClipboardText().then(async clipText => {
|
|
if (isUsableClipboardText(clipText)) {
|
|
return clipText
|
|
}
|
|
|
|
return readOsc52Clipboard(querier)
|
|
})
|
|
|
|
return readPreferredText.then(async preferredText => {
|
|
if (isUsableClipboardText(preferredText)) {
|
|
return handleResolvedPaste({ bracketed: false, cursor, text: preferredText, value })
|
|
}
|
|
|
|
void onClipboardPaste(false)
|
|
|
|
return null
|
|
})
|
|
}
|
|
|
|
return handleResolvedPaste({ bracketed: !!bracketed, cursor, text, value })
|
|
},
|
|
[handleResolvedPaste, onClipboardPaste, querier]
|
|
)
|
|
|
|
const openEditor = useCallback(async () => {
|
|
const dir = mkdtempSync(join(tmpdir(), 'hermes-'))
|
|
const file = join(dir, 'prompt.md')
|
|
const [cmd, ...args] = resolveEditor()
|
|
|
|
writeFileSync(file, [...inputBuf, input].join('\n'))
|
|
|
|
let exitCode: null | number = null
|
|
|
|
await withInkSuspended(async () => {
|
|
exitCode = spawnSync(cmd!, [...args, file], { stdio: 'inherit' }).status
|
|
})
|
|
|
|
try {
|
|
if (exitCode !== 0) {
|
|
return
|
|
}
|
|
|
|
const text = readFileSync(file, 'utf8').trimEnd()
|
|
|
|
if (!text) {
|
|
return
|
|
}
|
|
|
|
setInput('')
|
|
setInputBuf([])
|
|
submitRef.current(text)
|
|
} finally {
|
|
rmSync(dir, { force: true, recursive: true })
|
|
}
|
|
}, [input, inputBuf, submitRef])
|
|
|
|
const actions = useMemo(
|
|
() => ({
|
|
clearIn,
|
|
dequeue,
|
|
enqueue,
|
|
handleTextPaste,
|
|
openEditor,
|
|
pushHistory,
|
|
removeQueue: removeQ,
|
|
replaceQueue: replaceQ,
|
|
setCompIdx,
|
|
setHistoryIdx,
|
|
setInput,
|
|
setInputBuf,
|
|
setPasteSnips,
|
|
setQueueEdit,
|
|
syncQueue
|
|
}),
|
|
[
|
|
clearIn,
|
|
dequeue,
|
|
enqueue,
|
|
handleTextPaste,
|
|
openEditor,
|
|
pushHistory,
|
|
removeQ,
|
|
replaceQ,
|
|
setCompIdx,
|
|
setHistoryIdx,
|
|
setQueueEdit,
|
|
syncQueue
|
|
]
|
|
)
|
|
|
|
const refs = useMemo(
|
|
() => ({
|
|
historyDraftRef,
|
|
historyRef,
|
|
queueEditRef,
|
|
queueRef,
|
|
submitRef
|
|
}),
|
|
[historyDraftRef, historyRef, queueEditRef, queueRef, submitRef]
|
|
)
|
|
|
|
const state = useMemo(
|
|
() => ({
|
|
compIdx,
|
|
compReplace,
|
|
completions,
|
|
historyIdx,
|
|
input,
|
|
inputBuf,
|
|
pasteSnips,
|
|
queueEditIdx,
|
|
queuedDisplay
|
|
}),
|
|
[compIdx, compReplace, completions, historyIdx, input, inputBuf, pasteSnips, queueEditIdx, queuedDisplay]
|
|
)
|
|
|
|
return {
|
|
actions,
|
|
refs,
|
|
state
|
|
}
|
|
}
|