mirror of
https://github.com/openclaw/openclaw.git
synced 2026-05-09 19:30:44 +00:00
309 lines
9.9 KiB
TypeScript
309 lines
9.9 KiB
TypeScript
import type { Bot } from "grammy";
|
|
import {
|
|
createFinalizableDraftStreamControlsForState,
|
|
takeMessageIdAfterStop,
|
|
} from "openclaw/plugin-sdk/channel-lifecycle";
|
|
import { formatErrorMessage } from "openclaw/plugin-sdk/error-runtime";
|
|
import { buildTelegramThreadParams, type TelegramThreadSpec } from "./bot/helpers.js";
|
|
import { isSafeToRetrySendError, isTelegramClientRejection } from "./network-errors.js";
|
|
import { normalizeTelegramReplyToMessageId } from "./outbound-params.js";
|
|
|
|
const TELEGRAM_STREAM_MAX_CHARS = 4096;
|
|
const DEFAULT_THROTTLE_MS = 1000;
|
|
const THREAD_NOT_FOUND_RE = /400:\s*Bad Request:\s*message thread not found/i;
|
|
|
|
type TelegramSendMessageParams = Parameters<Bot["api"]["sendMessage"]>[2];
|
|
|
|
function hasNumericMessageThreadId(
|
|
params: TelegramSendMessageParams | undefined,
|
|
): params is TelegramSendMessageParams & { message_thread_id: number } {
|
|
return (
|
|
typeof params === "object" &&
|
|
params !== null &&
|
|
typeof (params as { message_thread_id?: unknown }).message_thread_id === "number"
|
|
);
|
|
}
|
|
|
|
export type TelegramDraftStream = {
|
|
update: (text: string) => void;
|
|
flush: () => Promise<void>;
|
|
messageId: () => number | undefined;
|
|
visibleSinceMs?: () => number | undefined;
|
|
previewRevision?: () => number;
|
|
lastDeliveredText?: () => string;
|
|
clear: () => Promise<void>;
|
|
stop: () => Promise<void>;
|
|
/** Stop without a final flush or delete. */
|
|
discard?: () => Promise<void>;
|
|
/** Return the current preview message id after pending updates settle. */
|
|
materialize?: () => Promise<number | undefined>;
|
|
/** Reset internal state so the next update creates a new message instead of editing. */
|
|
forceNewMessage: () => void;
|
|
/** True when a preview sendMessage was attempted but the response was lost. */
|
|
sendMayHaveLanded?: () => boolean;
|
|
};
|
|
|
|
type TelegramDraftPreview = {
|
|
text: string;
|
|
parseMode?: "HTML";
|
|
};
|
|
|
|
type SupersededTelegramPreview = {
|
|
messageId: number;
|
|
textSnapshot: string;
|
|
parseMode?: "HTML";
|
|
visibleSinceMs?: number;
|
|
};
|
|
|
|
export function createTelegramDraftStream(params: {
|
|
api: Bot["api"];
|
|
chatId: Parameters<Bot["api"]["sendMessage"]>[0];
|
|
maxChars?: number;
|
|
thread?: TelegramThreadSpec | null;
|
|
replyToMessageId?: number;
|
|
throttleMs?: number;
|
|
/** Minimum chars before sending first message (debounce for push notifications) */
|
|
minInitialChars?: number;
|
|
/** Optional preview renderer (e.g. markdown -> HTML + parse mode). */
|
|
renderText?: (text: string) => TelegramDraftPreview;
|
|
/** Called when a late send resolves after forceNewMessage() switched generations. */
|
|
onSupersededPreview?: (preview: SupersededTelegramPreview) => void;
|
|
log?: (message: string) => void;
|
|
warn?: (message: string) => void;
|
|
}): TelegramDraftStream {
|
|
const maxChars = Math.min(
|
|
params.maxChars ?? TELEGRAM_STREAM_MAX_CHARS,
|
|
TELEGRAM_STREAM_MAX_CHARS,
|
|
);
|
|
const throttleMs = Math.max(250, params.throttleMs ?? DEFAULT_THROTTLE_MS);
|
|
const minInitialChars = params.minInitialChars;
|
|
const chatId = params.chatId;
|
|
const threadParams = buildTelegramThreadParams(params.thread);
|
|
const replyToMessageId = normalizeTelegramReplyToMessageId(params.replyToMessageId);
|
|
const replyParams =
|
|
replyToMessageId != null
|
|
? {
|
|
...threadParams,
|
|
reply_to_message_id: replyToMessageId,
|
|
allow_sending_without_reply: true,
|
|
}
|
|
: threadParams;
|
|
|
|
const streamState = { stopped: false, final: false };
|
|
let messageSendAttempted = false;
|
|
let streamMessageId: number | undefined;
|
|
let streamVisibleSinceMs: number | undefined;
|
|
let lastSentText = "";
|
|
let lastDeliveredText = "";
|
|
let lastSentParseMode: "HTML" | undefined;
|
|
let previewRevision = 0;
|
|
let generation = 0;
|
|
type PreviewSendParams = {
|
|
renderedText: string;
|
|
renderedParseMode: "HTML" | undefined;
|
|
sendGeneration: number;
|
|
};
|
|
const sendRenderedMessageWithThreadFallback = async (sendArgs: {
|
|
renderedText: string;
|
|
renderedParseMode: "HTML" | undefined;
|
|
fallbackWarnMessage: string;
|
|
}) => {
|
|
const sendParams = sendArgs.renderedParseMode
|
|
? {
|
|
...replyParams,
|
|
parse_mode: sendArgs.renderedParseMode,
|
|
}
|
|
: replyParams;
|
|
const usedThreadParams = hasNumericMessageThreadId(sendParams);
|
|
try {
|
|
return {
|
|
sent: await params.api.sendMessage(chatId, sendArgs.renderedText, sendParams),
|
|
usedThreadParams,
|
|
};
|
|
} catch (err) {
|
|
if (!usedThreadParams || !THREAD_NOT_FOUND_RE.test(String(err))) {
|
|
throw err;
|
|
}
|
|
const threadlessParams: TelegramSendMessageParams = { ...sendParams };
|
|
delete threadlessParams.message_thread_id;
|
|
params.warn?.(sendArgs.fallbackWarnMessage);
|
|
return {
|
|
sent: await params.api.sendMessage(
|
|
chatId,
|
|
sendArgs.renderedText,
|
|
Object.keys(threadlessParams).length > 0 ? threadlessParams : undefined,
|
|
),
|
|
usedThreadParams: false,
|
|
};
|
|
}
|
|
};
|
|
const sendMessageTransportPreview = async ({
|
|
renderedText,
|
|
renderedParseMode,
|
|
sendGeneration,
|
|
}: PreviewSendParams): Promise<boolean> => {
|
|
if (typeof streamMessageId === "number") {
|
|
streamVisibleSinceMs ??= Date.now();
|
|
if (renderedParseMode) {
|
|
await params.api.editMessageText(chatId, streamMessageId, renderedText, {
|
|
parse_mode: renderedParseMode,
|
|
});
|
|
} else {
|
|
await params.api.editMessageText(chatId, streamMessageId, renderedText);
|
|
}
|
|
return true;
|
|
}
|
|
messageSendAttempted = true;
|
|
let sent: Awaited<ReturnType<typeof sendRenderedMessageWithThreadFallback>>["sent"];
|
|
try {
|
|
({ sent } = await sendRenderedMessageWithThreadFallback({
|
|
renderedText,
|
|
renderedParseMode,
|
|
fallbackWarnMessage:
|
|
"telegram stream preview send failed with message_thread_id, retrying without thread",
|
|
}));
|
|
} catch (err) {
|
|
if (isSafeToRetrySendError(err) || isTelegramClientRejection(err)) {
|
|
messageSendAttempted = false;
|
|
}
|
|
throw err;
|
|
}
|
|
const sentMessageId = sent?.message_id;
|
|
if (typeof sentMessageId !== "number" || !Number.isFinite(sentMessageId)) {
|
|
streamState.stopped = true;
|
|
params.warn?.("telegram stream preview stopped (missing message id from sendMessage)");
|
|
return false;
|
|
}
|
|
const normalizedMessageId = Math.trunc(sentMessageId);
|
|
const visibleSinceMs = Date.now();
|
|
if (sendGeneration !== generation) {
|
|
params.onSupersededPreview?.({
|
|
messageId: normalizedMessageId,
|
|
textSnapshot: renderedText,
|
|
parseMode: renderedParseMode,
|
|
visibleSinceMs,
|
|
});
|
|
return true;
|
|
}
|
|
streamMessageId = normalizedMessageId;
|
|
streamVisibleSinceMs = visibleSinceMs;
|
|
return true;
|
|
};
|
|
|
|
const sendOrEditStreamMessage = async (text: string): Promise<boolean> => {
|
|
if (streamState.stopped && !streamState.final) {
|
|
return false;
|
|
}
|
|
const trimmed = text.trimEnd();
|
|
if (!trimmed) {
|
|
return false;
|
|
}
|
|
const rendered = params.renderText?.(trimmed) ?? { text: trimmed };
|
|
const renderedText = rendered.text.trimEnd();
|
|
const renderedParseMode = rendered.parseMode;
|
|
if (!renderedText) {
|
|
return false;
|
|
}
|
|
if (renderedText.length > maxChars) {
|
|
streamState.stopped = true;
|
|
params.warn?.(
|
|
`telegram stream preview stopped (text length ${renderedText.length} > ${maxChars})`,
|
|
);
|
|
return false;
|
|
}
|
|
if (renderedText === lastSentText && renderedParseMode === lastSentParseMode) {
|
|
return true;
|
|
}
|
|
const sendGeneration = generation;
|
|
|
|
if (typeof streamMessageId !== "number" && minInitialChars != null && !streamState.final) {
|
|
if (renderedText.length < minInitialChars) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
lastSentText = renderedText;
|
|
lastSentParseMode = renderedParseMode;
|
|
try {
|
|
const sent = await sendMessageTransportPreview({
|
|
renderedText,
|
|
renderedParseMode,
|
|
sendGeneration,
|
|
});
|
|
if (sent) {
|
|
previewRevision += 1;
|
|
lastDeliveredText = trimmed;
|
|
}
|
|
return sent;
|
|
} catch (err) {
|
|
streamState.stopped = true;
|
|
params.warn?.(`telegram stream preview failed: ${formatErrorMessage(err)}`);
|
|
return false;
|
|
}
|
|
};
|
|
|
|
const { loop, update, stop, stopForClear } = createFinalizableDraftStreamControlsForState({
|
|
throttleMs,
|
|
state: streamState,
|
|
sendOrEditStreamMessage,
|
|
});
|
|
|
|
const clear = async () => {
|
|
const messageId = await takeMessageIdAfterStop({
|
|
stopForClear,
|
|
readMessageId: () => streamMessageId,
|
|
clearMessageId: () => {
|
|
streamMessageId = undefined;
|
|
},
|
|
});
|
|
if (typeof messageId === "number" && Number.isFinite(messageId)) {
|
|
try {
|
|
await params.api.deleteMessage(chatId, messageId);
|
|
params.log?.(`telegram stream preview deleted (chat=${chatId}, message=${messageId})`);
|
|
} catch (err) {
|
|
params.warn?.(`telegram stream preview cleanup failed: ${formatErrorMessage(err)}`);
|
|
}
|
|
return;
|
|
}
|
|
};
|
|
|
|
const discard = async () => {
|
|
await stopForClear();
|
|
};
|
|
|
|
const forceNewMessage = () => {
|
|
streamState.stopped = false;
|
|
streamState.final = false;
|
|
generation += 1;
|
|
messageSendAttempted = false;
|
|
streamMessageId = undefined;
|
|
streamVisibleSinceMs = undefined;
|
|
lastSentText = "";
|
|
lastSentParseMode = undefined;
|
|
loop.resetPending();
|
|
loop.resetThrottleWindow();
|
|
};
|
|
|
|
const materialize = async (): Promise<number | undefined> => {
|
|
await stop();
|
|
return streamMessageId;
|
|
};
|
|
|
|
params.log?.(`telegram stream preview ready (maxChars=${maxChars}, throttleMs=${throttleMs})`);
|
|
|
|
return {
|
|
update,
|
|
flush: loop.flush,
|
|
messageId: () => streamMessageId,
|
|
visibleSinceMs: () => streamVisibleSinceMs,
|
|
previewRevision: () => previewRevision,
|
|
lastDeliveredText: () => lastDeliveredText,
|
|
clear,
|
|
stop,
|
|
discard,
|
|
materialize,
|
|
forceNewMessage,
|
|
sendMayHaveLanded: () => messageSendAttempted && typeof streamMessageId !== "number",
|
|
};
|
|
}
|