import fs from "node:fs"; import os from "node:os"; import path from "node:path"; import { afterEach, describe, expect, it, vi } from "vitest"; import type { OpenClawConfig } from "../config/config.js"; import { withStateDirEnv } from "../test-helpers/state-dir-env.js"; import { resolveTelegramToken } from "./token.js"; import { readTelegramUpdateOffset, writeTelegramUpdateOffset } from "./update-offset-store.js"; function withTempDir(): string { return fs.mkdtempSync(path.join(os.tmpdir(), "openclaw-telegram-token-")); } describe("resolveTelegramToken", () => { afterEach(() => { vi.unstubAllEnvs(); }); it("prefers config token over env", () => { vi.stubEnv("TELEGRAM_BOT_TOKEN", "env-token"); const cfg = { channels: { telegram: { botToken: "cfg-token" } }, } as OpenClawConfig; const res = resolveTelegramToken(cfg); expect(res.token).toBe("cfg-token"); expect(res.source).toBe("config"); }); it("uses env token when config is missing", () => { vi.stubEnv("TELEGRAM_BOT_TOKEN", "env-token"); const cfg = { channels: { telegram: {} }, } as OpenClawConfig; const res = resolveTelegramToken(cfg); expect(res.token).toBe("env-token"); expect(res.source).toBe("env"); }); it("uses tokenFile when configured", () => { vi.stubEnv("TELEGRAM_BOT_TOKEN", ""); const dir = withTempDir(); const tokenFile = path.join(dir, "token.txt"); fs.writeFileSync(tokenFile, "file-token\n", "utf-8"); const cfg = { channels: { telegram: { tokenFile } } } as OpenClawConfig; const res = resolveTelegramToken(cfg); expect(res.token).toBe("file-token"); expect(res.source).toBe("tokenFile"); fs.rmSync(dir, { recursive: true, force: true }); }); it("falls back to config token when no env or tokenFile", () => { vi.stubEnv("TELEGRAM_BOT_TOKEN", ""); const cfg = { channels: { telegram: { botToken: "cfg-token" } }, } as OpenClawConfig; const res = resolveTelegramToken(cfg); expect(res.token).toBe("cfg-token"); expect(res.source).toBe("config"); }); it("does not fall back to config when tokenFile is missing", () => { vi.stubEnv("TELEGRAM_BOT_TOKEN", ""); const dir = withTempDir(); const tokenFile = path.join(dir, "missing-token.txt"); const cfg = { channels: { telegram: { tokenFile, botToken: "cfg-token" } }, } as OpenClawConfig; const res = resolveTelegramToken(cfg); expect(res.token).toBe(""); expect(res.source).toBe("none"); fs.rmSync(dir, { recursive: true, force: true }); }); it("resolves per-account tokens when the config account key casing doesn't match routing normalization", () => { vi.stubEnv("TELEGRAM_BOT_TOKEN", ""); const cfg = { channels: { telegram: { accounts: { // Note the mixed-case key; runtime accountId is normalized. careyNotifications: { botToken: "acct-token" }, }, }, }, } as OpenClawConfig; const res = resolveTelegramToken(cfg, { accountId: "careynotifications" }); expect(res.token).toBe("acct-token"); expect(res.source).toBe("config"); }); it("falls back to top-level token for non-default accounts without account token", () => { const cfg = { channels: { telegram: { botToken: "top-level-token", accounts: { work: {}, }, }, }, } as OpenClawConfig; const res = resolveTelegramToken(cfg, { accountId: "work" }); expect(res.token).toBe("top-level-token"); expect(res.source).toBe("config"); }); it("falls back to top-level tokenFile for non-default accounts", () => { const dir = withTempDir(); const tokenFile = path.join(dir, "token.txt"); fs.writeFileSync(tokenFile, "file-token\n", "utf-8"); const cfg = { channels: { telegram: { tokenFile, accounts: { work: {}, }, }, }, } as OpenClawConfig; const res = resolveTelegramToken(cfg, { accountId: "work" }); expect(res.token).toBe("file-token"); expect(res.source).toBe("tokenFile"); fs.rmSync(dir, { recursive: true, force: true }); }); it("throws when botToken is an unresolved SecretRef object", () => { const cfg = { channels: { telegram: { botToken: { source: "env", provider: "default", id: "TELEGRAM_BOT_TOKEN" }, }, }, } as unknown as OpenClawConfig; expect(() => resolveTelegramToken(cfg)).toThrow( /channels\.telegram\.botToken: unresolved SecretRef/i, ); }); }); describe("telegram update offset store", () => { it("persists and reloads the last update id", async () => { await withStateDirEnv("openclaw-telegram-", async () => { expect(await readTelegramUpdateOffset({ accountId: "primary" })).toBeNull(); await writeTelegramUpdateOffset({ accountId: "primary", updateId: 421, }); expect(await readTelegramUpdateOffset({ accountId: "primary" })).toBe(421); }); }); });