mirror of
https://github.com/openclaw/openclaw.git
synced 2026-05-30 13:01:02 +00:00
Route Telegram sendMessage action replies through durable outbound delivery so completed agent responses remain retryable when the gateway send path times out.
Verified with focused Telegram/outbound tests, extension test typecheck, prepare build/check/full test gates, and green CI rerun for head 20b45687e1.
837 lines
29 KiB
TypeScript
837 lines
29 KiB
TypeScript
import type { AgentToolResult } from "@earendil-works/pi-agent-core";
|
|
import { readBooleanParam } from "openclaw/plugin-sdk/boolean-param";
|
|
import {
|
|
jsonResult,
|
|
readNumberParam,
|
|
readReactionParams,
|
|
readStringArrayParam,
|
|
readStringOrNumberParam,
|
|
readStringParam,
|
|
resolvePollMaxSelections,
|
|
resolveReactionMessageId,
|
|
} from "openclaw/plugin-sdk/channel-actions";
|
|
import {
|
|
buildOutboundSessionContext,
|
|
sendDurableMessageBatch,
|
|
type DurableMessageBatchSendResult,
|
|
} from "openclaw/plugin-sdk/channel-outbound";
|
|
import type { OpenClawConfig } from "openclaw/plugin-sdk/config-contracts";
|
|
import {
|
|
normalizeMessagePresentation,
|
|
renderMessagePresentationFallbackText,
|
|
} from "openclaw/plugin-sdk/interactive-runtime";
|
|
import type { MessagePresentation } from "openclaw/plugin-sdk/interactive-runtime";
|
|
import type { ReplyPayload } from "openclaw/plugin-sdk/reply-runtime";
|
|
import { resolveStorePath } from "openclaw/plugin-sdk/session-store-runtime";
|
|
import {
|
|
createTelegramActionGate,
|
|
resolveDefaultTelegramAccountId,
|
|
resolveTelegramPollActionGateState,
|
|
} from "./accounts.js";
|
|
import { resolveTelegramInlineButtons } from "./button-types.js";
|
|
import { notifyTelegramInboundEventOutboundSuccess } from "./inbound-event-delivery.js";
|
|
import {
|
|
resolveTelegramInlineButtonsScope,
|
|
resolveTelegramTargetChatType,
|
|
} from "./inline-buttons.js";
|
|
import { resolveTelegramInteractiveTextFallback } from "./interactive-fallback.js";
|
|
import { resolveTelegramPollVisibility } from "./poll-visibility.js";
|
|
import { resolveTelegramReactionLevel } from "./reaction-level.js";
|
|
import {
|
|
createForumTopicTelegram,
|
|
deleteMessageTelegram,
|
|
editForumTopicTelegram,
|
|
editMessageTelegram,
|
|
pinMessageTelegram,
|
|
reactMessageTelegram,
|
|
sendMessageTelegram,
|
|
sendPollTelegram,
|
|
sendStickerTelegram,
|
|
} from "./send.js";
|
|
import { getCacheStats, searchStickers } from "./sticker-cache.js";
|
|
import { normalizeTelegramOutboundTarget, parseTelegramTarget } from "./targets.js";
|
|
import { resolveTelegramToken } from "./token.js";
|
|
import { resolveTopicNameCacheScope, updateTopicName } from "./topic-name-cache.js";
|
|
|
|
export const telegramActionRuntime = {
|
|
createForumTopicTelegram,
|
|
deleteMessageTelegram,
|
|
editForumTopicTelegram,
|
|
editMessageTelegram,
|
|
getCacheStats,
|
|
pinMessageTelegram,
|
|
reactMessageTelegram,
|
|
searchStickers,
|
|
sendDurableMessageBatch,
|
|
sendMessageTelegram,
|
|
sendPollTelegram,
|
|
sendStickerTelegram,
|
|
};
|
|
|
|
const TELEGRAM_FORUM_TOPIC_ICON_COLORS = [
|
|
0x6fb9f0, 0xffd67e, 0xcb86db, 0x8eee98, 0xff93b2, 0xfb6f5f,
|
|
] as const;
|
|
const TELEGRAM_ACTION_ALIASES = {
|
|
createForumTopic: "createForumTopic",
|
|
delete: "deleteMessage",
|
|
deleteMessage: "deleteMessage",
|
|
edit: "editMessage",
|
|
editForumTopic: "editForumTopic",
|
|
editMessage: "editMessage",
|
|
poll: "poll",
|
|
react: "react",
|
|
searchSticker: "searchSticker",
|
|
send: "sendMessage",
|
|
sendMessage: "sendMessage",
|
|
sendSticker: "sendSticker",
|
|
sticker: "sendSticker",
|
|
stickerCacheStats: "stickerCacheStats",
|
|
"sticker-search": "searchSticker",
|
|
"topic-create": "createForumTopic",
|
|
"topic-edit": "editForumTopic",
|
|
} as const;
|
|
|
|
type TelegramActionName = (typeof TELEGRAM_ACTION_ALIASES)[keyof typeof TELEGRAM_ACTION_ALIASES];
|
|
type TelegramForumTopicIconColor = (typeof TELEGRAM_FORUM_TOPIC_ICON_COLORS)[number];
|
|
|
|
function readTelegramForumTopicIconColor(
|
|
params: Record<string, unknown>,
|
|
): TelegramForumTopicIconColor | undefined {
|
|
const iconColor = readNumberParam(params, "iconColor", { integer: true });
|
|
if (iconColor == null) {
|
|
return undefined;
|
|
}
|
|
if (!TELEGRAM_FORUM_TOPIC_ICON_COLORS.includes(iconColor as TelegramForumTopicIconColor)) {
|
|
throw new Error("iconColor must be one of Telegram's supported forum topic colors.");
|
|
}
|
|
return iconColor as TelegramForumTopicIconColor;
|
|
}
|
|
function normalizeTelegramActionName(action: string): TelegramActionName {
|
|
const normalized = TELEGRAM_ACTION_ALIASES[action as keyof typeof TELEGRAM_ACTION_ALIASES];
|
|
if (!normalized) {
|
|
throw new Error(`Unsupported Telegram action: ${action}`);
|
|
}
|
|
return normalized;
|
|
}
|
|
|
|
function readTelegramChatId(params: Record<string, unknown>) {
|
|
return (
|
|
readStringOrNumberParam(params, "chatId") ??
|
|
readStringOrNumberParam(params, "channelId") ??
|
|
readStringOrNumberParam(params, "to", { required: true })
|
|
);
|
|
}
|
|
|
|
function readTelegramThreadId(params: Record<string, unknown>) {
|
|
return (
|
|
readNumberParam(params, "messageThreadId", { integer: true }) ??
|
|
readNumberParam(params, "threadId", { integer: true })
|
|
);
|
|
}
|
|
|
|
function resolveActionTopicNameCacheScope(cfg: OpenClawConfig, accountId?: string | null): string {
|
|
const storePath = resolveStorePath(cfg.session?.store, {
|
|
agentId: accountId ?? resolveDefaultTelegramAccountId(cfg),
|
|
});
|
|
return resolveTopicNameCacheScope(storePath);
|
|
}
|
|
|
|
function formatTelegramDeliveryTarget(to: string, messageThreadId?: number | null): string {
|
|
const parsed = parseTelegramTarget(to);
|
|
const topicId = parsed.messageThreadId ?? messageThreadId;
|
|
if (topicId == null) {
|
|
return to;
|
|
}
|
|
return `${parsed.chatId}:topic:${topicId}`;
|
|
}
|
|
|
|
function readTelegramReplyToMessageId(params: Record<string, unknown>) {
|
|
return (
|
|
readNumberParam(params, "replyToMessageId", { integer: true }) ??
|
|
readNumberParam(params, "replyTo", { integer: true })
|
|
);
|
|
}
|
|
|
|
function pushTelegramMediaUrl(mediaUrls: string[], seen: Set<string>, value: unknown): void {
|
|
if (typeof value !== "string") {
|
|
return;
|
|
}
|
|
const normalized = value.trim();
|
|
if (!normalized || seen.has(normalized)) {
|
|
return;
|
|
}
|
|
seen.add(normalized);
|
|
mediaUrls.push(normalized);
|
|
}
|
|
|
|
function readTelegramSendMediaUrls(params: Record<string, unknown>) {
|
|
const mediaUrls: string[] = [];
|
|
const seen = new Set<string>();
|
|
pushTelegramMediaUrl(mediaUrls, seen, params.mediaUrl);
|
|
pushTelegramMediaUrl(mediaUrls, seen, params.media);
|
|
pushTelegramMediaUrl(mediaUrls, seen, params.path);
|
|
pushTelegramMediaUrl(mediaUrls, seen, params.filePath);
|
|
pushTelegramMediaUrl(mediaUrls, seen, params.fileUrl);
|
|
if (Array.isArray(params.mediaUrls)) {
|
|
for (const mediaUrl of params.mediaUrls) {
|
|
pushTelegramMediaUrl(mediaUrls, seen, mediaUrl);
|
|
}
|
|
}
|
|
if (Array.isArray(params.attachments)) {
|
|
for (const attachment of params.attachments) {
|
|
if (!attachment || typeof attachment !== "object" || Array.isArray(attachment)) {
|
|
continue;
|
|
}
|
|
const record = attachment as Record<string, unknown>;
|
|
pushTelegramMediaUrl(mediaUrls, seen, record.media);
|
|
pushTelegramMediaUrl(mediaUrls, seen, record.mediaUrl);
|
|
pushTelegramMediaUrl(mediaUrls, seen, record.path);
|
|
pushTelegramMediaUrl(mediaUrls, seen, record.filePath);
|
|
pushTelegramMediaUrl(mediaUrls, seen, record.fileUrl);
|
|
pushTelegramMediaUrl(mediaUrls, seen, record.url);
|
|
}
|
|
}
|
|
return mediaUrls;
|
|
}
|
|
|
|
function resolveTelegramButtonsFromParams(
|
|
params: Record<string, unknown>,
|
|
presentation = normalizeMessagePresentation(params.presentation),
|
|
) {
|
|
return resolveTelegramInlineButtons({
|
|
presentation,
|
|
interactive: params.interactive,
|
|
});
|
|
}
|
|
|
|
function readTelegramSendContent(params: {
|
|
args: Record<string, unknown>;
|
|
mediaUrl?: string;
|
|
hasButtons: boolean;
|
|
interactive?: unknown;
|
|
presentation?: MessagePresentation;
|
|
}) {
|
|
const explicitContent =
|
|
readStringParam(params.args, "content", { allowEmpty: true }) ??
|
|
readStringParam(params.args, "message", { allowEmpty: true }) ??
|
|
readStringParam(params.args, "caption", { allowEmpty: true });
|
|
const presentationText =
|
|
explicitContent == null && params.presentation
|
|
? renderMessagePresentationFallbackText({ presentation: params.presentation })
|
|
: undefined;
|
|
const interactiveText =
|
|
explicitContent == null && !params.presentation
|
|
? resolveTelegramInteractiveTextFallback({ interactive: params.interactive })
|
|
: undefined;
|
|
let content =
|
|
explicitContent ??
|
|
(presentationText?.trim() ? presentationText : undefined) ??
|
|
(interactiveText?.trim() ? interactiveText : undefined);
|
|
if ((content == null || content.trim().length === 0) && !params.mediaUrl && params.hasButtons) {
|
|
const fallback = presentationText?.trim() ? presentationText : interactiveText;
|
|
if (fallback?.trim()) {
|
|
content = fallback;
|
|
}
|
|
}
|
|
if (content == null && !params.mediaUrl && !params.hasButtons) {
|
|
throw new Error("content required.");
|
|
}
|
|
return content ?? "";
|
|
}
|
|
|
|
function normalizeTelegramDeliveryPin(params: Record<string, unknown>) {
|
|
const delivery = params.delivery;
|
|
const pin =
|
|
delivery && typeof delivery === "object" && !Array.isArray(delivery)
|
|
? (delivery as { pin?: unknown }).pin
|
|
: params.pin === true
|
|
? true
|
|
: undefined;
|
|
if (pin === true) {
|
|
return { enabled: true } as const;
|
|
}
|
|
if (!pin || typeof pin !== "object" || Array.isArray(pin)) {
|
|
return undefined;
|
|
}
|
|
const raw = pin as { enabled?: unknown; notify?: unknown; required?: unknown };
|
|
if (raw.enabled !== true) {
|
|
return undefined;
|
|
}
|
|
return {
|
|
enabled: true,
|
|
...(raw.notify === true ? { notify: true } : {}),
|
|
...(raw.required === true ? { required: true } : {}),
|
|
} as const;
|
|
}
|
|
|
|
function buildTelegramActionSendPayload(params: {
|
|
content: string;
|
|
mediaUrls: string[];
|
|
asVoice?: boolean;
|
|
pin?: ReturnType<typeof normalizeTelegramDeliveryPin>;
|
|
buttons?: ReturnType<typeof resolveTelegramButtonsFromParams>;
|
|
quoteText?: string;
|
|
}): ReplyPayload {
|
|
const telegramData =
|
|
params.buttons || params.quoteText
|
|
? {
|
|
...(params.buttons ? { buttons: params.buttons } : {}),
|
|
...(params.quoteText ? { quoteText: params.quoteText } : {}),
|
|
}
|
|
: undefined;
|
|
return {
|
|
text: params.content,
|
|
...(params.mediaUrls.length > 0 ? { mediaUrls: params.mediaUrls } : {}),
|
|
...(params.asVoice === true ? { audioAsVoice: true } : {}),
|
|
...(params.pin ? { delivery: { pin: params.pin } } : {}),
|
|
...(telegramData ? { channelData: { telegram: telegramData } } : {}),
|
|
};
|
|
}
|
|
|
|
function getLastDurableTelegramActionResult(
|
|
result: Extract<DurableMessageBatchSendResult, { status: "sent" }>,
|
|
): { messageId?: string; chatId?: string } {
|
|
const lastResult = result.results.at(-1);
|
|
const receipt = result.receipt;
|
|
return {
|
|
messageId:
|
|
lastResult?.messageId ??
|
|
receipt.primaryPlatformMessageId ??
|
|
receipt.platformMessageIds.at(-1),
|
|
chatId: lastResult?.chatId,
|
|
};
|
|
}
|
|
|
|
export async function handleTelegramAction(
|
|
params: Record<string, unknown>,
|
|
cfg: OpenClawConfig,
|
|
options?: {
|
|
mediaLocalRoots?: readonly string[];
|
|
mediaReadFile?: (filePath: string) => Promise<Buffer>;
|
|
sessionKey?: string | null;
|
|
inboundEventKind?: string;
|
|
gatewayClientScopes?: readonly string[];
|
|
},
|
|
): Promise<AgentToolResult<unknown>> {
|
|
const { action, accountId } = {
|
|
action: normalizeTelegramActionName(readStringParam(params, "action", { required: true })),
|
|
accountId: readStringParam(params, "accountId"),
|
|
};
|
|
const isActionEnabled = createTelegramActionGate({
|
|
cfg,
|
|
accountId,
|
|
});
|
|
const notifyVisibleOutboundSuccess = (to: string, messageThreadId?: number | null) => {
|
|
notifyTelegramInboundEventOutboundSuccess({
|
|
sessionKey: options?.sessionKey ?? undefined,
|
|
to: formatTelegramDeliveryTarget(to, messageThreadId),
|
|
accountId,
|
|
inboundEventKind: options?.inboundEventKind,
|
|
});
|
|
};
|
|
|
|
if (action === "react") {
|
|
// All react failures return soft results (jsonResult with ok:false) instead
|
|
// of throwing, because hard tool errors can trigger model re-generation
|
|
// loops and duplicate content.
|
|
const reactionLevelInfo = resolveTelegramReactionLevel({
|
|
cfg,
|
|
accountId: accountId ?? undefined,
|
|
});
|
|
if (!reactionLevelInfo.agentReactionsEnabled) {
|
|
return jsonResult({
|
|
ok: false,
|
|
reason: "disabled",
|
|
hint: `Telegram agent reactions disabled (reactionLevel="${reactionLevelInfo.level}"). Do not retry.`,
|
|
});
|
|
}
|
|
if (!isActionEnabled("reactions")) {
|
|
return jsonResult({
|
|
ok: false,
|
|
reason: "disabled",
|
|
hint: "Telegram reactions are disabled via actions.reactions. Do not retry.",
|
|
});
|
|
}
|
|
const chatId = readTelegramChatId(params);
|
|
const messageId =
|
|
readNumberParam(params, "messageId", { integer: true }) ??
|
|
resolveReactionMessageId({ args: params });
|
|
if (typeof messageId !== "number" || !Number.isFinite(messageId) || messageId <= 0) {
|
|
return jsonResult({
|
|
ok: false,
|
|
reason: "missing_message_id",
|
|
hint: "Telegram reaction requires a valid messageId (or inbound context fallback). Do not retry.",
|
|
});
|
|
}
|
|
const { emoji, remove, isEmpty } = readReactionParams(params, {
|
|
removeErrorMessage: "Emoji is required to remove a Telegram reaction.",
|
|
});
|
|
const token = resolveTelegramToken(cfg, { accountId }).token;
|
|
if (!token) {
|
|
return jsonResult({
|
|
ok: false,
|
|
reason: "missing_token",
|
|
hint: "Telegram bot token missing. Do not retry.",
|
|
});
|
|
}
|
|
let reactionResult: Awaited<ReturnType<typeof telegramActionRuntime.reactMessageTelegram>>;
|
|
try {
|
|
reactionResult = await telegramActionRuntime.reactMessageTelegram(
|
|
chatId ?? "",
|
|
messageId ?? 0,
|
|
emoji ?? "",
|
|
{
|
|
cfg,
|
|
token,
|
|
remove,
|
|
accountId: accountId ?? undefined,
|
|
gatewayClientScopes: options?.gatewayClientScopes,
|
|
},
|
|
);
|
|
} catch (err) {
|
|
const isInvalid = String(err).includes("REACTION_INVALID");
|
|
return jsonResult({
|
|
ok: false,
|
|
reason: isInvalid ? "REACTION_INVALID" : "error",
|
|
emoji,
|
|
hint: isInvalid
|
|
? "This emoji is not supported for Telegram reactions. Add it to your reaction disallow list so you do not try it again."
|
|
: "Reaction failed. Do not retry.",
|
|
});
|
|
}
|
|
if (!reactionResult.ok) {
|
|
return jsonResult({
|
|
ok: false,
|
|
warning: reactionResult.warning,
|
|
...(remove || isEmpty ? { removed: true } : { added: emoji }),
|
|
});
|
|
}
|
|
if (!remove && !isEmpty) {
|
|
return jsonResult({ ok: true, added: emoji });
|
|
}
|
|
return jsonResult({ ok: true, removed: true });
|
|
}
|
|
|
|
if (action === "sendMessage") {
|
|
if (!isActionEnabled("sendMessage")) {
|
|
throw new Error("Telegram sendMessage is disabled.");
|
|
}
|
|
const to = normalizeTelegramOutboundTarget(readStringParam(params, "to", { required: true }));
|
|
const mediaUrls = readTelegramSendMediaUrls(params);
|
|
const firstMediaUrl = mediaUrls[0];
|
|
const presentation = normalizeMessagePresentation(params.presentation);
|
|
const buttons = resolveTelegramButtonsFromParams(params, presentation);
|
|
const content = readTelegramSendContent({
|
|
args: params,
|
|
mediaUrl: firstMediaUrl,
|
|
hasButtons: Array.isArray(buttons) && buttons.length > 0,
|
|
interactive: params.interactive,
|
|
presentation,
|
|
});
|
|
if (buttons) {
|
|
const inlineButtonsScope = resolveTelegramInlineButtonsScope({
|
|
cfg,
|
|
accountId: accountId ?? undefined,
|
|
});
|
|
if (inlineButtonsScope === "off") {
|
|
throw new Error(
|
|
'Telegram inline buttons are disabled. Set channels.telegram.capabilities.inlineButtons to "dm", "group", "all", or "allowlist".',
|
|
);
|
|
}
|
|
if (inlineButtonsScope === "dm" || inlineButtonsScope === "group") {
|
|
const targetType = resolveTelegramTargetChatType(to);
|
|
if (targetType === "unknown") {
|
|
throw new Error(
|
|
`Telegram inline buttons require a numeric chat id when inlineButtons="${inlineButtonsScope}".`,
|
|
);
|
|
}
|
|
if (inlineButtonsScope === "dm" && targetType !== "direct") {
|
|
throw new Error('Telegram inline buttons are limited to DMs when inlineButtons="dm".');
|
|
}
|
|
if (inlineButtonsScope === "group" && targetType !== "group") {
|
|
throw new Error(
|
|
'Telegram inline buttons are limited to groups when inlineButtons="group".',
|
|
);
|
|
}
|
|
}
|
|
}
|
|
// Optional threading parameters for forum topics and reply chains
|
|
const replyToMessageId = readTelegramReplyToMessageId(params);
|
|
const messageThreadId = readTelegramThreadId(params);
|
|
const quoteText = readStringParam(params, "quoteText");
|
|
const token = resolveTelegramToken(cfg, { accountId }).token;
|
|
if (!token) {
|
|
throw new Error(
|
|
"Telegram bot token missing. Set TELEGRAM_BOT_TOKEN or channels.telegram.botToken.",
|
|
);
|
|
}
|
|
const sendOptions = {
|
|
cfg,
|
|
accountId: accountId ?? undefined,
|
|
gatewayClientScopes: options?.gatewayClientScopes,
|
|
replyToMessageId: replyToMessageId ?? undefined,
|
|
messageThreadId: messageThreadId ?? undefined,
|
|
quoteText: quoteText ?? undefined,
|
|
asVoice: readBooleanParam(params, "asVoice"),
|
|
silent: readBooleanParam(params, "silent"),
|
|
forceDocument:
|
|
readBooleanParam(params, "forceDocument") ??
|
|
readBooleanParam(params, "asDocument") ??
|
|
false,
|
|
};
|
|
const payload = buildTelegramActionSendPayload({
|
|
content,
|
|
mediaUrls,
|
|
asVoice: sendOptions.asVoice,
|
|
pin: normalizeTelegramDeliveryPin(params),
|
|
buttons,
|
|
quoteText,
|
|
});
|
|
const mediaAccess =
|
|
options?.mediaLocalRoots || options?.mediaReadFile
|
|
? {
|
|
...(options.mediaLocalRoots ? { localRoots: options.mediaLocalRoots } : {}),
|
|
...(options.mediaReadFile ? { readFile: options.mediaReadFile } : {}),
|
|
}
|
|
: undefined;
|
|
const outboundSession = buildOutboundSessionContext({
|
|
cfg,
|
|
sessionKey: options?.sessionKey,
|
|
requesterAccountId: accountId,
|
|
});
|
|
const durableResult = await telegramActionRuntime.sendDurableMessageBatch({
|
|
cfg,
|
|
channel: "telegram",
|
|
to,
|
|
accountId: accountId ?? undefined,
|
|
payloads: [payload],
|
|
replyToId: replyToMessageId == null ? undefined : String(replyToMessageId),
|
|
threadId: messageThreadId,
|
|
forceDocument: sendOptions.forceDocument,
|
|
silent: sendOptions.silent,
|
|
durability: "required",
|
|
gatewayClientScopes: options?.gatewayClientScopes,
|
|
...(mediaAccess ? { mediaAccess } : {}),
|
|
...(outboundSession ? { session: outboundSession } : {}),
|
|
});
|
|
if (durableResult.status === "failed" || durableResult.status === "partial_failed") {
|
|
throw durableResult.error;
|
|
}
|
|
if (durableResult.status === "suppressed") {
|
|
throw new Error("Telegram sendMessage was suppressed before delivery.");
|
|
}
|
|
const result = getLastDurableTelegramActionResult(durableResult);
|
|
notifyVisibleOutboundSuccess(to, messageThreadId);
|
|
return jsonResult({
|
|
ok: true,
|
|
messageId: result.messageId,
|
|
chatId: result.chatId,
|
|
});
|
|
}
|
|
|
|
if (action === "poll") {
|
|
const pollActionState = resolveTelegramPollActionGateState(isActionEnabled);
|
|
if (!pollActionState.sendMessageEnabled) {
|
|
throw new Error("Telegram sendMessage is disabled.");
|
|
}
|
|
if (!pollActionState.pollEnabled) {
|
|
throw new Error("Telegram polls are disabled.");
|
|
}
|
|
const to = readStringParam(params, "to", { required: true });
|
|
const question =
|
|
readStringParam(params, "question") ??
|
|
readStringParam(params, "pollQuestion", { required: true });
|
|
const answers =
|
|
readStringArrayParam(params, "answers") ??
|
|
readStringArrayParam(params, "pollOption", { required: true });
|
|
const allowMultiselect =
|
|
readBooleanParam(params, "allowMultiselect") ?? readBooleanParam(params, "pollMulti");
|
|
const durationSeconds =
|
|
readNumberParam(params, "durationSeconds", { integer: true }) ??
|
|
readNumberParam(params, "pollDurationSeconds", {
|
|
integer: true,
|
|
strict: true,
|
|
});
|
|
const durationHours =
|
|
readNumberParam(params, "durationHours", { integer: true }) ??
|
|
readNumberParam(params, "pollDurationHours", {
|
|
integer: true,
|
|
strict: true,
|
|
});
|
|
const replyToMessageId = readTelegramReplyToMessageId(params);
|
|
const messageThreadId = readTelegramThreadId(params);
|
|
const isAnonymous =
|
|
readBooleanParam(params, "isAnonymous") ??
|
|
resolveTelegramPollVisibility({
|
|
pollAnonymous: readBooleanParam(params, "pollAnonymous"),
|
|
pollPublic: readBooleanParam(params, "pollPublic"),
|
|
});
|
|
const silent = readBooleanParam(params, "silent");
|
|
const token = resolveTelegramToken(cfg, { accountId }).token;
|
|
if (!token) {
|
|
throw new Error(
|
|
"Telegram bot token missing. Set TELEGRAM_BOT_TOKEN or channels.telegram.botToken.",
|
|
);
|
|
}
|
|
const result = await telegramActionRuntime.sendPollTelegram(
|
|
to,
|
|
{
|
|
question,
|
|
options: answers,
|
|
maxSelections: resolvePollMaxSelections(answers.length, allowMultiselect ?? false),
|
|
durationSeconds: durationSeconds ?? undefined,
|
|
durationHours: durationHours ?? undefined,
|
|
},
|
|
{
|
|
cfg,
|
|
token,
|
|
accountId: accountId ?? undefined,
|
|
replyToMessageId: replyToMessageId ?? undefined,
|
|
messageThreadId: messageThreadId ?? undefined,
|
|
isAnonymous: isAnonymous ?? undefined,
|
|
silent: silent ?? undefined,
|
|
gatewayClientScopes: options?.gatewayClientScopes,
|
|
},
|
|
);
|
|
notifyVisibleOutboundSuccess(to, messageThreadId);
|
|
return jsonResult({
|
|
ok: true,
|
|
messageId: result.messageId,
|
|
chatId: result.chatId,
|
|
pollId: result.pollId,
|
|
});
|
|
}
|
|
|
|
if (action === "deleteMessage") {
|
|
if (!isActionEnabled("deleteMessage")) {
|
|
throw new Error("Telegram deleteMessage is disabled.");
|
|
}
|
|
const chatId = readTelegramChatId(params);
|
|
const messageId = readNumberParam(params, "messageId", {
|
|
required: true,
|
|
integer: true,
|
|
});
|
|
const token = resolveTelegramToken(cfg, { accountId }).token;
|
|
if (!token) {
|
|
throw new Error(
|
|
"Telegram bot token missing. Set TELEGRAM_BOT_TOKEN or channels.telegram.botToken.",
|
|
);
|
|
}
|
|
const result = await telegramActionRuntime.deleteMessageTelegram(chatId ?? "", messageId ?? 0, {
|
|
cfg,
|
|
token,
|
|
accountId: accountId ?? undefined,
|
|
gatewayClientScopes: options?.gatewayClientScopes,
|
|
});
|
|
if (!result.ok) {
|
|
return jsonResult({ ok: false, deleted: false, warning: result.warning });
|
|
}
|
|
return jsonResult({ ok: true, deleted: true });
|
|
}
|
|
|
|
if (action === "editMessage") {
|
|
if (!isActionEnabled("editMessage")) {
|
|
throw new Error("Telegram editMessage is disabled.");
|
|
}
|
|
const chatId = readTelegramChatId(params);
|
|
const messageId = readNumberParam(params, "messageId", {
|
|
required: true,
|
|
integer: true,
|
|
});
|
|
const content =
|
|
readStringParam(params, "content", { allowEmpty: false }) ??
|
|
readStringParam(params, "message", { required: true, allowEmpty: false });
|
|
const buttons = resolveTelegramButtonsFromParams(params);
|
|
if (buttons) {
|
|
const inlineButtonsScope = resolveTelegramInlineButtonsScope({
|
|
cfg,
|
|
accountId: accountId ?? undefined,
|
|
});
|
|
if (inlineButtonsScope === "off") {
|
|
throw new Error(
|
|
'Telegram inline buttons are disabled. Set channels.telegram.capabilities.inlineButtons to "dm", "group", "all", or "allowlist".',
|
|
);
|
|
}
|
|
}
|
|
const token = resolveTelegramToken(cfg, { accountId }).token;
|
|
if (!token) {
|
|
throw new Error(
|
|
"Telegram bot token missing. Set TELEGRAM_BOT_TOKEN or channels.telegram.botToken.",
|
|
);
|
|
}
|
|
const result = await telegramActionRuntime.editMessageTelegram(
|
|
chatId ?? "",
|
|
messageId ?? 0,
|
|
content,
|
|
{
|
|
cfg,
|
|
token,
|
|
accountId: accountId ?? undefined,
|
|
buttons,
|
|
gatewayClientScopes: options?.gatewayClientScopes,
|
|
},
|
|
);
|
|
return jsonResult({
|
|
ok: true,
|
|
messageId: result.messageId,
|
|
chatId: result.chatId,
|
|
});
|
|
}
|
|
|
|
if (action === "sendSticker") {
|
|
if (!isActionEnabled("sticker", false)) {
|
|
throw new Error(
|
|
"Telegram sticker actions are disabled. Set channels.telegram.actions.sticker to true.",
|
|
);
|
|
}
|
|
const to =
|
|
readStringParam(params, "to") ?? readStringParam(params, "target", { required: true });
|
|
const fileId =
|
|
readStringParam(params, "fileId") ?? readStringArrayParam(params, "stickerId")?.[0];
|
|
if (!fileId) {
|
|
throw new Error("fileId is required.");
|
|
}
|
|
const replyToMessageId = readTelegramReplyToMessageId(params);
|
|
const messageThreadId = readTelegramThreadId(params);
|
|
const token = resolveTelegramToken(cfg, { accountId }).token;
|
|
if (!token) {
|
|
throw new Error(
|
|
"Telegram bot token missing. Set TELEGRAM_BOT_TOKEN or channels.telegram.botToken.",
|
|
);
|
|
}
|
|
const result = await telegramActionRuntime.sendStickerTelegram(to, fileId, {
|
|
cfg,
|
|
token,
|
|
accountId: accountId ?? undefined,
|
|
replyToMessageId: replyToMessageId ?? undefined,
|
|
messageThreadId: messageThreadId ?? undefined,
|
|
gatewayClientScopes: options?.gatewayClientScopes,
|
|
});
|
|
notifyVisibleOutboundSuccess(to, messageThreadId);
|
|
return jsonResult({
|
|
ok: true,
|
|
messageId: result.messageId,
|
|
chatId: result.chatId,
|
|
});
|
|
}
|
|
|
|
if (action === "searchSticker") {
|
|
if (!isActionEnabled("sticker", false)) {
|
|
throw new Error(
|
|
"Telegram sticker actions are disabled. Set channels.telegram.actions.sticker to true.",
|
|
);
|
|
}
|
|
const query = readStringParam(params, "query", { required: true });
|
|
const limit = readNumberParam(params, "limit", { integer: true }) ?? 5;
|
|
const results = telegramActionRuntime.searchStickers(query, limit);
|
|
return jsonResult({
|
|
ok: true,
|
|
count: results.length,
|
|
stickers: results.map((s) => ({
|
|
fileId: s.fileId,
|
|
emoji: s.emoji,
|
|
description: s.description,
|
|
setName: s.setName,
|
|
})),
|
|
});
|
|
}
|
|
|
|
if (action === "stickerCacheStats") {
|
|
const stats = telegramActionRuntime.getCacheStats();
|
|
return jsonResult({ ok: true, ...stats });
|
|
}
|
|
|
|
if (action === "createForumTopic") {
|
|
if (!isActionEnabled("createForumTopic")) {
|
|
throw new Error("Telegram createForumTopic is disabled.");
|
|
}
|
|
const chatId = readTelegramChatId(params);
|
|
const name = readStringParam(params, "name", { required: true });
|
|
const iconColor = readTelegramForumTopicIconColor(params);
|
|
const iconCustomEmojiId = readStringParam(params, "iconCustomEmojiId");
|
|
const token = resolveTelegramToken(cfg, { accountId }).token;
|
|
if (!token) {
|
|
throw new Error(
|
|
"Telegram bot token missing. Set TELEGRAM_BOT_TOKEN or channels.telegram.botToken.",
|
|
);
|
|
}
|
|
const result = await telegramActionRuntime.createForumTopicTelegram(chatId ?? "", name, {
|
|
cfg,
|
|
token,
|
|
accountId: accountId ?? undefined,
|
|
iconColor,
|
|
iconCustomEmojiId: iconCustomEmojiId ?? undefined,
|
|
gatewayClientScopes: options?.gatewayClientScopes,
|
|
});
|
|
if (result.topicId != null && result.chatId) {
|
|
await updateTopicName(
|
|
result.chatId,
|
|
result.topicId,
|
|
{
|
|
name,
|
|
...(iconColor != null ? { iconColor } : {}),
|
|
...(iconCustomEmojiId ? { iconCustomEmojiId } : {}),
|
|
},
|
|
resolveActionTopicNameCacheScope(cfg, accountId),
|
|
).catch(() => {});
|
|
}
|
|
return jsonResult({
|
|
ok: true,
|
|
topicId: result.topicId,
|
|
name: result.name,
|
|
chatId: result.chatId,
|
|
});
|
|
}
|
|
|
|
if (action === "editForumTopic") {
|
|
if (!isActionEnabled("editForumTopic")) {
|
|
throw new Error("Telegram editForumTopic is disabled.");
|
|
}
|
|
const chatId = readTelegramChatId(params);
|
|
const messageThreadId = readTelegramThreadId(params);
|
|
if (typeof messageThreadId !== "number") {
|
|
throw new Error("messageThreadId or threadId is required.");
|
|
}
|
|
const name = readStringParam(params, "name");
|
|
const iconCustomEmojiId = readStringParam(params, "iconCustomEmojiId");
|
|
const token = resolveTelegramToken(cfg, { accountId }).token;
|
|
if (!token) {
|
|
throw new Error(
|
|
"Telegram bot token missing. Set TELEGRAM_BOT_TOKEN or channels.telegram.botToken.",
|
|
);
|
|
}
|
|
const result = await telegramActionRuntime.editForumTopicTelegram(
|
|
chatId ?? "",
|
|
messageThreadId,
|
|
{
|
|
cfg,
|
|
token,
|
|
accountId: accountId ?? undefined,
|
|
name: name ?? undefined,
|
|
iconCustomEmojiId: iconCustomEmojiId ?? undefined,
|
|
gatewayClientScopes: options?.gatewayClientScopes,
|
|
},
|
|
);
|
|
if (result.chatId) {
|
|
const patch: { name?: string; iconCustomEmojiId?: string } = {};
|
|
if (name) {
|
|
patch.name = name;
|
|
}
|
|
if (iconCustomEmojiId) {
|
|
patch.iconCustomEmojiId = iconCustomEmojiId;
|
|
}
|
|
if (Object.keys(patch).length > 0) {
|
|
await updateTopicName(
|
|
result.chatId,
|
|
result.messageThreadId,
|
|
patch,
|
|
resolveActionTopicNameCacheScope(cfg, accountId),
|
|
).catch(() => {});
|
|
}
|
|
}
|
|
return jsonResult(result);
|
|
}
|
|
|
|
throw new Error(`Unsupported Telegram action: ${String(action)}`);
|
|
}
|