mirror of
https://github.com/openclaw/openclaw.git
synced 2026-06-02 15:21:04 +00:00
* refactor: share talk event metric extraction * refactor: reuse shared coercion helpers * refactor: reuse shared primitive guards * refactor: reuse shared record guard * refactor: reuse shared primitive helpers * refactor: reuse shared string guards * refactor: reuse shared non-empty string guard * refactor: share plugin primitive coercion helpers * refactor: reuse plugin coercion helpers * refactor: reuse plugin coercion helpers in more plugins * refactor: reuse channel coercion helpers * refactor: reuse monitor coercion helpers * refactor: reuse provider coercion helpers * refactor: reuse core coercion helpers * refactor: reuse runtime coercion helpers * refactor: reuse helper coercion in codex paths * refactor: reuse helper coercion in runtime paths * refactor: reuse codex app-server coercion helpers * refactor: reuse codex record helpers * refactor: reuse migration and qa record helpers * refactor: reuse feishu and core helper guards * refactor: reuse browser and policy coercion helpers * refactor: reuse memory wiki record helper * refactor: share boolean coercion helpers * refactor: reuse finite number coercion * refactor: reuse trimmed string list helpers * refactor: reuse string list normalization * refactor: reuse remaining string list helpers * refactor: reuse string entry normalizer * refactor: share sorted string helpers * refactor: share string list normalization * test: preserve command registry browser imports * refactor: reuse trimmed list helpers * refactor: reuse string dedupe helpers * refactor: reuse local dedupe helpers * refactor: reuse more string dedupe helpers * refactor: reuse command string dedupe helpers * refactor: dedupe memory path lists with helper * refactor: expose string dedupe helpers to plugins * refactor: reuse core string dedupe helpers * refactor: reuse shared unique value helpers * refactor: reuse unique helpers in agent utilities * refactor: reuse unique helpers in config plumbing * refactor: reuse unique helpers in extensions * refactor: reuse unique helpers in core utilities * refactor: reuse unique helpers in qa plugins * refactor: reuse unique helpers in memory plugins * refactor: reuse unique helpers in channel plugins * refactor: reuse unique helpers in core tails * refactor: reuse unique helper in comfy workflow * refactor: reuse unique helpers in test utilities * refactor: expose unique value helper to plugins * refactor: reuse unique helpers for numeric lists * refactor: replace index dedupe filters * refactor: reuse string entry normalization * refactor: reuse string normalization in plugin helpers * refactor: reuse string normalization in extension helpers * refactor: reuse string normalization in channel parsers * refactor: reuse string normalization in memory search * refactor: reuse string normalization in provider parsers * refactor: reuse string normalization in qa helpers * refactor: reuse string normalization in infra parsers * refactor: reuse string normalization in messaging parsers * refactor: reuse string normalization in core parsers * refactor: reuse string normalization in extension parsers * refactor: reuse string normalization in remaining parsers * refactor: reuse string normalization in final parser spots * refactor: reuse string normalization in qa media helpers * refactor: reuse normalization in provider and media lists * refactor: reuse normalization for remaining set filters * refactor: reuse normalization in policy allowlists * refactor: reuse normalization in session and owner lists * refactor: centralize primitive string lists * refactor: reuse lowercase entry helpers * refactor: reuse sorted string helpers * refactor: reuse unique trimmed helpers * refactor: reuse string normalization helpers * refactor: reuse catalog string helpers * refactor: reuse remaining string helpers * refactor: simplify remaining list normalization * refactor: reuse codex auth order normalization * chore: refresh plugin sdk api baseline * fix: make shared string sorting deterministic * chore: refresh plugin sdk api baseline * fix: align host env security ordering
175 lines
5.5 KiB
TypeScript
175 lines
5.5 KiB
TypeScript
import { adaptScopedAccountAccessor } from "openclaw/plugin-sdk/channel-config-helpers";
|
|
import {
|
|
noteChannelLookupFailure,
|
|
noteChannelLookupSummary,
|
|
resolveEntriesWithOptionalToken,
|
|
createSetupTranslator,
|
|
type OpenClawConfig,
|
|
parseMentionOrPrefixedId,
|
|
promptLegacyChannelAllowFromForAccount,
|
|
type WizardPrompter,
|
|
} from "openclaw/plugin-sdk/setup-runtime";
|
|
import type {
|
|
ChannelSetupWizard,
|
|
ChannelSetupWizardAllowFromEntry,
|
|
} from "openclaw/plugin-sdk/setup-runtime";
|
|
import { formatDocsLink } from "openclaw/plugin-sdk/setup-tools";
|
|
import { normalizeStringEntries } from "openclaw/plugin-sdk/string-coerce-runtime";
|
|
import {
|
|
resolveDefaultSlackAccountId,
|
|
resolveSlackAccount,
|
|
resolveSlackAccountAllowFrom,
|
|
type ResolvedSlackAccount,
|
|
} from "./accounts.js";
|
|
import { resolveSlackChannelAllowlist } from "./resolve-channels.js";
|
|
import { resolveSlackUserAllowlist } from "./resolve-users.js";
|
|
import { createSlackSetupWizardBase } from "./setup-core.js";
|
|
import { SLACK_CHANNEL as channel } from "./shared.js";
|
|
|
|
const t = createSetupTranslator();
|
|
|
|
async function resolveSlackAllowFromEntries(params: {
|
|
token?: string;
|
|
entries: string[];
|
|
}): Promise<ChannelSetupWizardAllowFromEntry[]> {
|
|
return await resolveEntriesWithOptionalToken({
|
|
token: params.token,
|
|
entries: params.entries,
|
|
buildWithoutToken: (input) => ({
|
|
input,
|
|
resolved: false,
|
|
id: null,
|
|
}),
|
|
resolveEntries: async ({ token, entries }) =>
|
|
(
|
|
await resolveSlackUserAllowlist({
|
|
token,
|
|
entries,
|
|
})
|
|
).map((entry) => ({
|
|
input: entry.input,
|
|
resolved: entry.resolved,
|
|
id: entry.id ?? null,
|
|
})),
|
|
});
|
|
}
|
|
|
|
async function promptSlackAllowFrom(params: {
|
|
cfg: OpenClawConfig;
|
|
prompter: WizardPrompter;
|
|
accountId?: string;
|
|
}): Promise<OpenClawConfig> {
|
|
const parseId = (value: string) =>
|
|
parseMentionOrPrefixedId({
|
|
value,
|
|
mentionPattern: /^<@([A-Z0-9]+)>$/i,
|
|
prefixPattern: /^(slack:|user:)/i,
|
|
idPattern: /^[A-Z][A-Z0-9]+$/i,
|
|
normalizeId: (id) => id.toUpperCase(),
|
|
});
|
|
|
|
return await promptLegacyChannelAllowFromForAccount<ResolvedSlackAccount>({
|
|
cfg: params.cfg,
|
|
channel,
|
|
prompter: params.prompter,
|
|
accountId: params.accountId,
|
|
defaultAccountId: resolveDefaultSlackAccountId(params.cfg),
|
|
resolveAccount: adaptScopedAccountAccessor(resolveSlackAccount),
|
|
resolveExisting: (account, cfg) =>
|
|
resolveSlackAccountAllowFrom({ cfg, accountId: account.accountId }) ?? [],
|
|
resolveToken: (account) => account.userToken ?? account.botToken ?? "",
|
|
noteTitle: t("wizard.slack.allowlistTitle"),
|
|
noteLines: [
|
|
t("wizard.slack.allowlistIntro"),
|
|
t("wizard.slack.examples"),
|
|
"- U12345678",
|
|
"- @alice",
|
|
t("wizard.slack.multipleEntries"),
|
|
t("wizard.channels.docs", { link: formatDocsLink("/slack", "slack") }),
|
|
],
|
|
message: t("wizard.slack.allowFromPrompt"),
|
|
placeholder: "@alice, U12345678",
|
|
parseId,
|
|
invalidWithoutTokenNote: t("wizard.slack.allowFromInvalidWithoutToken"),
|
|
resolveEntries: async ({ token, entries }) =>
|
|
(
|
|
await resolveSlackUserAllowlist({
|
|
token,
|
|
entries,
|
|
})
|
|
).map((entry) => ({
|
|
input: entry.input,
|
|
resolved: entry.resolved,
|
|
id: entry.id ?? null,
|
|
})),
|
|
});
|
|
}
|
|
|
|
async function resolveSlackGroupAllowlist(params: {
|
|
cfg: OpenClawConfig;
|
|
accountId: string;
|
|
credentialValues: { botToken?: string };
|
|
entries: string[];
|
|
prompter: { note: (message: string, title?: string) => Promise<void> };
|
|
}) {
|
|
let keys = params.entries;
|
|
const accountWithTokens = resolveSlackAccount({
|
|
cfg: params.cfg,
|
|
accountId: params.accountId,
|
|
});
|
|
const activeBotToken = accountWithTokens.botToken || params.credentialValues.botToken || "";
|
|
if (params.entries.length > 0) {
|
|
try {
|
|
const resolved = await resolveEntriesWithOptionalToken<{
|
|
input: string;
|
|
resolved: boolean;
|
|
id?: string;
|
|
}>({
|
|
token: activeBotToken,
|
|
entries: params.entries,
|
|
buildWithoutToken: (input) => ({ input, resolved: false, id: undefined }),
|
|
resolveEntries: async ({ token, entries }) =>
|
|
await resolveSlackChannelAllowlist({
|
|
token,
|
|
entries,
|
|
}),
|
|
});
|
|
const resolvedKeys = resolved
|
|
.filter((entry) => entry.resolved && entry.id)
|
|
.map((entry) => entry.id as string);
|
|
const unresolved = resolved.filter((entry) => !entry.resolved).map((entry) => entry.input);
|
|
keys = [...resolvedKeys, ...normalizeStringEntries(unresolved)];
|
|
await noteChannelLookupSummary({
|
|
prompter: params.prompter,
|
|
label: t("wizard.slack.channelsLabel"),
|
|
resolvedSections: [{ title: t("wizard.channels.resolvedTitle"), values: resolvedKeys }],
|
|
unresolved,
|
|
});
|
|
} catch (error) {
|
|
await noteChannelLookupFailure({
|
|
prompter: params.prompter,
|
|
label: t("wizard.slack.channelsLabel"),
|
|
error,
|
|
});
|
|
}
|
|
}
|
|
return keys;
|
|
}
|
|
|
|
export const slackSetupWizard: ChannelSetupWizard = createSlackSetupWizardBase({
|
|
promptAllowFrom: promptSlackAllowFrom,
|
|
resolveAllowFromEntries: async ({ credentialValues, entries }) =>
|
|
await resolveSlackAllowFromEntries({
|
|
token: credentialValues.botToken,
|
|
entries,
|
|
}),
|
|
resolveGroupAllowlist: async ({ cfg, accountId, credentialValues, entries, prompter }) =>
|
|
await resolveSlackGroupAllowlist({
|
|
cfg,
|
|
accountId,
|
|
credentialValues,
|
|
entries,
|
|
prompter,
|
|
}),
|
|
});
|