362 lines
10 KiB
TypeScript
362 lines
10 KiB
TypeScript
import { describe, expect, it, vi } from "vitest";
|
|
|
|
import type { ClawdbotConfig } from "../../config/config.js";
|
|
import { handleSlackAction } from "./slack-actions.js";
|
|
|
|
const deleteSlackMessage = vi.fn(async () => ({}));
|
|
const editSlackMessage = vi.fn(async () => ({}));
|
|
const getSlackMemberInfo = vi.fn(async () => ({}));
|
|
const listSlackEmojis = vi.fn(async () => ({}));
|
|
const listSlackPins = vi.fn(async () => ({}));
|
|
const listSlackReactions = vi.fn(async () => ({}));
|
|
const pinSlackMessage = vi.fn(async () => ({}));
|
|
const reactSlackMessage = vi.fn(async () => ({}));
|
|
const readSlackMessages = vi.fn(async () => ({}));
|
|
const removeOwnSlackReactions = vi.fn(async () => ["thumbsup"]);
|
|
const removeSlackReaction = vi.fn(async () => ({}));
|
|
const sendSlackMessage = vi.fn(async () => ({}));
|
|
const unpinSlackMessage = vi.fn(async () => ({}));
|
|
|
|
vi.mock("../../slack/actions.js", () => ({
|
|
deleteSlackMessage: (...args: unknown[]) => deleteSlackMessage(...args),
|
|
editSlackMessage: (...args: unknown[]) => editSlackMessage(...args),
|
|
getSlackMemberInfo: (...args: unknown[]) => getSlackMemberInfo(...args),
|
|
listSlackEmojis: (...args: unknown[]) => listSlackEmojis(...args),
|
|
listSlackPins: (...args: unknown[]) => listSlackPins(...args),
|
|
listSlackReactions: (...args: unknown[]) => listSlackReactions(...args),
|
|
pinSlackMessage: (...args: unknown[]) => pinSlackMessage(...args),
|
|
reactSlackMessage: (...args: unknown[]) => reactSlackMessage(...args),
|
|
readSlackMessages: (...args: unknown[]) => readSlackMessages(...args),
|
|
removeOwnSlackReactions: (...args: unknown[]) =>
|
|
removeOwnSlackReactions(...args),
|
|
removeSlackReaction: (...args: unknown[]) => removeSlackReaction(...args),
|
|
sendSlackMessage: (...args: unknown[]) => sendSlackMessage(...args),
|
|
unpinSlackMessage: (...args: unknown[]) => unpinSlackMessage(...args),
|
|
}));
|
|
|
|
describe("handleSlackAction", () => {
|
|
it("adds reactions", async () => {
|
|
const cfg = { slack: { botToken: "tok" } } as ClawdbotConfig;
|
|
await handleSlackAction(
|
|
{
|
|
action: "react",
|
|
channelId: "C1",
|
|
messageId: "123.456",
|
|
emoji: "✅",
|
|
},
|
|
cfg,
|
|
);
|
|
expect(reactSlackMessage).toHaveBeenCalledWith("C1", "123.456", "✅");
|
|
});
|
|
|
|
it("removes reactions on empty emoji", async () => {
|
|
const cfg = { slack: { botToken: "tok" } } as ClawdbotConfig;
|
|
await handleSlackAction(
|
|
{
|
|
action: "react",
|
|
channelId: "C1",
|
|
messageId: "123.456",
|
|
emoji: "",
|
|
},
|
|
cfg,
|
|
);
|
|
expect(removeOwnSlackReactions).toHaveBeenCalledWith("C1", "123.456");
|
|
});
|
|
|
|
it("removes reactions when remove flag set", async () => {
|
|
const cfg = { slack: { botToken: "tok" } } as ClawdbotConfig;
|
|
await handleSlackAction(
|
|
{
|
|
action: "react",
|
|
channelId: "C1",
|
|
messageId: "123.456",
|
|
emoji: "✅",
|
|
remove: true,
|
|
},
|
|
cfg,
|
|
);
|
|
expect(removeSlackReaction).toHaveBeenCalledWith("C1", "123.456", "✅");
|
|
});
|
|
|
|
it("rejects removes without emoji", async () => {
|
|
const cfg = { slack: { botToken: "tok" } } as ClawdbotConfig;
|
|
await expect(
|
|
handleSlackAction(
|
|
{
|
|
action: "react",
|
|
channelId: "C1",
|
|
messageId: "123.456",
|
|
emoji: "",
|
|
remove: true,
|
|
},
|
|
cfg,
|
|
),
|
|
).rejects.toThrow(/Emoji is required/);
|
|
});
|
|
|
|
it("respects reaction gating", async () => {
|
|
const cfg = {
|
|
slack: { botToken: "tok", actions: { reactions: false } },
|
|
} as ClawdbotConfig;
|
|
await expect(
|
|
handleSlackAction(
|
|
{
|
|
action: "react",
|
|
channelId: "C1",
|
|
messageId: "123.456",
|
|
emoji: "✅",
|
|
},
|
|
cfg,
|
|
),
|
|
).rejects.toThrow(/Slack reactions are disabled/);
|
|
});
|
|
|
|
it("passes threadTs to sendSlackMessage for thread replies", async () => {
|
|
const cfg = { slack: { botToken: "tok" } } as ClawdbotConfig;
|
|
await handleSlackAction(
|
|
{
|
|
action: "sendMessage",
|
|
to: "channel:C123",
|
|
content: "Hello thread",
|
|
threadTs: "1234567890.123456",
|
|
},
|
|
cfg,
|
|
);
|
|
expect(sendSlackMessage).toHaveBeenCalledWith(
|
|
"channel:C123",
|
|
"Hello thread",
|
|
{
|
|
mediaUrl: undefined,
|
|
threadTs: "1234567890.123456",
|
|
},
|
|
);
|
|
});
|
|
|
|
it("auto-injects threadTs from context when replyToMode=all", async () => {
|
|
const cfg = { slack: { botToken: "tok" } } as ClawdbotConfig;
|
|
sendSlackMessage.mockClear();
|
|
await handleSlackAction(
|
|
{
|
|
action: "sendMessage",
|
|
to: "channel:C123",
|
|
content: "Auto-threaded",
|
|
},
|
|
cfg,
|
|
{
|
|
currentChannelId: "C123",
|
|
currentThreadTs: "1111111111.111111",
|
|
replyToMode: "all",
|
|
},
|
|
);
|
|
expect(sendSlackMessage).toHaveBeenCalledWith(
|
|
"channel:C123",
|
|
"Auto-threaded",
|
|
{
|
|
mediaUrl: undefined,
|
|
threadTs: "1111111111.111111",
|
|
},
|
|
);
|
|
});
|
|
|
|
it("replyToMode=first threads first message then stops", async () => {
|
|
const cfg = { slack: { botToken: "tok" } } as ClawdbotConfig;
|
|
sendSlackMessage.mockClear();
|
|
const hasRepliedRef = { value: false };
|
|
const context = {
|
|
currentChannelId: "C123",
|
|
currentThreadTs: "1111111111.111111",
|
|
replyToMode: "first" as const,
|
|
hasRepliedRef,
|
|
};
|
|
|
|
// First message should be threaded
|
|
await handleSlackAction(
|
|
{ action: "sendMessage", to: "channel:C123", content: "First" },
|
|
cfg,
|
|
context,
|
|
);
|
|
expect(sendSlackMessage).toHaveBeenLastCalledWith("channel:C123", "First", {
|
|
mediaUrl: undefined,
|
|
threadTs: "1111111111.111111",
|
|
});
|
|
expect(hasRepliedRef.value).toBe(true);
|
|
|
|
// Second message should NOT be threaded
|
|
await handleSlackAction(
|
|
{ action: "sendMessage", to: "channel:C123", content: "Second" },
|
|
cfg,
|
|
context,
|
|
);
|
|
expect(sendSlackMessage).toHaveBeenLastCalledWith(
|
|
"channel:C123",
|
|
"Second",
|
|
{
|
|
mediaUrl: undefined,
|
|
threadTs: undefined,
|
|
},
|
|
);
|
|
});
|
|
|
|
it("replyToMode=first marks hasRepliedRef even when threadTs is explicit", async () => {
|
|
const cfg = { slack: { botToken: "tok" } } as ClawdbotConfig;
|
|
sendSlackMessage.mockClear();
|
|
const hasRepliedRef = { value: false };
|
|
const context = {
|
|
currentChannelId: "C123",
|
|
currentThreadTs: "1111111111.111111",
|
|
replyToMode: "first" as const,
|
|
hasRepliedRef,
|
|
};
|
|
|
|
await handleSlackAction(
|
|
{
|
|
action: "sendMessage",
|
|
to: "channel:C123",
|
|
content: "Explicit",
|
|
threadTs: "2222222222.222222",
|
|
},
|
|
cfg,
|
|
context,
|
|
);
|
|
expect(sendSlackMessage).toHaveBeenLastCalledWith(
|
|
"channel:C123",
|
|
"Explicit",
|
|
{
|
|
mediaUrl: undefined,
|
|
threadTs: "2222222222.222222",
|
|
},
|
|
);
|
|
expect(hasRepliedRef.value).toBe(true);
|
|
|
|
await handleSlackAction(
|
|
{ action: "sendMessage", to: "channel:C123", content: "Second" },
|
|
cfg,
|
|
context,
|
|
);
|
|
expect(sendSlackMessage).toHaveBeenLastCalledWith(
|
|
"channel:C123",
|
|
"Second",
|
|
{
|
|
mediaUrl: undefined,
|
|
threadTs: undefined,
|
|
},
|
|
);
|
|
});
|
|
|
|
it("replyToMode=first without hasRepliedRef does not thread", async () => {
|
|
const cfg = { slack: { botToken: "tok" } } as ClawdbotConfig;
|
|
sendSlackMessage.mockClear();
|
|
await handleSlackAction(
|
|
{ action: "sendMessage", to: "channel:C123", content: "No ref" },
|
|
cfg,
|
|
{
|
|
currentChannelId: "C123",
|
|
currentThreadTs: "1111111111.111111",
|
|
replyToMode: "first",
|
|
// no hasRepliedRef
|
|
},
|
|
);
|
|
expect(sendSlackMessage).toHaveBeenCalledWith("channel:C123", "No ref", {
|
|
mediaUrl: undefined,
|
|
threadTs: undefined,
|
|
});
|
|
});
|
|
|
|
it("does not auto-inject threadTs when replyToMode=off", async () => {
|
|
const cfg = { slack: { botToken: "tok" } } as ClawdbotConfig;
|
|
sendSlackMessage.mockClear();
|
|
await handleSlackAction(
|
|
{
|
|
action: "sendMessage",
|
|
to: "channel:C123",
|
|
content: "Off mode",
|
|
},
|
|
cfg,
|
|
{
|
|
currentChannelId: "C123",
|
|
currentThreadTs: "1111111111.111111",
|
|
replyToMode: "off",
|
|
},
|
|
);
|
|
expect(sendSlackMessage).toHaveBeenCalledWith("channel:C123", "Off mode", {
|
|
mediaUrl: undefined,
|
|
threadTs: undefined,
|
|
});
|
|
});
|
|
|
|
it("does not auto-inject threadTs when sending to different channel", async () => {
|
|
const cfg = { slack: { botToken: "tok" } } as ClawdbotConfig;
|
|
sendSlackMessage.mockClear();
|
|
await handleSlackAction(
|
|
{
|
|
action: "sendMessage",
|
|
to: "channel:C999",
|
|
content: "Different channel",
|
|
},
|
|
cfg,
|
|
{
|
|
currentChannelId: "C123",
|
|
currentThreadTs: "1111111111.111111",
|
|
replyToMode: "all",
|
|
},
|
|
);
|
|
expect(sendSlackMessage).toHaveBeenCalledWith(
|
|
"channel:C999",
|
|
"Different channel",
|
|
{
|
|
mediaUrl: undefined,
|
|
threadTs: undefined,
|
|
},
|
|
);
|
|
});
|
|
|
|
it("explicit threadTs overrides context threadTs", async () => {
|
|
const cfg = { slack: { botToken: "tok" } } as ClawdbotConfig;
|
|
sendSlackMessage.mockClear();
|
|
await handleSlackAction(
|
|
{
|
|
action: "sendMessage",
|
|
to: "channel:C123",
|
|
content: "Explicit thread",
|
|
threadTs: "2222222222.222222",
|
|
},
|
|
cfg,
|
|
{
|
|
currentChannelId: "C123",
|
|
currentThreadTs: "1111111111.111111",
|
|
replyToMode: "all",
|
|
},
|
|
);
|
|
expect(sendSlackMessage).toHaveBeenCalledWith(
|
|
"channel:C123",
|
|
"Explicit thread",
|
|
{
|
|
mediaUrl: undefined,
|
|
threadTs: "2222222222.222222",
|
|
},
|
|
);
|
|
});
|
|
|
|
it("handles channel target without prefix when replyToMode=all", async () => {
|
|
const cfg = { slack: { botToken: "tok" } } as ClawdbotConfig;
|
|
sendSlackMessage.mockClear();
|
|
await handleSlackAction(
|
|
{
|
|
action: "sendMessage",
|
|
to: "C123",
|
|
content: "No prefix",
|
|
},
|
|
cfg,
|
|
{
|
|
currentChannelId: "C123",
|
|
currentThreadTs: "1111111111.111111",
|
|
replyToMode: "all",
|
|
},
|
|
);
|
|
expect(sendSlackMessage).toHaveBeenCalledWith("C123", "No prefix", {
|
|
mediaUrl: undefined,
|
|
threadTs: "1111111111.111111",
|
|
});
|
|
});
|
|
});
|