Spaces:
Sleeping
Sleeping
| import type { OpenClawConfig } from "../config/config.js"; | |
| import type { TelegramAccountConfig } from "../config/types.js"; | |
| import { isTruthyEnvValue } from "../infra/env.js"; | |
| import { listBoundAccountIds, resolveDefaultAgentBoundAccountId } from "../routing/bindings.js"; | |
| import { DEFAULT_ACCOUNT_ID, normalizeAccountId } from "../routing/session-key.js"; | |
| import { resolveTelegramToken } from "./token.js"; | |
| const debugAccounts = (...args: unknown[]) => { | |
| if (isTruthyEnvValue(process.env.OPENCLAW_DEBUG_TELEGRAM_ACCOUNTS)) { | |
| console.warn("[telegram:accounts]", ...args); | |
| } | |
| }; | |
| export type ResolvedTelegramAccount = { | |
| accountId: string; | |
| enabled: boolean; | |
| name?: string; | |
| token: string; | |
| tokenSource: "env" | "tokenFile" | "config" | "none"; | |
| config: TelegramAccountConfig; | |
| }; | |
| function listConfiguredAccountIds(cfg: OpenClawConfig): string[] { | |
| const accounts = cfg.channels?.telegram?.accounts; | |
| if (!accounts || typeof accounts !== "object") { | |
| return []; | |
| } | |
| const ids = new Set<string>(); | |
| for (const key of Object.keys(accounts)) { | |
| if (!key) { | |
| continue; | |
| } | |
| ids.add(normalizeAccountId(key)); | |
| } | |
| return [...ids]; | |
| } | |
| export function listTelegramAccountIds(cfg: OpenClawConfig): string[] { | |
| const ids = Array.from( | |
| new Set([...listConfiguredAccountIds(cfg), ...listBoundAccountIds(cfg, "telegram")]), | |
| ); | |
| debugAccounts("listTelegramAccountIds", ids); | |
| if (ids.length === 0) { | |
| return [DEFAULT_ACCOUNT_ID]; | |
| } | |
| return ids.toSorted((a, b) => a.localeCompare(b)); | |
| } | |
| export function resolveDefaultTelegramAccountId(cfg: OpenClawConfig): string { | |
| const boundDefault = resolveDefaultAgentBoundAccountId(cfg, "telegram"); | |
| if (boundDefault) { | |
| return boundDefault; | |
| } | |
| const ids = listTelegramAccountIds(cfg); | |
| if (ids.includes(DEFAULT_ACCOUNT_ID)) { | |
| return DEFAULT_ACCOUNT_ID; | |
| } | |
| return ids[0] ?? DEFAULT_ACCOUNT_ID; | |
| } | |
| function resolveAccountConfig( | |
| cfg: OpenClawConfig, | |
| accountId: string, | |
| ): TelegramAccountConfig | undefined { | |
| const accounts = cfg.channels?.telegram?.accounts; | |
| if (!accounts || typeof accounts !== "object") { | |
| return undefined; | |
| } | |
| const direct = accounts[accountId] as TelegramAccountConfig | undefined; | |
| if (direct) { | |
| return direct; | |
| } | |
| const normalized = normalizeAccountId(accountId); | |
| const matchKey = Object.keys(accounts).find((key) => normalizeAccountId(key) === normalized); | |
| return matchKey ? (accounts[matchKey] as TelegramAccountConfig | undefined) : undefined; | |
| } | |
| function mergeTelegramAccountConfig(cfg: OpenClawConfig, accountId: string): TelegramAccountConfig { | |
| const { accounts: _ignored, ...base } = (cfg.channels?.telegram ?? | |
| {}) as TelegramAccountConfig & { accounts?: unknown }; | |
| const account = resolveAccountConfig(cfg, accountId) ?? {}; | |
| return { ...base, ...account }; | |
| } | |
| export function resolveTelegramAccount(params: { | |
| cfg: OpenClawConfig; | |
| accountId?: string | null; | |
| }): ResolvedTelegramAccount { | |
| const hasExplicitAccountId = Boolean(params.accountId?.trim()); | |
| const baseEnabled = params.cfg.channels?.telegram?.enabled !== false; | |
| const resolve = (accountId: string) => { | |
| const merged = mergeTelegramAccountConfig(params.cfg, accountId); | |
| const accountEnabled = merged.enabled !== false; | |
| const enabled = baseEnabled && accountEnabled; | |
| const tokenResolution = resolveTelegramToken(params.cfg, { accountId }); | |
| debugAccounts("resolve", { | |
| accountId, | |
| enabled, | |
| tokenSource: tokenResolution.source, | |
| }); | |
| return { | |
| accountId, | |
| enabled, | |
| name: merged.name?.trim() || undefined, | |
| token: tokenResolution.token, | |
| tokenSource: tokenResolution.source, | |
| config: merged, | |
| } satisfies ResolvedTelegramAccount; | |
| }; | |
| const normalized = normalizeAccountId(params.accountId); | |
| const primary = resolve(normalized); | |
| if (hasExplicitAccountId) { | |
| return primary; | |
| } | |
| if (primary.tokenSource !== "none") { | |
| return primary; | |
| } | |
| // If accountId is omitted, prefer a configured account token over failing on | |
| // the implicit "default" account. This keeps env-based setups working while | |
| // making config-only tokens work for things like heartbeats. | |
| const fallbackId = resolveDefaultTelegramAccountId(params.cfg); | |
| if (fallbackId === primary.accountId) { | |
| return primary; | |
| } | |
| const fallback = resolve(fallbackId); | |
| if (fallback.tokenSource === "none") { | |
| return primary; | |
| } | |
| return fallback; | |
| } | |
| export function listEnabledTelegramAccounts(cfg: OpenClawConfig): ResolvedTelegramAccount[] { | |
| return listTelegramAccountIds(cfg) | |
| .map((accountId) => resolveTelegramAccount({ cfg, accountId })) | |
| .filter((account) => account.enabled); | |
| } | |