fix(telegram): report unauthorized startup tokens

This commit is contained in:
Peter Steinberger
2026-04-28 06:50:16 +01:00
parent 76a07b9a07
commit 4397717322
5 changed files with 155 additions and 8 deletions

View File

@@ -0,0 +1,118 @@
import {
createPluginRuntimeMock,
createStartAccountContext,
} from "openclaw/plugin-sdk/channel-test-helpers";
import type { OpenClawConfig } from "openclaw/plugin-sdk/config-types";
import { afterEach, describe, expect, it, vi } from "vitest";
import { telegramPlugin } from "./channel.js";
import type { TelegramMonitorFn } from "./monitor.types.js";
import { clearTelegramRuntime, setTelegramRuntime } from "./runtime.js";
import type { TelegramProbeFn } from "./runtime.types.js";
import type { TelegramRuntime } from "./runtime.types.js";
const probeTelegram = vi.fn();
const monitorTelegramProvider = vi.fn();
function installTelegramRuntime() {
const runtime = createPluginRuntimeMock();
setTelegramRuntime({
...runtime,
channel: {
...runtime.channel,
telegram: {
probeTelegram: probeTelegram as TelegramProbeFn,
monitorTelegramProvider: monitorTelegramProvider as TelegramMonitorFn,
},
},
} as unknown as TelegramRuntime);
}
function createTelegramConfig(accountId = "default"): OpenClawConfig {
if (accountId === "default") {
return {
channels: {
telegram: {
botToken: "123456:bad-token",
},
},
} as OpenClawConfig;
}
return {
channels: {
telegram: {
accounts: {
[accountId]: {
botToken: "123456:bad-token",
},
},
},
},
} as OpenClawConfig;
}
function startTelegramAccount(accountId = "default") {
const cfg = createTelegramConfig(accountId);
const account = telegramPlugin.config.resolveAccount(cfg, accountId);
const startAccount = telegramPlugin.gateway?.startAccount;
expect(startAccount).toBeDefined();
const ctx = createStartAccountContext({
account,
cfg,
});
return {
ctx,
task: startAccount!(ctx),
};
}
afterEach(() => {
clearTelegramRuntime();
probeTelegram.mockReset();
monitorTelegramProvider.mockReset();
});
describe("telegramPlugin gateway startup", () => {
it("stops before monitor startup when getMe rejects the token", async () => {
installTelegramRuntime();
probeTelegram.mockResolvedValue({
ok: false,
status: 401,
error: "Unauthorized",
elapsedMs: 12,
});
const { ctx, task } = startTelegramAccount("ops");
await expect(task).rejects.toThrow(
'Telegram bot token unauthorized for account "ops" (getMe returned 401',
);
await expect(task).rejects.toThrow("channels.telegram.accounts.ops.botToken/tokenFile");
expect(monitorTelegramProvider).not.toHaveBeenCalled();
expect(ctx.log?.error).toHaveBeenCalledWith(
expect.stringContaining('Telegram bot token unauthorized for account "ops"'),
);
});
it("keeps existing fallback startup for non-auth probe failures", async () => {
installTelegramRuntime();
probeTelegram.mockResolvedValue({
ok: false,
status: 500,
error: "Bad Gateway",
elapsedMs: 12,
});
monitorTelegramProvider.mockResolvedValue(undefined);
const { task } = startTelegramAccount();
await expect(task).resolves.toBeUndefined();
expect(monitorTelegramProvider).toHaveBeenCalledWith(
expect.objectContaining({
token: "123456:bad-token",
accountId: "default",
useWebhook: false,
}),
);
});
});

View File

@@ -129,6 +129,16 @@ function resolveTelegramMonitor() {
);
}
function formatTelegramUnauthorizedTokenError(account: ResolvedTelegramAccount): string {
const source =
account.tokenSource === "none" ? "no configured token" : `${account.tokenSource} token`;
const credentialPath =
account.accountId === DEFAULT_ACCOUNT_ID
? "channels.telegram.botToken, channels.telegram.tokenFile, or TELEGRAM_BOT_TOKEN"
: `channels.telegram.accounts.${account.accountId}.botToken/tokenFile`;
return `Telegram bot token unauthorized for account "${account.accountId}" (getMe returned 401 from Telegram; source: ${source}). Update ${credentialPath} with the current BotFather token.`;
}
function getOptionalTelegramRuntime() {
try {
return getTelegramRuntime();
@@ -880,6 +890,7 @@ export const telegramPlugin = createChatChannelPlugin({
}
const token = (account.token ?? "").trim();
let telegramBotLabel = "";
let unauthorizedTokenReason: string | null = null;
try {
const probe = await resolveTelegramProbe()(token, 2500, {
accountId: account.accountId,
@@ -892,11 +903,18 @@ export const telegramPlugin = createChatChannelPlugin({
if (username) {
telegramBotLabel = ` (@${username})`;
}
if (!probe.ok && probe.status === 401) {
unauthorizedTokenReason = formatTelegramUnauthorizedTokenError(account);
}
} catch (err) {
if (getTelegramRuntime().logging.shouldLogVerbose()) {
ctx.log?.debug?.(`[${account.accountId}] bot probe failed: ${String(err)}`);
}
}
if (unauthorizedTokenReason) {
ctx.log?.error?.(`[${account.accountId}] ${unauthorizedTokenReason}`);
throw new Error(unauthorizedTokenReason);
}
ctx.log?.info(`[${account.accountId}] starting provider${telegramBotLabel}`);
const setStatus = createAccountStatusSink({
accountId: account.accountId,