import {
type ChannelSetupWizard,
type OpenClawConfig,
type WizardPrompter,
} from "openclaw/plugin-sdk/setup-runtime" ;
import { formatDocsLink } from "openclaw/plugin-sdk/setup-tools" ;
import { resolveDiscordChannelAllowlist } from "./resolve-channels.js" ;
import { resolveDiscordUserAllowlist } from "./resolve-users.js" ;
import {
resolveDefaultDiscordSetupAccountId,
resolveDiscordSetupAccountConfig,
} from "./setup-account-state.js" ;
import { createDiscordSetupWizardBase, parseDiscordAllowFromId } from "./setup-core.js" ;
import {
promptLegacyChannelAllowFromForAccount,
resolveEntriesWithOptionalToken,
} from "./setup-runtime-helpers.js" ;
import { resolveDiscordToken } from "./token.js" ;
const channel = "discord" as const ;
async function resolveDiscordAllowFromEntries(params: { token?: string; entries: string[] }) {
return await resolveEntriesWithOptionalToken({
token: params.token,
entries: params.entries,
buildWithoutToken: (input) => ({
input,
resolved: false ,
id: null ,
}),
resolveEntries: async ({ token, entries }) =>
(
await resolveDiscordUserAllowlist({
token,
entries,
})
).map((entry) => ({
input: entry.input,
resolved: entry.resolved,
id: entry.id ?? null ,
})),
});
}
async function promptDiscordAllowFrom(params: {
cfg: OpenClawConfig;
prompter: WizardPrompter;
accountId?: string;
}): Promise<OpenClawConfig> {
return await promptLegacyChannelAllowFromForAccount({
cfg: params.cfg,
channel,
prompter: params.prompter,
accountId: params.accountId,
defaultAccountId: resolveDefaultDiscordSetupAccountId(params.cfg),
resolveAccount: (cfg, accountId) => resolveDiscordSetupAccountConfig({ cfg, accountId }),
noteTitle: "Discord allowlist" ,
noteLines: [
"Allowlist Discord DMs by username (we resolve to user ids)." ,
"Examples:" ,
"- 123456789012345678" ,
"- @alice" ,
"- alice#1234" ,
"Multiple entries: comma-separated." ,
`Docs: ${formatDocsLink("/discord" , "discord" )}`,
],
message: "Discord allowFrom (usernames or ids)" ,
placeholder: "@alice, 123456789012345678" ,
parseId: parseDiscordAllowFromId,
invalidWithoutTokenNote: "Bot token missing; use numeric user ids (or mention form) only." ,
resolveExisting: (account) => {
const config = account.config;
return config.allowFrom ?? config.dm?.allowFrom ?? [];
},
resolveToken: (account) =>
resolveDiscordToken(params.cfg, { accountId: account.accountId }).token,
resolveEntries: async ({ token, entries }) =>
(
await resolveDiscordUserAllowlist({
token,
entries,
})
).map((entry) => ({
input: entry.input,
resolved: entry.resolved,
id: entry.id ?? null ,
})),
});
}
async function resolveDiscordGroupAllowlist(params: {
cfg: OpenClawConfig;
accountId: string;
credentialValues: { token?: string };
entries: string[];
}) {
return await resolveEntriesWithOptionalToken({
token:
resolveDiscordToken(params.cfg, { accountId: params.accountId }).token ||
(typeof params.credentialValues.token === "string" ? params.credentialValues.token : "" ),
entries: params.entries,
buildWithoutToken: (input) => ({
input,
resolved: false ,
}),
resolveEntries: async ({ token, entries }) =>
await resolveDiscordChannelAllowlist({
token,
entries,
}),
});
}
export const discordSetupWizard: ChannelSetupWizard = createDiscordSetupWizardBase({
promptAllowFrom: promptDiscordAllowFrom,
resolveAllowFromEntries: async ({ cfg, accountId, credentialValues, entries }) =>
await resolveDiscordAllowFromEntries({
token:
resolveDiscordToken(cfg, { accountId }).token ||
(typeof credentialValues.token === "string" ? credentialValues.token : "" ),
entries,
}),
resolveGroupAllowlist: async ({ cfg, accountId, credentialValues, entries }) =>
await resolveDiscordGroupAllowlist({
cfg,
accountId,
credentialValues,
entries,
}),
});
Messung V0.5 in Prozent C=100 H=100 G=100
¤ Dauer der Verarbeitung: 0.10 Sekunden
(vorverarbeitet am 2026-06-09)
¤
*© Formatika GbR, Deutschland