function getTimerApi(runtime) { const rawSetTimeout = typeof runtime?.setTimeout === "function" ? runtime.setTimeout : globalThis.setTimeout; const rawClearTimeout = typeof runtime?.clearTimeout === "function" ? runtime.clearTimeout : globalThis.clearTimeout; return { setTimeout(...args) { return Reflect.apply(rawSetTimeout, globalThis, args); }, clearTimeout(...args) { return Reflect.apply(rawClearTimeout, globalThis, args); }, }; } export function registerBeforeCombinePromptsController(runtime, listener) { const makeFirst = runtime.getEventMakeFirst(); if (typeof makeFirst === "function") { return makeFirst( runtime.eventTypes.GENERATE_BEFORE_COMBINE_PROMPTS, listener, ); } runtime.console.warn("[ST-BME] eventMakeFirst 不可用,回退到普通事件注册"); runtime.eventSource.on( runtime.eventTypes.GENERATE_BEFORE_COMBINE_PROMPTS, listener, ); return null; } export function registerGenerationAfterCommandsController(runtime, listener) { const makeFirst = runtime.getEventMakeFirst(); const eventName = runtime.eventTypes.GENERATION_AFTER_COMMANDS; console.warn("[ST-BME:DIAG] Registering GENERATION_AFTER_COMMANDS:", { eventName, hasMakeFirst: typeof makeFirst === "function", hasListener: typeof listener === "function", }); if (typeof makeFirst === "function") { const cleanup = makeFirst(eventName, listener); console.warn("[ST-BME:DIAG] Registered via makeFirst, cleanup:", typeof cleanup); return cleanup; } runtime.console.warn( "[ST-BME] eventMakeFirst 不可用,GENERATION_AFTER_COMMANDS 回退到普通事件注册", ); runtime.eventSource.on(eventName, listener); return null; } export function scheduleSendIntentHookRetryController(runtime, delayMs = 400) { const timers = getTimerApi(runtime); timers.clearTimeout(runtime.getSendIntentHookRetryTimer()); const timer = timers.setTimeout(() => { runtime.setSendIntentHookRetryTimer(null); runtime.installSendIntentHooks(); }, delayMs); runtime.setSendIntentHookRetryTimer(timer); } export function installSendIntentHooksController(runtime) { for (const cleanup of runtime.consumeSendIntentHookCleanup()) { try { cleanup(); } catch (error) { runtime.console.warn("[ST-BME] 清理发送意图钩子失败:", error); } } const sendButton = runtime.document.getElementById("send_but"); const sendTextarea = runtime.document.getElementById("send_textarea"); if (sendButton) { const captureSendIntent = () => { runtime.recordRecallSendIntent( runtime.getSendTextareaValue(), "send-button", ); }; sendButton.addEventListener("click", captureSendIntent, true); sendButton.addEventListener("pointerup", captureSendIntent, true); sendButton.addEventListener("touchend", captureSendIntent, true); runtime.pushSendIntentHookCleanup(() => { sendButton.removeEventListener("click", captureSendIntent, true); sendButton.removeEventListener("pointerup", captureSendIntent, true); sendButton.removeEventListener("touchend", captureSendIntent, true); }); } if (sendTextarea) { const captureEnterIntent = (event) => { if ( (event.key === "Enter" || event.key === "NumpadEnter") && !event.shiftKey ) { runtime.recordRecallSendIntent( runtime.getSendTextareaValue(), "textarea-enter", ); } }; sendTextarea.addEventListener("keydown", captureEnterIntent, true); runtime.pushSendIntentHookCleanup(() => { sendTextarea.removeEventListener("keydown", captureEnterIntent, true); }); } if (!sendButton || !sendTextarea) { runtime.scheduleSendIntentHookRetry(); } } export function registerCoreEventHooksController(runtime) { const { eventSource, eventTypes, handlers } = runtime; const registrationState = runtime.getCoreEventBindingState?.() || {}; if (registrationState.registered) { runtime.console?.warn?.("[ST-BME] 核心事件已注册,跳过重复绑定"); return registrationState; } const cleanups = []; const bind = (eventName, listener) => { if (!eventName || typeof listener !== "function") return; eventSource.on(eventName, listener); if (typeof eventSource.off === "function") { cleanups.push(() => eventSource.off(eventName, listener)); } else if (typeof eventSource.removeListener === "function") { cleanups.push(() => eventSource.removeListener(eventName, listener)); } }; bind(eventTypes.CHAT_CHANGED, handlers.onChatChanged); if (eventTypes.CHAT_LOADED) { bind(eventTypes.CHAT_LOADED, handlers.onChatLoaded); } if (eventTypes.MESSAGE_SENT) { bind(eventTypes.MESSAGE_SENT, handlers.onMessageSent); } if (eventTypes.GENERATION_STARTED) { bind(eventTypes.GENERATION_STARTED, handlers.onGenerationStarted); } if (eventTypes.GENERATION_ENDED) { bind(eventTypes.GENERATION_ENDED, handlers.onGenerationEnded); } const beforeCombineCleanup = runtime.registerBeforeCombinePrompts( handlers.onBeforeCombinePrompts, ); if (typeof beforeCombineCleanup === "function") { cleanups.push(beforeCombineCleanup); } const afterCommandsCleanup = runtime.registerGenerationAfterCommands( handlers.onGenerationAfterCommands, ); if (typeof afterCommandsCleanup === "function") { cleanups.push(afterCommandsCleanup); } bind(eventTypes.MESSAGE_RECEIVED, handlers.onMessageReceived); bind(eventTypes.MESSAGE_DELETED, handlers.onMessageDeleted); bind(eventTypes.MESSAGE_EDITED, handlers.onMessageEdited); bind(eventTypes.MESSAGE_SWIPED, handlers.onMessageSwiped); if (eventTypes.MESSAGE_UPDATED) { bind(eventTypes.MESSAGE_UPDATED, handlers.onMessageEdited); } const nextState = { registered: true, cleanups, registeredAt: Date.now(), }; runtime.setCoreEventBindingState?.(nextState); return nextState; } export function onChatChangedController(runtime) { const timers = getTimerApi(runtime); runtime.clearPendingHistoryMutationChecks(); timers.clearTimeout(runtime.getPendingHistoryRecoveryTimer()); runtime.setPendingHistoryRecoveryTimer(null); runtime.setPendingHistoryRecoveryTrigger(""); runtime.clearPendingAutoExtraction?.(); runtime.clearPendingGraphLoadRetry(); runtime.setSkipBeforeCombineRecallUntil(0); runtime.setLastPreGenerationRecallKey(""); runtime.setLastPreGenerationRecallAt(0); runtime.clearGenerationRecallTransactionsForChat("", { clearAll: true }); runtime.abortAllRunningStages(); runtime.dismissAllStageNotices(); runtime.syncGraphLoadFromLiveContext({ source: "chat-changed", force: true, }); runtime.clearInjectionState(); runtime.clearRecallInputTracking(); runtime.installSendIntentHooks(); runtime.refreshPersistedRecallMessageUi?.(); } export function onChatLoadedController(runtime) { runtime.syncGraphLoadFromLiveContext({ source: "chat-loaded", }); runtime.refreshPersistedRecallMessageUi?.(); } export function onMessageSentController(runtime, messageId) { const context = runtime.getContext(); const chat = context?.chat; const message = Array.isArray(chat) && Number.isFinite(messageId) ? chat[messageId] : null; if (!message?.is_user) return; runtime.recordRecallSentUserMessage(messageId, message.mes || ""); runtime.refreshPersistedRecallMessageUi?.(); } export function onGenerationStartedController( runtime, type, params = {}, dryRun = false, ) { if (dryRun) { runtime.markDryRunPromptPreview?.(); return null; } runtime.clearDryRunPromptPreview?.(); if (params?.automatic_trigger || params?.quiet_prompt) return null; const generationType = String(type || "normal").trim() || "normal"; if (generationType !== "normal") return null; const pendingSendIntent = runtime.getPendingRecallSendIntent?.(); const pendingIntentText = runtime.isFreshRecallInputRecord?.( pendingSendIntent, ) ? pendingSendIntent.text : ""; const textareaText = typeof runtime.getSendTextareaValue === "function" ? runtime.getSendTextareaValue() : ""; const snapshotText = runtime.normalizeRecallInputText?.(pendingIntentText || textareaText) || ""; if (!snapshotText) return null; return runtime.freezeHostGenerationInputSnapshot( snapshotText, pendingIntentText ? "generation-started-send-intent" : "generation-started-textarea", ); } export function onMessageDeletedController( runtime, chatLengthOrMessageId, meta = null, ) { runtime.invalidateRecallAfterHistoryMutation("消息已删除"); runtime.scheduleHistoryMutationRecheck( "message-deleted", chatLengthOrMessageId, meta, ); runtime.refreshPersistedRecallMessageUi?.(); } export function onMessageEditedController(runtime, messageId, meta = null) { runtime.invalidateRecallAfterHistoryMutation("消息已编辑"); runtime.scheduleHistoryMutationRecheck("message-edited", messageId, meta); runtime.refreshPersistedRecallMessageUi?.(); } export async function onMessageSwipedController(runtime, messageId, meta = null) { runtime.invalidateRecallAfterHistoryMutation("已切换楼层 swipe"); const parsedFloor = Number(messageId); const fromFloor = Number.isFinite(parsedFloor) ? parsedFloor : undefined; let result = { success: false, rollbackPerformed: false, extractionTriggered: false, requestedFloor: fromFloor ?? null, effectiveFromFloor: null, recoveryPath: "reroll-handler-unavailable", affectedBatchCount: 0, error: "swipe reroll handler unavailable", }; if (typeof runtime.onReroll === "function") { try { result = await runtime.onReroll({ fromFloor, meta }); } catch (error) { runtime.console?.error?.("[ST-BME] swipe reroll failed:", error); result = { success: false, rollbackPerformed: false, extractionTriggered: false, requestedFloor: fromFloor ?? null, effectiveFromFloor: null, recoveryPath: "reroll-threw", affectedBatchCount: 0, error: error?.message || String(error) || "swipe reroll failed", }; } } else { runtime.console?.warn?.( "[ST-BME] MESSAGE_SWIPED missing onReroll; skip generic history recovery fallback.", { messageId, meta }, ); } runtime.refreshPersistedRecallMessageUi?.(); return result; } export async function onGenerationAfterCommandsController( runtime, type, params = {}, dryRun = false, ) { console.warn("[ST-BME:DIAG] GENERATION_AFTER_COMMANDS fired", { type, dryRun, paramsKeys: Object.keys(params || {}) }); if (dryRun) { console.warn("[ST-BME:DIAG] EXIT: dryRun=true"); return; } const generationType = String(type || "normal").trim() || "normal"; const frozenInputSnapshot = generationType === "normal" ? runtime.consumeHostGenerationInputSnapshot?.({ preserve: true }) || runtime.consumeHostGenerationInputSnapshot?.() : null; console.warn("[ST-BME:DIAG] frozenInputSnapshot:", frozenInputSnapshot?.text ? `"${frozenInputSnapshot.text.slice(0,50)}"` : "(empty)", "fresh:", !!frozenInputSnapshot?.at); const context = runtime.getContext(); const chat = context?.chat; console.warn("[ST-BME:DIAG] chat length:", chat?.length, "last msg:", chat?.length ? { is_user: chat[chat.length-1]?.is_user, mes: (chat[chat.length-1]?.mes||"").slice(0,50) } : "(no chat)"); const recallOptions = runtime.buildGenerationAfterCommandsRecallInput( type, { ...params, frozenInputSnapshot, }, chat, ); if (!recallOptions) { console.warn("[ST-BME:DIAG] EXIT: buildGenerationAfterCommandsRecallInput returned null"); return; } console.warn("[ST-BME:DIAG] recallOptions:", { generationType: recallOptions.generationType, overrideUserMessage: recallOptions.overrideUserMessage?.slice(0,50), overrideSource: recallOptions.overrideSource, targetIdx: recallOptions.targetUserMessageIndex }); const recallContext = runtime.createGenerationRecallContext({ hookName: "GENERATION_AFTER_COMMANDS", generationType, recallOptions, }); if (!recallContext.shouldRun && !recallContext.transaction) { console.warn("[ST-BME:DIAG] EXIT: shouldRun=false, no transaction. guardReason:", recallContext.guardReason); return; } console.warn("[ST-BME:DIAG] recallContext:", { shouldRun: recallContext.shouldRun, guardReason: recallContext.guardReason, transactionId: recallContext.transaction?.id }); const runtimeRecallOptions = recallContext.recallOptions || recallOptions || {}; const deliveryMode = runtime.resolveGenerationRecallDeliveryMode?.( recallContext.hookName, recallContext.generationType, runtimeRecallOptions, ) || "immediate"; let recallResult = runtime.getGenerationRecallTransactionResult?.( recallContext.transaction, ); console.warn("[ST-BME:DIAG] deliveryMode:", deliveryMode, "shouldRun:", recallContext.shouldRun); if (recallContext.shouldRun) { runtime.markGenerationRecallTransactionHookState( recallContext.transaction, recallContext.hookName, "running", ); if (deliveryMode === "deferred") { runtime.clearLiveRecallInjectionPromptForRewrite?.(); } console.warn("[ST-BME:DIAG] >>> Starting runRecall..."); recallResult = await runtime.runRecall({ ...runtimeRecallOptions, deliveryMode, recallKey: recallContext.recallKey, hookName: recallContext.hookName, signal: params?.signal, }); console.warn("[ST-BME:DIAG] <<< runRecall finished:", { status: recallResult?.status, ok: recallResult?.ok, reason: recallResult?.reason, injectionText: recallResult?.injectionText?.slice(0,80) }); runtime.storeGenerationRecallTransactionResult?.( recallContext.transaction, recallResult, { hookName: recallContext.hookName, deliveryMode, }, ); runtime.markGenerationRecallTransactionHookState( recallContext.transaction, recallContext.hookName, runtime.getGenerationRecallHookStateFromResult(recallResult), ); } // immediate 模式下,runRecall → applyRecallInjection 内部已通过 // setExtensionPrompt 完成了注入,此处直接返回召回结果。 // 后续 GENERATE_BEFORE_COMBINE_PROMPTS 阶段会通过 // applyFinalRecallInjectionForGeneration 做 deferred rewrite 兜底。 if (deliveryMode === "immediate") { console.warn("[ST-BME:DIAG] DONE: immediate mode, injection via setExtensionPrompt in runRecall"); return recallResult; } return runtime.applyFinalRecallInjectionForGeneration({ generationType: recallContext.generationType, freshRecallResult: recallResult, transaction: recallContext.transaction, hookName: recallContext.hookName, }); } export async function onBeforeCombinePromptsController( runtime, promptData = null, ) { if (runtime.consumeDryRunPromptPreview?.()) { return { skipped: true, reason: "dry-run-preview", }; } const frozenInputSnapshot = runtime.consumeHostGenerationInputSnapshot?.() || runtime.getPendingHostGenerationInputSnapshot?.() || runtime.createRecallInputRecord?.() || {}; const context = runtime.getContext(); const chat = context?.chat; const recallOptions = runtime.buildNormalGenerationRecallInput(chat, { frozenInputSnapshot, }) || runtime.buildHistoryGenerationRecallInput(chat) || {}; const recallContext = runtime.createGenerationRecallContext({ hookName: "GENERATE_BEFORE_COMBINE_PROMPTS", generationType: "normal", recallOptions, }); if (!recallContext.shouldRun && !recallContext.transaction) { return; } const runtimeRecallOptions = recallContext.recallOptions || recallOptions || {}; const deliveryMode = runtime.resolveGenerationRecallDeliveryMode?.( recallContext.hookName, recallContext.generationType, runtimeRecallOptions, ) || "deferred"; let recallResult = runtime.getGenerationRecallTransactionResult?.( recallContext.transaction, ); if (recallContext.shouldRun) { runtime.markGenerationRecallTransactionHookState( recallContext.transaction, recallContext.hookName, "running", ); if (deliveryMode === "deferred") { runtime.clearLiveRecallInjectionPromptForRewrite?.(); } recallResult = await runtime.runRecall({ ...runtimeRecallOptions, deliveryMode, recallKey: recallContext.recallKey, hookName: recallContext.hookName, }); runtime.storeGenerationRecallTransactionResult?.( recallContext.transaction, recallResult, { hookName: recallContext.hookName, deliveryMode, }, ); runtime.markGenerationRecallTransactionHookState( recallContext.transaction, recallContext.hookName, runtime.getGenerationRecallHookStateFromResult(recallResult), ); } return runtime.applyFinalRecallInjectionForGeneration({ generationType: recallContext.generationType, freshRecallResult: recallResult, transaction: recallContext.transaction, promptData, hookName: recallContext.hookName, }); } export function onMessageReceivedController( runtime, messageId = null, _type = "", ) { const enqueueMicrotask = typeof globalThis.queueMicrotask === "function" ? globalThis.queueMicrotask.bind(globalThis) : typeof runtime.queueMicrotask === "function" ? (task) => Reflect.apply(runtime.queueMicrotask, globalThis, [task]) : (task) => Promise.resolve().then(task); const persistenceState = runtime.getGraphPersistenceState?.() || {}; const loadState = persistenceState.loadState || ""; const dbReady = persistenceState.dbReady ?? (loadState === "loaded" || loadState === "empty-confirmed"); if ( !dbReady || loadState === "loading" || loadState === "shadow-restored" || loadState === "blocked" ) { runtime.syncGraphLoadFromLiveContext?.({ source: "message-received-reconcile", }); } if (runtime.getCurrentGraph()) { if ( runtime.getGraphPersistenceState()?.pendingPersist && runtime.isGraphMetadataWriteAllowed() ) { runtime.maybeFlushQueuedGraphPersist("message-received-pending-flush"); } } const pendingRecallSendIntent = runtime.getPendingRecallSendIntent(); if ( pendingRecallSendIntent?.text && !runtime.isFreshRecallInputRecord(pendingRecallSendIntent) ) { runtime.setPendingRecallSendIntent(runtime.createRecallInputRecord()); } const context = runtime.getContext(); const chat = context?.chat; const receivedMessage = Array.isArray(chat) && Number.isFinite(Number(messageId)) ? chat[Number(messageId)] : null; const lastMessage = Array.isArray(chat) && chat.length > 0 ? chat[chat.length - 1] : null; const targetMessage = runtime.isAssistantChatMessage(receivedMessage) ? receivedMessage : lastMessage; if (runtime.isAssistantChatMessage(targetMessage)) { runtime.console?.debug?.( "[ST-BME] assistant message received, queueing auto extraction", { messageId: Number.isFinite(Number(messageId)) ? Number(messageId) : null, chatLength: Array.isArray(chat) ? chat.length : 0, loadState, dbReady, }, ); enqueueMicrotask(() => { void runtime.runExtraction().catch((error) => { runtime.console.error("[ST-BME] 异步自动提取失败:", error); runtime.notifyExtractionIssue( error?.message || String(error) || "自动提取失败", ); }); }); } runtime.refreshPersistedRecallMessageUi?.(); }