mirror of
https://github.com/openclaw/openclaw.git
synced 2026-05-10 15:45:01 +00:00
feat(typing): add TTL safety-net for stuck indicators (land #27428, thanks @Crpdim)
Co-authored-by: Crpdim <crpdim@users.noreply.github.com>
This commit is contained in:
@@ -3,21 +3,27 @@ import { createTypingKeepaliveLoop } from "./typing-lifecycle.js";
|
||||
export type TypingCallbacks = {
|
||||
onReplyStart: () => Promise<void>;
|
||||
onIdle?: () => void;
|
||||
/** Called when the typing controller is cleaned up (e.g., on NO_REPLY). */
|
||||
/** Called when the typing controller is cleaned up (e.g. on NO_REPLY). */
|
||||
onCleanup?: () => void;
|
||||
};
|
||||
|
||||
export function createTypingCallbacks(params: {
|
||||
export type CreateTypingCallbacksParams = {
|
||||
start: () => Promise<void>;
|
||||
stop?: () => Promise<void>;
|
||||
onStartError: (err: unknown) => void;
|
||||
onStopError?: (err: unknown) => void;
|
||||
keepaliveIntervalMs?: number;
|
||||
}): TypingCallbacks {
|
||||
/** Maximum duration for typing indicator before auto-cleanup (safety TTL). Default: 60s */
|
||||
maxDurationMs?: number;
|
||||
};
|
||||
|
||||
export function createTypingCallbacks(params: CreateTypingCallbacksParams): TypingCallbacks {
|
||||
const stop = params.stop;
|
||||
const keepaliveIntervalMs = params.keepaliveIntervalMs ?? 3_000;
|
||||
const maxDurationMs = params.maxDurationMs ?? 60_000; // Default 60s TTL
|
||||
let stopSent = false;
|
||||
let closed = false;
|
||||
let ttlTimer: ReturnType<typeof setTimeout> | undefined;
|
||||
|
||||
const fireStart = async () => {
|
||||
if (closed) {
|
||||
@@ -35,19 +41,43 @@ export function createTypingCallbacks(params: {
|
||||
onTick: fireStart,
|
||||
});
|
||||
|
||||
// TTL safety: auto-stop typing after maxDurationMs
|
||||
const startTtlTimer = () => {
|
||||
if (maxDurationMs <= 0) {
|
||||
return;
|
||||
}
|
||||
clearTtlTimer();
|
||||
ttlTimer = setTimeout(() => {
|
||||
if (!closed) {
|
||||
console.warn(`[typing] TTL exceeded (${maxDurationMs}ms), auto-stopping typing indicator`);
|
||||
fireStop();
|
||||
}
|
||||
}, maxDurationMs);
|
||||
};
|
||||
|
||||
const clearTtlTimer = () => {
|
||||
if (ttlTimer) {
|
||||
clearTimeout(ttlTimer);
|
||||
ttlTimer = undefined;
|
||||
}
|
||||
};
|
||||
|
||||
const onReplyStart = async () => {
|
||||
if (closed) {
|
||||
return;
|
||||
}
|
||||
stopSent = false;
|
||||
keepaliveLoop.stop();
|
||||
clearTtlTimer();
|
||||
await fireStart();
|
||||
keepaliveLoop.start();
|
||||
startTtlTimer(); // Start TTL safety timer
|
||||
};
|
||||
|
||||
const fireStop = () => {
|
||||
closed = true;
|
||||
keepaliveLoop.stop();
|
||||
clearTtlTimer(); // Clear TTL timer on normal stop
|
||||
if (!stop || stopSent) {
|
||||
return;
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user