import type { ClawdbotConfig } from "openclaw/plugin-sdk"; import { DEFAULT_ACCOUNT_ID, normalizeAccountId } from "./sdk-compat.js"; import type { FeishuConfig, FeishuAccountConfig, FeishuDomain, ResolvedFeishuAccount } from "./types.js"; /** * List all configured account IDs from the accounts field. */ function listConfiguredAccountIds(cfg: ClawdbotConfig): string[] { const accounts = (cfg.channels?.feishu as FeishuConfig)?.accounts; if (!accounts || typeof accounts !== "object") { return []; } return [...new Set( Object.keys(accounts) .filter((accountId) => accountId.trim()) .map((accountId) => normalizeAccountId(accountId)), )]; } /** * List all Feishu account IDs. * If no accounts are configured, returns [DEFAULT_ACCOUNT_ID] for backward compatibility. */ export function listFeishuAccountIds(cfg: ClawdbotConfig): string[] { const ids = listConfiguredAccountIds(cfg); if (ids.length === 0) { // Backward compatibility: no accounts configured, use default return [DEFAULT_ACCOUNT_ID]; } return [...ids].sort((a, b) => a.localeCompare(b)); } /** * Resolve the default account ID. */ export function resolveDefaultFeishuAccountId(cfg: ClawdbotConfig): string { const ids = listFeishuAccountIds(cfg); if (ids.includes(DEFAULT_ACCOUNT_ID)) { return DEFAULT_ACCOUNT_ID; } return ids[0] ?? DEFAULT_ACCOUNT_ID; } /** * Resolve the configured account key for a normalized account id. * Preserves the original config key casing so write paths can avoid shadow entries. */ export function resolveConfiguredFeishuAccountKey( cfg: ClawdbotConfig, accountId: string, ): string | undefined { const accounts = (cfg.channels?.feishu as FeishuConfig)?.accounts; if (!accounts || typeof accounts !== "object") { return undefined; } if (Object.prototype.hasOwnProperty.call(accounts, accountId)) { return accountId; } const normalizedId = normalizeAccountId(accountId); return Object.keys(accounts).find((key) => normalizeAccountId(key) === normalizedId); } /** * Get the raw account-specific config. * Preserves mixed-case config keys by resolving through the normalized account id. */ function resolveAccountConfig( cfg: ClawdbotConfig, accountId: string, ): FeishuAccountConfig | undefined { const accounts = (cfg.channels?.feishu as FeishuConfig)?.accounts; const matchedKey = resolveConfiguredFeishuAccountKey(cfg, accountId); return matchedKey && accounts ? accounts[matchedKey] : undefined; } /** * Merge top-level config with account-specific config. * Account-specific fields override top-level fields. */ function mergeFeishuAccountConfig( cfg: ClawdbotConfig, accountId: string, ): FeishuConfig { const feishuCfg = cfg.channels?.feishu as FeishuConfig | undefined; // Extract base config (exclude accounts field to avoid recursion) const { accounts: _ignored, ...base } = feishuCfg ?? {}; // Get account-specific overrides const account = resolveAccountConfig(cfg, accountId) ?? {}; // Merge: account config overrides base config return { ...base, ...account } as FeishuConfig; } /** * Resolve Feishu credentials from a config. */ export function resolveFeishuCredentials(cfg?: FeishuConfig): { appId: string; appSecret: string; encryptKey?: string; verificationToken?: string; domain: FeishuDomain; } | null { const appId = cfg?.appId?.trim(); const appSecret = cfg?.appSecret?.trim(); if (!appId || !appSecret) return null; return { appId, appSecret, encryptKey: cfg?.encryptKey?.trim() || undefined, verificationToken: cfg?.verificationToken?.trim() || undefined, domain: cfg?.domain ?? "feishu", }; } /** * Resolve a complete Feishu account with merged config. */ export function resolveFeishuAccount(params: { cfg: ClawdbotConfig; accountId?: string | null; }): ResolvedFeishuAccount { const accountId = normalizeAccountId(params.accountId); const feishuCfg = params.cfg.channels?.feishu as FeishuConfig | undefined; // Base enabled state (top-level) const baseEnabled = feishuCfg?.enabled !== false; // Merge configs const merged = mergeFeishuAccountConfig(params.cfg, accountId); // Account-level enabled state const accountEnabled = merged.enabled !== false; const enabled = baseEnabled && accountEnabled; // Resolve credentials from merged config const creds = resolveFeishuCredentials(merged); return { accountId, enabled, configured: Boolean(creds), name: (merged as FeishuAccountConfig).name?.trim() || undefined, appId: creds?.appId, appSecret: creds?.appSecret, encryptKey: creds?.encryptKey, verificationToken: creds?.verificationToken, domain: creds?.domain ?? "feishu", config: merged, }; } /** * List all enabled and configured accounts. */ export function listEnabledFeishuAccounts(cfg: ClawdbotConfig): ResolvedFeishuAccount[] { return listFeishuAccountIds(cfg) .map((accountId) => resolveFeishuAccount({ cfg, accountId })) .filter((account) => account.enabled && account.configured); }