UNPKG

@gguf/claw

Version:

Multi-channel AI gateway with extensible messaging integrations

271 lines (269 loc) 9.84 kB
import { D as getChatChannelMeta, O as listChatChannels, Q as isRecord, j as normalizeChatChannelId } from "./entry.js"; import { L as normalizeProviderId } from "./auth-profiles-DFa1zzNy.js"; import { t as hasAnyWhatsAppAuth } from "./accounts-BpzwDfBB.js"; import { n as getChannelPluginCatalogEntry, r as listChannelPluginCatalogEntries } from "./catalog-vmrRUSxJ.js"; import { t as ensurePluginAllowlisted } from "./plugins-allowlist-BPRqC3zX.js"; //#region src/config/plugin-auto-enable.ts const CHANNEL_PLUGIN_IDS = Array.from(new Set([...listChatChannels().map((meta) => meta.id), ...listChannelPluginCatalogEntries().map((entry) => entry.id)])); const PROVIDER_PLUGIN_IDS = [ { pluginId: "google-antigravity-auth", providerId: "google-antigravity" }, { pluginId: "google-gemini-cli-auth", providerId: "google-gemini-cli" }, { pluginId: "qwen-portal-auth", providerId: "qwen-portal" }, { pluginId: "copilot-proxy", providerId: "copilot-proxy" }, { pluginId: "minimax-portal-auth", providerId: "minimax-portal" } ]; function hasNonEmptyString(value) { return typeof value === "string" && value.trim().length > 0; } function recordHasKeys(value) { return isRecord(value) && Object.keys(value).length > 0; } function accountsHaveKeys(value, keys) { if (!isRecord(value)) return false; for (const account of Object.values(value)) { if (!isRecord(account)) continue; for (const key of keys) if (hasNonEmptyString(account[key])) return true; } return false; } function resolveChannelConfig(cfg, channelId) { const entry = cfg.channels?.[channelId]; return isRecord(entry) ? entry : null; } function isTelegramConfigured(cfg, env) { if (hasNonEmptyString(env.TELEGRAM_BOT_TOKEN)) return true; const entry = resolveChannelConfig(cfg, "telegram"); if (!entry) return false; if (hasNonEmptyString(entry.botToken) || hasNonEmptyString(entry.tokenFile)) return true; if (accountsHaveKeys(entry.accounts, ["botToken", "tokenFile"])) return true; return recordHasKeys(entry); } function isDiscordConfigured(cfg, env) { if (hasNonEmptyString(env.DISCORD_BOT_TOKEN)) return true; const entry = resolveChannelConfig(cfg, "discord"); if (!entry) return false; if (hasNonEmptyString(entry.token)) return true; if (accountsHaveKeys(entry.accounts, ["token"])) return true; return recordHasKeys(entry); } function isIrcConfigured(cfg, env) { if (hasNonEmptyString(env.IRC_HOST) && hasNonEmptyString(env.IRC_NICK)) return true; const entry = resolveChannelConfig(cfg, "irc"); if (!entry) return false; if (hasNonEmptyString(entry.host) || hasNonEmptyString(entry.nick)) return true; if (accountsHaveKeys(entry.accounts, ["host", "nick"])) return true; return recordHasKeys(entry); } function isSlackConfigured(cfg, env) { if (hasNonEmptyString(env.SLACK_BOT_TOKEN) || hasNonEmptyString(env.SLACK_APP_TOKEN) || hasNonEmptyString(env.SLACK_USER_TOKEN)) return true; const entry = resolveChannelConfig(cfg, "slack"); if (!entry) return false; if (hasNonEmptyString(entry.botToken) || hasNonEmptyString(entry.appToken) || hasNonEmptyString(entry.userToken)) return true; if (accountsHaveKeys(entry.accounts, [ "botToken", "appToken", "userToken" ])) return true; return recordHasKeys(entry); } function isSignalConfigured(cfg) { const entry = resolveChannelConfig(cfg, "signal"); if (!entry) return false; if (hasNonEmptyString(entry.account) || hasNonEmptyString(entry.httpUrl) || hasNonEmptyString(entry.httpHost) || typeof entry.httpPort === "number" || hasNonEmptyString(entry.cliPath)) return true; if (accountsHaveKeys(entry.accounts, [ "account", "httpUrl", "httpHost", "cliPath" ])) return true; return recordHasKeys(entry); } function isIMessageConfigured(cfg) { const entry = resolveChannelConfig(cfg, "imessage"); if (!entry) return false; if (hasNonEmptyString(entry.cliPath)) return true; return recordHasKeys(entry); } function isWhatsAppConfigured(cfg) { if (hasAnyWhatsAppAuth(cfg)) return true; const entry = resolveChannelConfig(cfg, "whatsapp"); if (!entry) return false; return recordHasKeys(entry); } function isGenericChannelConfigured(cfg, channelId) { return recordHasKeys(resolveChannelConfig(cfg, channelId)); } function isChannelConfigured(cfg, channelId, env = process.env) { switch (channelId) { case "whatsapp": return isWhatsAppConfigured(cfg); case "telegram": return isTelegramConfigured(cfg, env); case "discord": return isDiscordConfigured(cfg, env); case "irc": return isIrcConfigured(cfg, env); case "slack": return isSlackConfigured(cfg, env); case "signal": return isSignalConfigured(cfg); case "imessage": return isIMessageConfigured(cfg); default: return isGenericChannelConfigured(cfg, channelId); } } function collectModelRefs(cfg) { const refs = []; const pushModelRef = (value) => { if (typeof value === "string" && value.trim()) refs.push(value.trim()); }; const collectFromAgent = (agent) => { if (!agent) return; const model = agent.model; if (typeof model === "string") pushModelRef(model); else if (isRecord(model)) { pushModelRef(model.primary); const fallbacks = model.fallbacks; if (Array.isArray(fallbacks)) for (const entry of fallbacks) pushModelRef(entry); } const models = agent.models; if (isRecord(models)) for (const key of Object.keys(models)) pushModelRef(key); }; const defaults = cfg.agents?.defaults; collectFromAgent(defaults); const list = cfg.agents?.list; if (Array.isArray(list)) { for (const entry of list) if (isRecord(entry)) collectFromAgent(entry); } return refs; } function extractProviderFromModelRef(value) { const trimmed = value.trim(); const slash = trimmed.indexOf("/"); if (slash <= 0) return null; return normalizeProviderId(trimmed.slice(0, slash)); } function isProviderConfigured(cfg, providerId) { const normalized = normalizeProviderId(providerId); const profiles = cfg.auth?.profiles; if (profiles && typeof profiles === "object") for (const profile of Object.values(profiles)) { if (!isRecord(profile)) continue; if (normalizeProviderId(String(profile.provider ?? "")) === normalized) return true; } const providerConfig = cfg.models?.providers; if (providerConfig && typeof providerConfig === "object") { for (const key of Object.keys(providerConfig)) if (normalizeProviderId(key) === normalized) return true; } const modelRefs = collectModelRefs(cfg); for (const ref of modelRefs) { const provider = extractProviderFromModelRef(ref); if (provider && provider === normalized) return true; } return false; } function resolveConfiguredPlugins(cfg, env) { const changes = []; const channelIds = new Set(CHANNEL_PLUGIN_IDS); const configuredChannels = cfg.channels; if (configuredChannels && typeof configuredChannels === "object") for (const key of Object.keys(configuredChannels)) { if (key === "defaults") continue; channelIds.add(key); } for (const channelId of channelIds) { if (!channelId) continue; if (isChannelConfigured(cfg, channelId, env)) changes.push({ pluginId: channelId, reason: `${channelId} configured` }); } for (const mapping of PROVIDER_PLUGIN_IDS) if (isProviderConfigured(cfg, mapping.providerId)) changes.push({ pluginId: mapping.pluginId, reason: `${mapping.providerId} auth configured` }); return changes; } function isPluginExplicitlyDisabled(cfg, pluginId) { return (cfg.plugins?.entries?.[pluginId])?.enabled === false; } function isPluginDenied(cfg, pluginId) { const deny = cfg.plugins?.deny; return Array.isArray(deny) && deny.includes(pluginId); } function resolvePreferredOverIds(pluginId) { const normalized = normalizeChatChannelId(pluginId); if (normalized) return getChatChannelMeta(normalized).preferOver ?? []; return getChannelPluginCatalogEntry(pluginId)?.meta.preferOver ?? []; } function shouldSkipPreferredPluginAutoEnable(cfg, entry, configured) { for (const other of configured) { if (other.pluginId === entry.pluginId) continue; if (isPluginDenied(cfg, other.pluginId)) continue; if (isPluginExplicitlyDisabled(cfg, other.pluginId)) continue; if (resolvePreferredOverIds(other.pluginId).includes(entry.pluginId)) return true; } return false; } function registerPluginEntry(cfg, pluginId) { const entries = { ...cfg.plugins?.entries, [pluginId]: { ...cfg.plugins?.entries?.[pluginId], enabled: true } }; return { ...cfg, plugins: { ...cfg.plugins, entries } }; } function formatAutoEnableChange(entry) { let reason = entry.reason.trim(); const channelId = normalizeChatChannelId(entry.pluginId); if (channelId) { const label = getChatChannelMeta(channelId).label; reason = reason.replace(new RegExp(`^${channelId}\\b`, "i"), label); } return `${reason}, enabled automatically.`; } function applyPluginAutoEnable(params) { const env = params.env ?? process.env; const configured = resolveConfiguredPlugins(params.config, env); if (configured.length === 0) return { config: params.config, changes: [] }; let next = params.config; const changes = []; if (next.plugins?.enabled === false) return { config: next, changes }; for (const entry of configured) { if (isPluginDenied(next, entry.pluginId)) continue; if (isPluginExplicitlyDisabled(next, entry.pluginId)) continue; if (shouldSkipPreferredPluginAutoEnable(next, entry, configured)) continue; const allow = next.plugins?.allow; const allowMissing = Array.isArray(allow) && !allow.includes(entry.pluginId); if (next.plugins?.entries?.[entry.pluginId]?.enabled === true && !allowMissing) continue; next = registerPluginEntry(next, entry.pluginId); next = ensurePluginAllowlisted(next, entry.pluginId); changes.push(formatAutoEnableChange(entry)); } return { config: next, changes }; } //#endregion export { isChannelConfigured as n, applyPluginAutoEnable as t };