Files
openclaw/scripts/e2e/telegram-bot-api.ts
2026-05-28 22:01:36 +02:00

128 lines
3.9 KiB
TypeScript

type JsonObject = Record<string, unknown>;
type TelegramBotApiOptions = {
baseUrl?: string;
fetchImpl?: (url: string, init: RequestInit) => Promise<Response>;
maxBodyBytes?: number;
timeoutMs?: number;
};
const DEFAULT_BASE_URL =
process.env.OPENCLAW_TELEGRAM_USER_BOT_API_BASE_URL ?? "https://api.telegram.org";
const DEFAULT_TIMEOUT_MS = readPositiveInt(
process.env.OPENCLAW_TELEGRAM_USER_BOT_API_TIMEOUT_MS,
30000,
);
const DEFAULT_BODY_MAX_BYTES = readPositiveInt(
process.env.OPENCLAW_TELEGRAM_USER_BOT_API_BODY_MAX_BYTES,
1024 * 1024,
);
function readPositiveInt(raw: string | undefined, fallback: number) {
const parsed = Number.parseInt(raw ?? "", 10);
return Number.isInteger(parsed) && parsed > 0 ? parsed : fallback;
}
function optionalString(source: JsonObject, key: string) {
const value = source[key];
return typeof value === "string" && value.trim() ? value.trim() : undefined;
}
function taggedError(message: string, code: string) {
return Object.assign(new Error(message), { code });
}
async function readBoundedResponseText(
response: Response,
label: string,
byteLimit: number,
timeoutPromise: Promise<never>,
) {
const contentLength = response.headers.get("content-length");
if (contentLength) {
const parsedLength = Number(contentLength);
if (Number.isSafeInteger(parsedLength) && parsedLength > byteLimit) {
await response.body?.cancel().catch(() => {});
throw taggedError(`${label} response body exceeded ${byteLimit} bytes`, "ETOOBIG");
}
}
if (!response.body) {
return "";
}
const reader = response.body.getReader();
const decoder = new TextDecoder();
let byteCount = 0;
let text = "";
try {
while (true) {
const { done, value } = await Promise.race([reader.read(), timeoutPromise]);
if (done) {
return text + decoder.decode();
}
byteCount += value.byteLength;
if (byteCount > byteLimit) {
await reader.cancel().catch(() => {});
throw taggedError(`${label} response body exceeded ${byteLimit} bytes`, "ETOOBIG");
}
text += decoder.decode(value, { stream: true });
}
} finally {
reader.releaseLock();
}
}
function parseJsonPayload(rawPayload: string, label: string) {
try {
return JSON.parse(rawPayload) as JsonObject;
} catch (error) {
throw new Error(`${label} returned invalid JSON`, { cause: error });
}
}
export async function telegramBotApi(
token: string,
method: string,
body: JsonObject = {},
options: TelegramBotApiOptions = {},
) {
const baseUrl = options.baseUrl ?? DEFAULT_BASE_URL;
const timeoutMs = Math.max(1, options.timeoutMs ?? DEFAULT_TIMEOUT_MS);
const maxBodyBytes = Math.max(1, options.maxBodyBytes ?? DEFAULT_BODY_MAX_BYTES);
const label = `Telegram Bot API ${method}`;
const timeoutError = taggedError(`${label} timed out after ${timeoutMs}ms`, "ETIMEDOUT");
const controller = new AbortController();
let timeout: NodeJS.Timeout | undefined;
const timeoutPromise = new Promise<never>((_, reject) => {
timeout = setTimeout(() => {
controller.abort(timeoutError);
reject(timeoutError);
}, timeoutMs);
timeout.unref?.();
});
try {
const response = await Promise.race([
(options.fetchImpl ?? fetch)(`${baseUrl}/bot${token}/${method}`, {
method: "POST",
headers: { "content-type": "application/json" },
body: JSON.stringify(body),
signal: controller.signal,
}),
timeoutPromise,
]);
const rawPayload = await readBoundedResponseText(response, label, maxBodyBytes, timeoutPromise);
const payload = parseJsonPayload(rawPayload, label);
if (!response.ok || payload.ok !== true) {
throw new Error(
optionalString(payload, "description") ?? `${method} failed with HTTP ${response.status}`,
);
}
return payload.result;
} finally {
if (timeout) {
clearTimeout(timeout);
}
}
}