Files
openclaw/extensions/msteams/src/graph-teams.test.ts
sudie-codes 928a5128f4 msteams: add channel-list and channel-info actions (#57529)
* msteams: add channel-list and channel-info actions via Graph API

* msteams: use action helpers, add channel-list pagination

* msteams: address PR #57529 review feedback
2026-04-04 02:43:08 -05:00

216 lines
6.5 KiB
TypeScript

import { beforeEach, describe, expect, it, vi } from "vitest";
import type { OpenClawConfig } from "../runtime-api.js";
import { getChannelInfoMSTeams, listChannelsMSTeams } from "./graph-teams.js";
const mockState = vi.hoisted(() => ({
resolveGraphToken: vi.fn(),
fetchGraphJson: vi.fn(),
}));
vi.mock("./graph.js", async (importOriginal) => {
const actual = await importOriginal<typeof import("./graph.js")>();
return {
...actual,
resolveGraphToken: mockState.resolveGraphToken,
fetchGraphJson: mockState.fetchGraphJson,
};
});
const TOKEN = "test-graph-token";
describe("listChannelsMSTeams", () => {
beforeEach(() => {
mockState.resolveGraphToken.mockReset().mockResolvedValue(TOKEN);
mockState.fetchGraphJson.mockReset();
});
it("returns channels with all fields mapped", async () => {
mockState.fetchGraphJson.mockResolvedValue({
value: [
{
id: "ch-1",
displayName: "General",
description: "The default channel",
membershipType: "standard",
},
{
id: "ch-2",
displayName: "Engineering",
description: "Engineering discussions",
membershipType: "private",
},
],
});
const result = await listChannelsMSTeams({
cfg: {} as OpenClawConfig,
teamId: "team-abc",
});
expect(result.channels).toEqual([
{
id: "ch-1",
displayName: "General",
description: "The default channel",
membershipType: "standard",
},
{
id: "ch-2",
displayName: "Engineering",
description: "Engineering discussions",
membershipType: "private",
},
]);
expect(mockState.fetchGraphJson).toHaveBeenCalledWith({
token: TOKEN,
path: `/teams/${encodeURIComponent("team-abc")}/channels?$select=id,displayName,description,membershipType`,
});
});
it("returns empty array when team has no channels", async () => {
mockState.fetchGraphJson.mockResolvedValue({ value: [] });
const result = await listChannelsMSTeams({
cfg: {} as OpenClawConfig,
teamId: "team-empty",
});
expect(result.channels).toEqual([]);
});
it("returns empty array when value is undefined", async () => {
mockState.fetchGraphJson.mockResolvedValue({});
const result = await listChannelsMSTeams({
cfg: {} as OpenClawConfig,
teamId: "team-no-value",
});
expect(result.channels).toEqual([]);
});
it("follows @odata.nextLink across multiple pages", async () => {
mockState.fetchGraphJson
.mockResolvedValueOnce({
value: [
{ id: "ch-1", displayName: "General", description: null, membershipType: "standard" },
],
"@odata.nextLink":
"https://graph.microsoft.com/v1.0/teams/team-paged/channels?$select=id,displayName,description,membershipType&$skip=1",
})
.mockResolvedValueOnce({
value: [
{ id: "ch-2", displayName: "Random", description: "Fun", membershipType: "standard" },
],
"@odata.nextLink":
"https://graph.microsoft.com/v1.0/teams/team-paged/channels?$select=id,displayName,description,membershipType&$skip=2",
})
.mockResolvedValueOnce({
value: [
{ id: "ch-3", displayName: "Private", description: null, membershipType: "private" },
],
});
const result = await listChannelsMSTeams({
cfg: {} as OpenClawConfig,
teamId: "team-paged",
});
expect(result.channels).toHaveLength(3);
expect(result.channels.map((ch) => ch.id)).toEqual(["ch-1", "ch-2", "ch-3"]);
expect(result.truncated).toBe(false);
expect(mockState.fetchGraphJson).toHaveBeenCalledTimes(3);
// Second call should use the relative path stripped from the nextLink
const secondCallPath = mockState.fetchGraphJson.mock.calls[1]?.[0]?.path;
expect(secondCallPath).toBe(
"/teams/team-paged/channels?$select=id,displayName,description,membershipType&$skip=1",
);
});
it("stops after 10 pages to avoid runaway pagination", async () => {
for (let i = 0; i < 11; i++) {
mockState.fetchGraphJson.mockResolvedValueOnce({
value: [
{
id: `ch-${i}`,
displayName: `Channel ${i}`,
description: null,
membershipType: "standard",
},
],
"@odata.nextLink": `https://graph.microsoft.com/v1.0/teams/team-huge/channels?$skip=${i + 1}`,
});
}
const result = await listChannelsMSTeams({
cfg: {} as OpenClawConfig,
teamId: "team-huge",
});
// Should stop at 10 pages even though more nextLinks are available
expect(result.channels).toHaveLength(10);
expect(mockState.fetchGraphJson).toHaveBeenCalledTimes(10);
expect(result.truncated).toBe(true);
});
});
describe("getChannelInfoMSTeams", () => {
beforeEach(() => {
mockState.resolveGraphToken.mockReset().mockResolvedValue(TOKEN);
mockState.fetchGraphJson.mockReset();
});
it("returns channel with all fields", async () => {
mockState.fetchGraphJson.mockResolvedValue({
id: "ch-1",
displayName: "General",
description: "The default channel",
membershipType: "standard",
webUrl: "https://teams.microsoft.com/l/channel/ch-1/General",
createdDateTime: "2026-01-15T09:00:00Z",
});
const result = await getChannelInfoMSTeams({
cfg: {} as OpenClawConfig,
teamId: "team-abc",
channelId: "ch-1",
});
expect(result.channel).toEqual({
id: "ch-1",
displayName: "General",
description: "The default channel",
membershipType: "standard",
webUrl: "https://teams.microsoft.com/l/channel/ch-1/General",
createdDateTime: "2026-01-15T09:00:00Z",
});
expect(mockState.fetchGraphJson).toHaveBeenCalledWith({
token: TOKEN,
path: `/teams/${encodeURIComponent("team-abc")}/channels/${encodeURIComponent("ch-1")}?$select=id,displayName,description,membershipType,webUrl,createdDateTime`,
});
});
it("handles missing optional fields gracefully", async () => {
mockState.fetchGraphJson.mockResolvedValue({
id: "ch-2",
displayName: "Private Channel",
});
const result = await getChannelInfoMSTeams({
cfg: {} as OpenClawConfig,
teamId: "team-abc",
channelId: "ch-2",
});
expect(result.channel).toEqual({
id: "ch-2",
displayName: "Private Channel",
description: undefined,
membershipType: undefined,
webUrl: undefined,
createdDateTime: undefined,
});
});
});