mirror of
https://github.com/openclaw/openclaw.git
synced 2026-03-19 05:50:47 +00:00
Move all Slack channel implementation files from src/slack/ to extensions/slack/src/ and replace originals with shim re-exports. This follows the extension migration pattern for channel plugins. - Copy all .ts files to extensions/slack/src/ (preserving directory structure: monitor/, http/, monitor/events/, monitor/message-handler/) - Transform import paths: external src/ imports use relative paths back to src/, internal slack imports stay relative within extension - Replace all src/slack/ files with shim re-exports pointing to the extension copies - Update tsconfig.plugin-sdk.dts.json rootDir from "src" to "." so the DTS build can follow shim chains into extensions/ - Update write-plugin-sdk-entry-dts.ts re-export path accordingly - Preserve extensions/slack/index.ts, package.json, openclaw.plugin.json, src/channel.ts, src/runtime.ts, src/channel.test.ts (untouched)
187 lines
5.5 KiB
TypeScript
187 lines
5.5 KiB
TypeScript
import type { WebClient } from "@slack/web-api";
|
|
import { afterEach, beforeEach, describe, expect, it, vi } from "vitest";
|
|
import { installSlackBlockTestMocks } from "./blocks.test-helpers.js";
|
|
|
|
// --- Module mocks (must precede dynamic import) ---
|
|
installSlackBlockTestMocks();
|
|
const fetchWithSsrFGuard = vi.fn(
|
|
async (params: { url: string; init?: RequestInit }) =>
|
|
({
|
|
response: await fetch(params.url, params.init),
|
|
finalUrl: params.url,
|
|
release: async () => {},
|
|
}) as const,
|
|
);
|
|
|
|
vi.mock("../../../src/infra/net/fetch-guard.js", () => ({
|
|
fetchWithSsrFGuard: (...args: unknown[]) =>
|
|
fetchWithSsrFGuard(...(args as [params: { url: string; init?: RequestInit }])),
|
|
withTrustedEnvProxyGuardedFetchMode: (params: Record<string, unknown>) => ({
|
|
...params,
|
|
mode: "trusted_env_proxy",
|
|
}),
|
|
}));
|
|
|
|
vi.mock("../../whatsapp/src/media.js", () => ({
|
|
loadWebMedia: vi.fn(async () => ({
|
|
buffer: Buffer.from("fake-image"),
|
|
contentType: "image/png",
|
|
kind: "image",
|
|
fileName: "screenshot.png",
|
|
})),
|
|
}));
|
|
|
|
const { sendMessageSlack } = await import("./send.js");
|
|
|
|
type UploadTestClient = WebClient & {
|
|
conversations: { open: ReturnType<typeof vi.fn> };
|
|
chat: { postMessage: ReturnType<typeof vi.fn> };
|
|
files: {
|
|
getUploadURLExternal: ReturnType<typeof vi.fn>;
|
|
completeUploadExternal: ReturnType<typeof vi.fn>;
|
|
};
|
|
};
|
|
|
|
function createUploadTestClient(): UploadTestClient {
|
|
return {
|
|
conversations: {
|
|
open: vi.fn(async () => ({ channel: { id: "D99RESOLVED" } })),
|
|
},
|
|
chat: {
|
|
postMessage: vi.fn(async () => ({ ts: "171234.567" })),
|
|
},
|
|
files: {
|
|
getUploadURLExternal: vi.fn(async () => ({
|
|
ok: true,
|
|
upload_url: "https://uploads.slack.test/upload",
|
|
file_id: "F001",
|
|
})),
|
|
completeUploadExternal: vi.fn(async () => ({ ok: true })),
|
|
},
|
|
} as unknown as UploadTestClient;
|
|
}
|
|
|
|
describe("sendMessageSlack file upload with user IDs", () => {
|
|
const originalFetch = globalThis.fetch;
|
|
|
|
beforeEach(() => {
|
|
globalThis.fetch = vi.fn(
|
|
async () => new Response("ok", { status: 200 }),
|
|
) as unknown as typeof fetch;
|
|
fetchWithSsrFGuard.mockClear();
|
|
});
|
|
|
|
afterEach(() => {
|
|
globalThis.fetch = originalFetch;
|
|
vi.restoreAllMocks();
|
|
});
|
|
|
|
it("resolves bare user ID to DM channel before completing upload", async () => {
|
|
const client = createUploadTestClient();
|
|
|
|
// Bare user ID — parseSlackTarget classifies this as kind="channel"
|
|
await sendMessageSlack("U2ZH3MFSR", "screenshot", {
|
|
token: "xoxb-test",
|
|
client,
|
|
mediaUrl: "/tmp/screenshot.png",
|
|
});
|
|
|
|
// Should call conversations.open to resolve user ID → DM channel
|
|
expect(client.conversations.open).toHaveBeenCalledWith({
|
|
users: "U2ZH3MFSR",
|
|
});
|
|
|
|
expect(client.files.completeUploadExternal).toHaveBeenCalledWith(
|
|
expect.objectContaining({
|
|
channel_id: "D99RESOLVED",
|
|
files: [expect.objectContaining({ id: "F001", title: "screenshot.png" })],
|
|
}),
|
|
);
|
|
});
|
|
|
|
it("resolves prefixed user ID to DM channel before completing upload", async () => {
|
|
const client = createUploadTestClient();
|
|
|
|
await sendMessageSlack("user:UABC123", "image", {
|
|
token: "xoxb-test",
|
|
client,
|
|
mediaUrl: "/tmp/photo.png",
|
|
});
|
|
|
|
expect(client.conversations.open).toHaveBeenCalledWith({
|
|
users: "UABC123",
|
|
});
|
|
expect(client.files.completeUploadExternal).toHaveBeenCalledWith(
|
|
expect.objectContaining({ channel_id: "D99RESOLVED" }),
|
|
);
|
|
});
|
|
|
|
it("sends file directly to channel without conversations.open", async () => {
|
|
const client = createUploadTestClient();
|
|
|
|
await sendMessageSlack("channel:C123CHAN", "chart", {
|
|
token: "xoxb-test",
|
|
client,
|
|
mediaUrl: "/tmp/chart.png",
|
|
});
|
|
|
|
expect(client.conversations.open).not.toHaveBeenCalled();
|
|
expect(client.files.completeUploadExternal).toHaveBeenCalledWith(
|
|
expect.objectContaining({ channel_id: "C123CHAN" }),
|
|
);
|
|
});
|
|
|
|
it("resolves mention-style user ID before file upload", async () => {
|
|
const client = createUploadTestClient();
|
|
|
|
await sendMessageSlack("<@U777TEST>", "report", {
|
|
token: "xoxb-test",
|
|
client,
|
|
mediaUrl: "/tmp/report.png",
|
|
});
|
|
|
|
expect(client.conversations.open).toHaveBeenCalledWith({
|
|
users: "U777TEST",
|
|
});
|
|
expect(client.files.completeUploadExternal).toHaveBeenCalledWith(
|
|
expect.objectContaining({ channel_id: "D99RESOLVED" }),
|
|
);
|
|
});
|
|
|
|
it("uploads bytes to the presigned URL and completes with thread+caption", async () => {
|
|
const client = createUploadTestClient();
|
|
|
|
await sendMessageSlack("channel:C123CHAN", "caption", {
|
|
token: "xoxb-test",
|
|
client,
|
|
mediaUrl: "/tmp/threaded.png",
|
|
threadTs: "171.222",
|
|
});
|
|
|
|
expect(client.files.getUploadURLExternal).toHaveBeenCalledWith({
|
|
filename: "screenshot.png",
|
|
length: Buffer.from("fake-image").length,
|
|
});
|
|
expect(globalThis.fetch).toHaveBeenCalledWith(
|
|
"https://uploads.slack.test/upload",
|
|
expect.objectContaining({
|
|
method: "POST",
|
|
}),
|
|
);
|
|
expect(fetchWithSsrFGuard).toHaveBeenCalledWith(
|
|
expect.objectContaining({
|
|
url: "https://uploads.slack.test/upload",
|
|
mode: "trusted_env_proxy",
|
|
auditContext: "slack-upload-file",
|
|
}),
|
|
);
|
|
expect(client.files.completeUploadExternal).toHaveBeenCalledWith(
|
|
expect.objectContaining({
|
|
channel_id: "C123CHAN",
|
|
initial_comment: "caption",
|
|
thread_ts: "171.222",
|
|
}),
|
|
);
|
|
});
|
|
});
|