Files
openclaw/extensions/discord/src/monitor/message-handler.preflight.test.ts
Vincent Koc 9e389cff3d fix(config): migrate legacy group allow aliases (#60597)
* fix(config): migrate legacy group allow aliases

* fix(config): inline legacy streaming migration helpers

* refactor(config): rename legacy account matcher helper

* chore(agents): codify config contract boundaries

* fix(config): keep legacy allow aliases writable

* Update AGENTS.md
2026-04-04 11:15:32 +09:00

1130 lines
32 KiB
TypeScript

import { ChannelType } from "@buape/carbon";
import { beforeAll, beforeEach, describe, expect, it, vi } from "vitest";
const transcribeFirstAudioMock = vi.hoisted(() => vi.fn());
const resolveDiscordDmCommandAccessMock = vi.hoisted(() => vi.fn());
const handleDiscordDmCommandDecisionMock = vi.hoisted(() => vi.fn(async () => {}));
vi.mock("./preflight-audio.runtime.js", () => ({
transcribeFirstAudio: transcribeFirstAudioMock,
}));
vi.mock("./dm-command-auth.js", () => ({
resolveDiscordDmCommandAccess: resolveDiscordDmCommandAccessMock,
}));
vi.mock("./dm-command-decision.js", () => ({
handleDiscordDmCommandDecision: handleDiscordDmCommandDecisionMock,
}));
import {
__testing as sessionBindingTesting,
registerSessionBindingAdapter,
} from "openclaw/plugin-sdk/conversation-runtime";
import {
createDiscordMessage,
createDiscordPreflightArgs,
createGuildEvent,
createGuildTextClient,
DEFAULT_PREFLIGHT_CFG,
type DiscordClient,
type DiscordConfig,
type DiscordMessageEvent,
} from "./message-handler.preflight.test-helpers.js";
let preflightDiscordMessage: typeof import("./message-handler.preflight.js").preflightDiscordMessage;
let resolvePreflightMentionRequirement: typeof import("./message-handler.preflight.js").resolvePreflightMentionRequirement;
let shouldIgnoreBoundThreadWebhookMessage: typeof import("./message-handler.preflight.js").shouldIgnoreBoundThreadWebhookMessage;
let threadBindingTesting: typeof import("./thread-bindings.js").__testing;
let createThreadBindingManager: typeof import("./thread-bindings.js").createThreadBindingManager;
beforeAll(async () => {
({
preflightDiscordMessage,
resolvePreflightMentionRequirement,
shouldIgnoreBoundThreadWebhookMessage,
} = await import("./message-handler.preflight.js"));
({ __testing: threadBindingTesting, createThreadBindingManager } =
await import("./thread-bindings.js"));
});
function createThreadBinding(
overrides?: Partial<import("openclaw/plugin-sdk/conversation-runtime").SessionBindingRecord>,
) {
return {
bindingId: "default:thread-1",
targetSessionKey: "agent:main:subagent:child-1",
targetKind: "subagent",
conversation: {
channel: "discord",
accountId: "default",
conversationId: "thread-1",
parentConversationId: "parent-1",
},
status: "active",
boundAt: 1,
metadata: {
agentId: "main",
boundBy: "test",
webhookId: "wh-1",
webhookToken: "tok-1",
},
...overrides,
} satisfies import("openclaw/plugin-sdk/conversation-runtime").SessionBindingRecord;
}
function createPreflightArgs(params: {
cfg: import("openclaw/plugin-sdk/config-runtime").OpenClawConfig;
discordConfig: DiscordConfig;
data: DiscordMessageEvent;
client: DiscordClient;
}): Parameters<typeof preflightDiscordMessage>[0] {
return createDiscordPreflightArgs(params);
}
function createThreadClient(params: { threadId: string; parentId: string }): DiscordClient {
return {
fetchChannel: async (channelId: string) => {
if (channelId === params.threadId) {
return {
id: params.threadId,
type: ChannelType.PublicThread,
name: "focus",
parentId: params.parentId,
ownerId: "owner-1",
};
}
if (channelId === params.parentId) {
return {
id: params.parentId,
type: ChannelType.GuildText,
name: "general",
};
}
return null;
},
} as unknown as DiscordClient;
}
function createDmClient(channelId: string): DiscordClient {
return {
fetchChannel: async (id: string) => {
if (id === channelId) {
return {
id: channelId,
type: ChannelType.DM,
};
}
return null;
},
} as unknown as DiscordClient;
}
async function runThreadBoundPreflight(params: {
threadId: string;
parentId: string;
message: import("@buape/carbon").Message;
threadBinding: import("openclaw/plugin-sdk/conversation-runtime").SessionBindingRecord;
discordConfig: DiscordConfig;
registerBindingAdapter?: boolean;
}) {
if (params.registerBindingAdapter) {
registerSessionBindingAdapter({
channel: "discord",
accountId: "default",
listBySession: () => [],
resolveByConversation: (ref) =>
ref.conversationId === params.threadId ? params.threadBinding : null,
});
}
const client = createThreadClient({
threadId: params.threadId,
parentId: params.parentId,
});
return preflightDiscordMessage({
...createPreflightArgs({
cfg: DEFAULT_PREFLIGHT_CFG,
discordConfig: params.discordConfig,
data: createGuildEvent({
channelId: params.threadId,
guildId: "guild-1",
author: params.message.author,
message: params.message,
}),
client,
}),
threadBindings: {
getByThreadId: (id: string) => (id === params.threadId ? params.threadBinding : undefined),
} as import("./thread-bindings.js").ThreadBindingManager,
});
}
async function runGuildPreflight(params: {
channelId: string;
guildId: string;
message: import("@buape/carbon").Message;
discordConfig: DiscordConfig;
cfg?: import("openclaw/plugin-sdk/config-runtime").OpenClawConfig;
guildEntries?: Parameters<typeof preflightDiscordMessage>[0]["guildEntries"];
includeGuildObject?: boolean;
}) {
return preflightDiscordMessage({
...createPreflightArgs({
cfg: params.cfg ?? DEFAULT_PREFLIGHT_CFG,
discordConfig: params.discordConfig,
data: createGuildEvent({
channelId: params.channelId,
guildId: params.guildId,
author: params.message.author,
message: params.message,
includeGuildObject: params.includeGuildObject,
}),
client: createGuildTextClient(params.channelId),
}),
guildEntries: params.guildEntries,
});
}
async function runDmPreflight(params: {
channelId: string;
message: import("@buape/carbon").Message;
discordConfig: DiscordConfig;
}) {
return preflightDiscordMessage({
...createPreflightArgs({
cfg: DEFAULT_PREFLIGHT_CFG,
discordConfig: params.discordConfig,
data: {
channel_id: params.channelId,
author: params.message.author,
message: params.message,
} as DiscordMessageEvent,
client: createDmClient(params.channelId),
}),
});
}
async function runMentionOnlyBotPreflight(params: {
channelId: string;
guildId: string;
message: import("@buape/carbon").Message;
}) {
return runGuildPreflight({
channelId: params.channelId,
guildId: params.guildId,
message: params.message,
discordConfig: {
allowBots: "mentions",
} as DiscordConfig,
});
}
async function runIgnoreOtherMentionsPreflight(params: {
channelId: string;
guildId: string;
message: import("@buape/carbon").Message;
}) {
return runGuildPreflight({
channelId: params.channelId,
guildId: params.guildId,
message: params.message,
discordConfig: {} as DiscordConfig,
guildEntries: {
[params.guildId]: {
requireMention: false,
ignoreOtherMentions: true,
},
},
});
}
describe("resolvePreflightMentionRequirement", () => {
it("requires mention when config requires mention and thread is not bound", () => {
expect(
resolvePreflightMentionRequirement({
shouldRequireMention: true,
bypassMentionRequirement: false,
}),
).toBe(true);
});
it("disables mention requirement when the route explicitly bypasses mentions", () => {
expect(
resolvePreflightMentionRequirement({
shouldRequireMention: true,
bypassMentionRequirement: true,
}),
).toBe(false);
});
it("keeps mention requirement disabled when config already disables it", () => {
expect(
resolvePreflightMentionRequirement({
shouldRequireMention: false,
bypassMentionRequirement: false,
}),
).toBe(false);
});
});
describe("preflightDiscordMessage", () => {
beforeEach(() => {
sessionBindingTesting.resetSessionBindingAdaptersForTests();
transcribeFirstAudioMock.mockReset();
resolveDiscordDmCommandAccessMock.mockReset();
resolveDiscordDmCommandAccessMock.mockResolvedValue({
commandAuthorized: true,
decision: "allow",
allowMatch: { allowed: true, matchedBy: "allowFrom", value: "123" },
});
handleDiscordDmCommandDecisionMock.mockReset();
handleDiscordDmCommandDecisionMock.mockResolvedValue(undefined);
});
it("drops bound-thread bot system messages to prevent ACP self-loop", async () => {
const threadBinding = createThreadBinding({
targetKind: "session",
targetSessionKey: "agent:main:acp:discord-thread-1",
});
const threadId = "thread-system-1";
const parentId = "channel-parent-1";
const message = createDiscordMessage({
id: "m-system-1",
channelId: threadId,
content:
"⚙️ codex-acp session active (auto-unfocus in 24h). Messages here go directly to this session.",
author: {
id: "relay-bot-1",
bot: true,
username: "OpenClaw",
},
});
const result = await runThreadBoundPreflight({
threadId,
parentId,
message,
threadBinding,
discordConfig: {
allowBots: true,
} as DiscordConfig,
});
expect(result).toBeNull();
});
it("restores direct-message bindings by user target instead of DM channel id", async () => {
registerSessionBindingAdapter({
channel: "discord",
accountId: "default",
listBySession: () => [],
resolveByConversation: (ref) =>
ref.conversationId === "user:user-1"
? createThreadBinding({
conversation: {
channel: "discord",
accountId: "default",
conversationId: "user:user-1",
},
metadata: {
pluginBindingOwner: "plugin",
pluginId: "openclaw-codex-app-server",
pluginRoot: "/Users/huntharo/github/openclaw-app-server",
},
})
: null,
});
const result = await runDmPreflight({
channelId: "dm-channel-1",
message: createDiscordMessage({
id: "m-dm-1",
channelId: "dm-channel-1",
content: "who are you",
author: {
id: "user-1",
bot: false,
username: "alice",
},
}),
discordConfig: {
allowBots: true,
dmPolicy: "open",
} as DiscordConfig,
});
expect(result).not.toBeNull();
expect(result?.threadBinding).toMatchObject({
conversation: {
channel: "discord",
accountId: "default",
conversationId: "user:user-1",
},
metadata: {
pluginBindingOwner: "plugin",
pluginId: "openclaw-codex-app-server",
},
});
});
it("falls back to the default discord account for omitted-account dm authorization", async () => {
const message = createDiscordMessage({
id: "m-dm-default-account",
channelId: "dm-channel-default-account",
content: "who are you",
author: {
id: "user-1",
bot: false,
username: "alice",
},
});
await preflightDiscordMessage({
...createPreflightArgs({
cfg: {
...DEFAULT_PREFLIGHT_CFG,
channels: {
discord: {
defaultAccount: "work",
accounts: {
default: {
token: "token-default",
},
work: {
token: "token-work",
},
},
},
},
},
discordConfig: {
defaultAccount: "work",
dmPolicy: "allowlist",
} as DiscordConfig,
data: {
channel_id: "dm-channel-default-account",
author: message.author,
message,
} as DiscordMessageEvent,
client: createDmClient("dm-channel-default-account"),
}),
});
expect(resolveDiscordDmCommandAccessMock).toHaveBeenCalledWith(
expect.objectContaining({
accountId: "default",
}),
);
});
it("keeps bound-thread regular bot messages flowing when allowBots=true", async () => {
const threadBinding = createThreadBinding({
targetKind: "session",
targetSessionKey: "agent:main:acp:discord-thread-1",
});
const threadId = "thread-bot-regular-1";
const parentId = "channel-parent-regular-1";
const message = createDiscordMessage({
id: "m-bot-regular-1",
channelId: threadId,
content: "here is tool output chunk",
author: {
id: "relay-bot-1",
bot: true,
username: "Relay",
},
});
const result = await runThreadBoundPreflight({
threadId,
parentId,
message,
threadBinding,
discordConfig: {
allowBots: true,
} as DiscordConfig,
registerBindingAdapter: true,
});
expect(result).not.toBeNull();
expect(result?.boundSessionKey).toBe(threadBinding.targetSessionKey);
});
it("drops hydrated bound-thread webhook echoes after fetching an empty payload", async () => {
const threadBinding = createThreadBinding({
targetKind: "session",
targetSessionKey: "agent:main:acp:discord-thread-1",
});
const threadId = "thread-webhook-hydrated-1";
const parentId = "channel-parent-webhook-hydrated-1";
const message = createDiscordMessage({
id: "m-webhook-hydrated-1",
channelId: threadId,
content: "",
author: {
id: "relay-bot-1",
bot: true,
username: "Relay",
},
});
const restGet = vi.fn(async () => ({
id: message.id,
content: "webhook relay",
webhook_id: "wh-1",
attachments: [],
embeds: [],
mentions: [],
mention_roles: [],
mention_everyone: false,
author: {
id: "relay-bot-1",
username: "Relay",
bot: true,
},
}));
const client = {
...createThreadClient({ threadId, parentId }),
rest: {
get: restGet,
},
} as unknown as DiscordClient;
const result = await preflightDiscordMessage({
...createPreflightArgs({
cfg: DEFAULT_PREFLIGHT_CFG,
discordConfig: {
allowBots: true,
} as DiscordConfig,
data: createGuildEvent({
channelId: threadId,
guildId: "guild-1",
author: message.author,
message,
}),
client,
}),
threadBindings: {
getByThreadId: (id: string) => (id === threadId ? threadBinding : undefined),
} as import("./thread-bindings.js").ThreadBindingManager,
});
expect(restGet).toHaveBeenCalledTimes(1);
expect(result).toBeNull();
});
it("bypasses mention gating in bound threads for allowed bot senders", async () => {
const threadBinding = createThreadBinding();
const threadId = "thread-bot-focus";
const parentId = "channel-parent-focus";
const client = createThreadClient({ threadId, parentId });
const message = createDiscordMessage({
id: "m-bot-1",
channelId: threadId,
content: "relay message without mention",
author: {
id: "relay-bot-1",
bot: true,
username: "Relay",
},
});
registerSessionBindingAdapter({
channel: "discord",
accountId: "default",
listBySession: () => [],
resolveByConversation: (ref) => (ref.conversationId === threadId ? threadBinding : null),
});
const result = await preflightDiscordMessage(
createPreflightArgs({
cfg: {
...DEFAULT_PREFLIGHT_CFG,
} as import("openclaw/plugin-sdk/config-runtime").OpenClawConfig,
discordConfig: {
allowBots: true,
} as DiscordConfig,
data: createGuildEvent({
channelId: threadId,
guildId: "guild-1",
author: message.author,
message,
}),
client,
}),
);
expect(result).not.toBeNull();
expect(result?.boundSessionKey).toBe(threadBinding.targetSessionKey);
expect(result?.shouldRequireMention).toBe(false);
});
it("drops bot messages without mention when allowBots=mentions", async () => {
const channelId = "channel-bot-mentions-off";
const guildId = "guild-bot-mentions-off";
const message = createDiscordMessage({
id: "m-bot-mentions-off",
channelId,
content: "relay chatter",
author: {
id: "relay-bot-1",
bot: true,
username: "Relay",
},
});
const result = await runMentionOnlyBotPreflight({ channelId, guildId, message });
expect(result).toBeNull();
});
it("allows bot messages with explicit mention when allowBots=mentions", async () => {
const channelId = "channel-bot-mentions-on";
const guildId = "guild-bot-mentions-on";
const message = createDiscordMessage({
id: "m-bot-mentions-on",
channelId,
content: "hi <@openclaw-bot>",
mentionedUsers: [{ id: "openclaw-bot" }],
author: {
id: "relay-bot-1",
bot: true,
username: "Relay",
},
});
const result = await runMentionOnlyBotPreflight({ channelId, guildId, message });
expect(result).not.toBeNull();
});
it("treats @everyone as a mention when requireMention is true", async () => {
const channelId = "channel-everyone-mention";
const guildId = "guild-everyone-mention";
const message = createDiscordMessage({
id: "m-everyone-mention",
channelId,
content: "@everyone standup time!",
mentionedEveryone: true,
author: {
id: "user-1",
bot: false,
username: "Peter",
},
});
const result = await runGuildPreflight({
channelId,
guildId,
message,
discordConfig: {
botId: "openclaw-bot",
} as DiscordConfig,
guildEntries: {
[guildId]: {
channels: {
[channelId]: {
enabled: true,
requireMention: true,
},
},
},
},
});
expect(result).not.toBeNull();
expect(result?.shouldRequireMention).toBe(true);
expect(result?.wasMentioned).toBe(true);
});
it("accepts allowlisted guild messages when guild object is missing", async () => {
const message = createDiscordMessage({
id: "m-guild-id-only",
channelId: "ch-1",
content: "hello from maintainers",
author: {
id: "user-1",
bot: false,
username: "Peter",
},
});
const result = await runGuildPreflight({
channelId: "ch-1",
guildId: "guild-1",
message,
discordConfig: {} as DiscordConfig,
guildEntries: {
"guild-1": {
channels: {
"ch-1": {
enabled: true,
requireMention: false,
},
},
},
},
includeGuildObject: false,
});
expect(result).not.toBeNull();
expect(result?.guildInfo?.id).toBe("guild-1");
expect(result?.channelConfig?.allowed).toBe(true);
expect(result?.shouldRequireMention).toBe(false);
});
it("inherits parent thread allowlist when guild object is missing", async () => {
const threadId = "thread-1";
const parentId = "parent-1";
const message = createDiscordMessage({
id: "m-thread-id-only",
channelId: threadId,
content: "thread hello",
author: {
id: "user-1",
bot: false,
username: "Peter",
},
});
const result = await preflightDiscordMessage({
...createPreflightArgs({
cfg: DEFAULT_PREFLIGHT_CFG,
discordConfig: {} as DiscordConfig,
data: createGuildEvent({
channelId: threadId,
guildId: "guild-1",
author: message.author,
message,
includeGuildObject: false,
}),
client: createThreadClient({
threadId,
parentId,
}),
}),
guildEntries: {
"guild-1": {
channels: {
[parentId]: {
enabled: true,
requireMention: false,
},
},
},
},
});
expect(result).not.toBeNull();
expect(result?.guildInfo?.id).toBe("guild-1");
expect(result?.threadParentId).toBe(parentId);
expect(result?.channelConfig?.allowed).toBe(true);
expect(result?.shouldRequireMention).toBe(false);
});
it("drops guild messages that mention another user when ignoreOtherMentions=true", async () => {
const channelId = "channel-other-mention-1";
const guildId = "guild-other-mention-1";
const message = createDiscordMessage({
id: "m-other-mention-1",
channelId,
content: "hello <@999>",
mentionedUsers: [{ id: "999" }],
author: {
id: "user-1",
bot: false,
username: "Alice",
},
});
const result = await runIgnoreOtherMentionsPreflight({ channelId, guildId, message });
expect(result).toBeNull();
});
it("does not drop @everyone messages when ignoreOtherMentions=true", async () => {
const channelId = "channel-other-mention-everyone";
const guildId = "guild-other-mention-everyone";
const message = createDiscordMessage({
id: "m-other-mention-everyone",
channelId,
content: "@everyone heads up",
mentionedEveryone: true,
author: {
id: "user-1",
bot: false,
username: "Alice",
},
});
const result = await runIgnoreOtherMentionsPreflight({ channelId, guildId, message });
expect(result).not.toBeNull();
expect(result?.hasAnyMention).toBe(true);
});
it("ignores bot-sent @everyone mentions for detection", async () => {
const channelId = "channel-everyone-1";
const guildId = "guild-everyone-1";
const client = createGuildTextClient(channelId);
const message = createDiscordMessage({
id: "m-everyone-1",
channelId,
content: "@everyone heads up",
mentionedEveryone: true,
author: {
id: "relay-bot-1",
bot: true,
username: "Relay",
},
});
const result = await preflightDiscordMessage({
...createPreflightArgs({
cfg: DEFAULT_PREFLIGHT_CFG,
discordConfig: {
allowBots: true,
} as DiscordConfig,
data: createGuildEvent({
channelId,
guildId,
author: message.author,
message,
}),
client,
}),
guildEntries: {
[guildId]: {
requireMention: false,
},
},
});
expect(result).not.toBeNull();
expect(result?.hasAnyMention).toBe(false);
});
it("does not treat bot-sent @everyone as wasMentioned", async () => {
const channelId = "channel-everyone-2";
const guildId = "guild-everyone-2";
const client = createGuildTextClient(channelId);
const message = createDiscordMessage({
id: "m-everyone-2",
channelId,
content: "@everyone relay message",
mentionedEveryone: true,
author: {
id: "relay-bot-2",
bot: true,
username: "RelayBot",
},
});
const result = await preflightDiscordMessage({
...createPreflightArgs({
cfg: DEFAULT_PREFLIGHT_CFG,
discordConfig: {
allowBots: true,
} as DiscordConfig,
data: createGuildEvent({
channelId,
guildId,
author: message.author,
message,
}),
client,
}),
guildEntries: {
[guildId]: {
requireMention: false,
},
},
});
expect(result).not.toBeNull();
expect(result?.wasMentioned).toBe(false);
});
it("uses attachment content_type for guild audio preflight mention detection", async () => {
transcribeFirstAudioMock.mockResolvedValue("hey openclaw");
const channelId = "channel-audio-1";
const client = createGuildTextClient(channelId);
const message = createDiscordMessage({
id: "m-audio-1",
channelId,
content: "",
attachments: [
{
id: "att-1",
url: "https://cdn.discordapp.com/attachments/voice.ogg",
content_type: "audio/ogg",
filename: "voice.ogg",
},
],
author: {
id: "user-1",
bot: false,
username: "Alice",
},
});
const result = await preflightDiscordMessage({
...createPreflightArgs({
cfg: {
...DEFAULT_PREFLIGHT_CFG,
messages: {
groupChat: {
mentionPatterns: ["openclaw"],
},
},
} as import("openclaw/plugin-sdk/config-runtime").OpenClawConfig,
discordConfig: {} as DiscordConfig,
data: createGuildEvent({
channelId,
guildId: "guild-1",
author: message.author,
message,
}),
client,
}),
guildEntries: {
"guild-1": {
channels: {
[channelId]: {
enabled: true,
requireMention: true,
},
},
},
},
});
expect(transcribeFirstAudioMock).toHaveBeenCalledTimes(1);
expect(transcribeFirstAudioMock).toHaveBeenCalledWith(
expect.objectContaining({
ctx: expect.objectContaining({
MediaUrls: ["https://cdn.discordapp.com/attachments/voice.ogg"],
MediaTypes: ["audio/ogg"],
}),
}),
);
expect(result).not.toBeNull();
expect(result?.wasMentioned).toBe(true);
});
it("does not transcribe guild audio from unauthorized members", async () => {
const channelId = "channel-audio-unauthorized-1";
const guildId = "guild-audio-unauthorized-1";
const client = createGuildTextClient(channelId);
const message = createDiscordMessage({
id: "m-audio-unauthorized-1",
channelId,
content: "",
attachments: [
{
id: "att-1",
url: "https://cdn.discordapp.com/attachments/voice.ogg",
content_type: "audio/ogg",
filename: "voice.ogg",
},
],
author: {
id: "user-2",
bot: false,
username: "Mallory",
},
});
const result = await preflightDiscordMessage({
...createPreflightArgs({
cfg: {
...DEFAULT_PREFLIGHT_CFG,
messages: {
groupChat: {
mentionPatterns: ["openclaw"],
},
},
} as import("openclaw/plugin-sdk/config-runtime").OpenClawConfig,
discordConfig: {} as DiscordConfig,
data: createGuildEvent({
channelId,
guildId,
author: message.author,
message,
}),
client,
}),
guildEntries: {
[guildId]: {
channels: {
[channelId]: {
enabled: true,
requireMention: true,
users: ["user-1"],
},
},
},
},
});
expect(transcribeFirstAudioMock).not.toHaveBeenCalled();
expect(result).toBeNull();
});
it("drops guild message without mention when channel has configuredBinding and requireMention: true", async () => {
const conversationRuntime = await import("openclaw/plugin-sdk/conversation-runtime");
const channelId = "ch-binding-1";
const bindingRoute = {
bindingResolution: {
record: {
targetSessionKey: "agent:main:acp:binding:discord:default:abc",
targetKind: "session",
},
} as never,
route: { agentId: "main", matchedBy: "binding.channel" } as never,
boundSessionKey: "agent:main:acp:binding:discord:default:abc",
boundAgentId: "main",
};
const routeSpy = vi
.spyOn(conversationRuntime, "resolveConfiguredBindingRoute")
.mockReturnValue(bindingRoute);
const ensureSpy = vi
.spyOn(conversationRuntime, "ensureConfiguredBindingRouteReady")
.mockResolvedValue({ ok: true });
try {
const result = await runGuildPreflight({
channelId,
guildId: "guild-1",
message: createDiscordMessage({
id: "m-binding-1",
channelId,
content: "hello without mention",
author: { id: "user-1", bot: false, username: "alice" },
}),
discordConfig: {} as DiscordConfig,
guildEntries: {
"guild-1": { channels: { [channelId]: { enabled: true, requireMention: true } } },
},
});
expect(result).toBeNull();
} finally {
routeSpy.mockRestore();
ensureSpy.mockRestore();
}
});
it("allows guild message with mention when channel has configuredBinding and requireMention: true", async () => {
const conversationRuntime = await import("openclaw/plugin-sdk/conversation-runtime");
const channelId = "ch-binding-2";
const bindingRoute = {
bindingResolution: {
record: {
targetSessionKey: "agent:main:acp:binding:discord:default:def",
targetKind: "session",
},
} as never,
route: { agentId: "main", matchedBy: "binding.channel" } as never,
boundSessionKey: "agent:main:acp:binding:discord:default:def",
boundAgentId: "main",
};
const routeSpy = vi
.spyOn(conversationRuntime, "resolveConfiguredBindingRoute")
.mockReturnValue(bindingRoute);
const ensureSpy = vi
.spyOn(conversationRuntime, "ensureConfiguredBindingRouteReady")
.mockResolvedValue({ ok: true });
try {
const result = await runGuildPreflight({
channelId,
guildId: "guild-1",
message: createDiscordMessage({
id: "m-binding-2",
channelId,
content: "hello <@openclaw-bot>",
author: { id: "user-1", bot: false, username: "alice" },
mentionedUsers: [{ id: "openclaw-bot" }],
}),
discordConfig: {} as DiscordConfig,
guildEntries: {
"guild-1": { channels: { [channelId]: { enabled: true, requireMention: true } } },
},
});
expect(result).not.toBeNull();
} finally {
routeSpy.mockRestore();
ensureSpy.mockRestore();
}
});
});
describe("shouldIgnoreBoundThreadWebhookMessage", () => {
beforeEach(() => {
sessionBindingTesting.resetSessionBindingAdaptersForTests();
threadBindingTesting.resetThreadBindingsForTests();
});
it("returns true when inbound webhook id matches the bound thread webhook", () => {
expect(
shouldIgnoreBoundThreadWebhookMessage({
webhookId: "wh-1",
threadBinding: createThreadBinding(),
}),
).toBe(true);
});
it("returns false when webhook ids differ", () => {
expect(
shouldIgnoreBoundThreadWebhookMessage({
webhookId: "wh-other",
threadBinding: createThreadBinding(),
}),
).toBe(false);
});
it("returns false when there is no bound thread webhook", () => {
expect(
shouldIgnoreBoundThreadWebhookMessage({
webhookId: "wh-1",
threadBinding: createThreadBinding({
metadata: {
webhookId: undefined,
},
}),
}),
).toBe(false);
});
it("returns true for recently unbound thread webhook echoes", async () => {
const manager = createThreadBindingManager({
accountId: "default",
persist: false,
enableSweeper: false,
});
const binding = await manager.bindTarget({
threadId: "thread-1",
channelId: "parent-1",
targetKind: "subagent",
targetSessionKey: "agent:main:subagent:child-1",
agentId: "main",
webhookId: "wh-1",
webhookToken: "tok-1",
});
expect(binding).not.toBeNull();
manager.unbindThread({
threadId: "thread-1",
sendFarewell: false,
});
expect(
shouldIgnoreBoundThreadWebhookMessage({
accountId: "default",
threadId: "thread-1",
webhookId: "wh-1",
}),
).toBe(true);
});
});