mirror of
https://github.com/openclaw/openclaw.git
synced 2026-04-20 21:51:28 +00:00
312 lines
8.9 KiB
TypeScript
312 lines
8.9 KiB
TypeScript
import { beforeEach, describe, expect, it, vi } from "vitest";
|
|
import type { OpenClawConfig } from "../runtime-api.js";
|
|
import { getMessageMSTeams, listPinsMSTeams, listReactionsMSTeams } from "./graph-messages.js";
|
|
|
|
const mockState = vi.hoisted(() => ({
|
|
resolveGraphToken: vi.fn(),
|
|
fetchGraphJson: vi.fn(),
|
|
postGraphJson: vi.fn(),
|
|
postGraphBetaJson: vi.fn(),
|
|
deleteGraphRequest: vi.fn(),
|
|
findPreferredDmByUserId: vi.fn(),
|
|
}));
|
|
|
|
vi.mock("./graph.js", () => {
|
|
return {
|
|
resolveGraphToken: mockState.resolveGraphToken,
|
|
fetchGraphJson: mockState.fetchGraphJson,
|
|
postGraphJson: mockState.postGraphJson,
|
|
postGraphBetaJson: mockState.postGraphBetaJson,
|
|
deleteGraphRequest: mockState.deleteGraphRequest,
|
|
escapeOData: vi.fn((value: string) => value.replaceAll("'", "''")),
|
|
};
|
|
});
|
|
|
|
vi.mock("./conversation-store-fs.js", () => ({
|
|
createMSTeamsConversationStoreFs: () => ({
|
|
findPreferredDmByUserId: mockState.findPreferredDmByUserId,
|
|
}),
|
|
}));
|
|
|
|
const TOKEN = "test-graph-token";
|
|
const CHAT_ID = "19:abc@thread.tacv2";
|
|
const CHANNEL_TO = "team-id-1/channel-id-1";
|
|
|
|
describe("getMessageMSTeams", () => {
|
|
beforeEach(() => {
|
|
vi.clearAllMocks();
|
|
mockState.resolveGraphToken.mockResolvedValue(TOKEN);
|
|
});
|
|
|
|
it("resolves user: target using graphChatId from store", async () => {
|
|
mockState.findPreferredDmByUserId.mockResolvedValue({
|
|
conversationId: "a:bot-framework-dm-id",
|
|
reference: { graphChatId: "19:graph-native-chat@thread.tacv2" },
|
|
});
|
|
mockState.fetchGraphJson.mockResolvedValue({
|
|
id: "msg-1",
|
|
body: { content: "From user DM" },
|
|
createdDateTime: "2026-03-23T12:00:00Z",
|
|
});
|
|
|
|
await getMessageMSTeams({
|
|
cfg: {} as OpenClawConfig,
|
|
to: "user:aad-object-id-123",
|
|
messageId: "msg-1",
|
|
});
|
|
|
|
expect(mockState.findPreferredDmByUserId).toHaveBeenCalledWith("aad-object-id-123");
|
|
expect(mockState.fetchGraphJson).toHaveBeenCalledWith({
|
|
token: TOKEN,
|
|
path: `/chats/${encodeURIComponent("19:graph-native-chat@thread.tacv2")}/messages/msg-1`,
|
|
});
|
|
});
|
|
|
|
it("falls back to conversationId when it starts with 19:", async () => {
|
|
mockState.findPreferredDmByUserId.mockResolvedValue({
|
|
conversationId: "19:resolved-chat@thread.tacv2",
|
|
reference: {},
|
|
});
|
|
mockState.fetchGraphJson.mockResolvedValue({
|
|
id: "msg-1",
|
|
body: { content: "Hello" },
|
|
createdDateTime: "2026-03-23T10:00:00Z",
|
|
});
|
|
|
|
await getMessageMSTeams({
|
|
cfg: {} as OpenClawConfig,
|
|
to: "user:aad-id",
|
|
messageId: "msg-1",
|
|
});
|
|
|
|
expect(mockState.fetchGraphJson).toHaveBeenCalledWith({
|
|
token: TOKEN,
|
|
path: `/chats/${encodeURIComponent("19:resolved-chat@thread.tacv2")}/messages/msg-1`,
|
|
});
|
|
});
|
|
|
|
it("throws when user: target has no stored conversation", async () => {
|
|
mockState.findPreferredDmByUserId.mockResolvedValue(null);
|
|
|
|
await expect(
|
|
getMessageMSTeams({
|
|
cfg: {} as OpenClawConfig,
|
|
to: "user:unknown-user",
|
|
messageId: "msg-1",
|
|
}),
|
|
).rejects.toThrow("No conversation found for user:unknown-user");
|
|
});
|
|
|
|
it("throws when user: target has Bot Framework ID and no graphChatId", async () => {
|
|
mockState.findPreferredDmByUserId.mockResolvedValue({
|
|
conversationId: "a:bot-framework-dm-id",
|
|
reference: {},
|
|
});
|
|
|
|
await expect(
|
|
getMessageMSTeams({
|
|
cfg: {} as OpenClawConfig,
|
|
to: "user:some-user",
|
|
messageId: "msg-1",
|
|
}),
|
|
).rejects.toThrow("Bot Framework ID");
|
|
});
|
|
|
|
it("strips conversation: prefix from target", async () => {
|
|
mockState.fetchGraphJson.mockResolvedValue({
|
|
id: "msg-1",
|
|
body: { content: "Hello" },
|
|
from: undefined,
|
|
createdDateTime: "2026-03-23T10:00:00Z",
|
|
});
|
|
|
|
await getMessageMSTeams({
|
|
cfg: {} as OpenClawConfig,
|
|
to: `conversation:${CHAT_ID}`,
|
|
messageId: "msg-1",
|
|
});
|
|
|
|
expect(mockState.fetchGraphJson).toHaveBeenCalledWith({
|
|
token: TOKEN,
|
|
path: `/chats/${encodeURIComponent(CHAT_ID)}/messages/msg-1`,
|
|
});
|
|
});
|
|
|
|
it("reads a message from a chat conversation", async () => {
|
|
mockState.fetchGraphJson.mockResolvedValue({
|
|
id: "msg-1",
|
|
body: { content: "Hello world", contentType: "text" },
|
|
from: { user: { id: "user-1", displayName: "Alice" } },
|
|
createdDateTime: "2026-03-23T10:00:00Z",
|
|
});
|
|
|
|
const result = await getMessageMSTeams({
|
|
cfg: {} as OpenClawConfig,
|
|
to: CHAT_ID,
|
|
messageId: "msg-1",
|
|
});
|
|
|
|
expect(result).toEqual({
|
|
id: "msg-1",
|
|
text: "Hello world",
|
|
from: { user: { id: "user-1", displayName: "Alice" } },
|
|
createdAt: "2026-03-23T10:00:00Z",
|
|
});
|
|
expect(mockState.fetchGraphJson).toHaveBeenCalledWith({
|
|
token: TOKEN,
|
|
path: `/chats/${encodeURIComponent(CHAT_ID)}/messages/msg-1`,
|
|
});
|
|
});
|
|
|
|
it("reads a message from a channel conversation", async () => {
|
|
mockState.fetchGraphJson.mockResolvedValue({
|
|
id: "msg-2",
|
|
body: { content: "Channel message" },
|
|
from: { application: { id: "app-1", displayName: "Bot" } },
|
|
createdDateTime: "2026-03-23T11:00:00Z",
|
|
});
|
|
|
|
const result = await getMessageMSTeams({
|
|
cfg: {} as OpenClawConfig,
|
|
to: CHANNEL_TO,
|
|
messageId: "msg-2",
|
|
});
|
|
|
|
expect(result).toEqual({
|
|
id: "msg-2",
|
|
text: "Channel message",
|
|
from: { application: { id: "app-1", displayName: "Bot" } },
|
|
createdAt: "2026-03-23T11:00:00Z",
|
|
});
|
|
expect(mockState.fetchGraphJson).toHaveBeenCalledWith({
|
|
token: TOKEN,
|
|
path: "/teams/team-id-1/channels/channel-id-1/messages/msg-2",
|
|
});
|
|
});
|
|
});
|
|
|
|
describe("listPinsMSTeams", () => {
|
|
beforeEach(() => {
|
|
vi.clearAllMocks();
|
|
mockState.resolveGraphToken.mockResolvedValue(TOKEN);
|
|
});
|
|
|
|
it("lists pinned messages in a chat", async () => {
|
|
mockState.fetchGraphJson.mockResolvedValue({
|
|
value: [
|
|
{
|
|
id: "pinned-1",
|
|
message: { id: "msg-1", body: { content: "Pinned msg" } },
|
|
},
|
|
{
|
|
id: "pinned-2",
|
|
message: { id: "msg-2", body: { content: "Another pin" } },
|
|
},
|
|
],
|
|
});
|
|
|
|
const result = await listPinsMSTeams({
|
|
cfg: {} as OpenClawConfig,
|
|
to: CHAT_ID,
|
|
});
|
|
|
|
expect(result.pins).toEqual([
|
|
{ id: "pinned-1", pinnedMessageId: "pinned-1", messageId: "msg-1", text: "Pinned msg" },
|
|
{ id: "pinned-2", pinnedMessageId: "pinned-2", messageId: "msg-2", text: "Another pin" },
|
|
]);
|
|
expect(mockState.fetchGraphJson).toHaveBeenCalledWith({
|
|
token: TOKEN,
|
|
path: `/chats/${encodeURIComponent(CHAT_ID)}/pinnedMessages?$expand=message`,
|
|
});
|
|
});
|
|
|
|
it("returns empty array when no pins exist", async () => {
|
|
mockState.fetchGraphJson.mockResolvedValue({ value: [] });
|
|
|
|
const result = await listPinsMSTeams({
|
|
cfg: {} as OpenClawConfig,
|
|
to: CHAT_ID,
|
|
});
|
|
|
|
expect(result.pins).toEqual([]);
|
|
});
|
|
});
|
|
|
|
describe("listReactionsMSTeams", () => {
|
|
beforeEach(() => {
|
|
vi.clearAllMocks();
|
|
mockState.resolveGraphToken.mockResolvedValue(TOKEN);
|
|
});
|
|
|
|
it("lists reactions grouped by type with user details", async () => {
|
|
mockState.fetchGraphJson.mockResolvedValue({
|
|
id: "msg-1",
|
|
body: { content: "Hello" },
|
|
reactions: [
|
|
{ reactionType: "like", user: { id: "u1", displayName: "Alice" } },
|
|
{ reactionType: "like", user: { id: "u2", displayName: "Bob" } },
|
|
{ reactionType: "heart", user: { id: "u1", displayName: "Alice" } },
|
|
],
|
|
});
|
|
|
|
const result = await listReactionsMSTeams({
|
|
cfg: {} as OpenClawConfig,
|
|
to: CHAT_ID,
|
|
messageId: "msg-1",
|
|
});
|
|
|
|
expect(result.reactions).toEqual([
|
|
{
|
|
reactionType: "like",
|
|
count: 2,
|
|
users: [
|
|
{ id: "u1", displayName: "Alice" },
|
|
{ id: "u2", displayName: "Bob" },
|
|
],
|
|
},
|
|
{
|
|
reactionType: "heart",
|
|
count: 1,
|
|
users: [{ id: "u1", displayName: "Alice" }],
|
|
},
|
|
]);
|
|
});
|
|
|
|
it("returns empty array when message has no reactions", async () => {
|
|
mockState.fetchGraphJson.mockResolvedValue({
|
|
id: "msg-1",
|
|
body: { content: "No reactions" },
|
|
});
|
|
|
|
const result = await listReactionsMSTeams({
|
|
cfg: {} as OpenClawConfig,
|
|
to: CHAT_ID,
|
|
messageId: "msg-1",
|
|
});
|
|
|
|
expect(result.reactions).toEqual([]);
|
|
});
|
|
|
|
it("fetches from channel path for channel targets", async () => {
|
|
mockState.fetchGraphJson.mockResolvedValue({
|
|
id: "msg-2",
|
|
body: { content: "Channel msg" },
|
|
reactions: [{ reactionType: "surprised", user: { id: "u3", displayName: "Carol" } }],
|
|
});
|
|
|
|
const result = await listReactionsMSTeams({
|
|
cfg: {} as OpenClawConfig,
|
|
to: CHANNEL_TO,
|
|
messageId: "msg-2",
|
|
});
|
|
|
|
expect(result.reactions).toEqual([
|
|
{ reactionType: "surprised", count: 1, users: [{ id: "u3", displayName: "Carol" }] },
|
|
]);
|
|
expect(mockState.fetchGraphJson).toHaveBeenCalledWith({
|
|
token: TOKEN,
|
|
path: "/teams/team-id-1/channels/channel-id-1/messages/msg-2",
|
|
});
|
|
});
|
|
});
|