openclaw/src/infra/outbound/target-resolver.test.ts
Teconomix 6d0547dc2e
mattermost: fix DM media upload for unprefixed user IDs (#29925)
Merged via squash.

Prepared head SHA: 5cffcb072cc82394fe4c93d6c1c0c520325180b7
Co-authored-by: teconomix <6959299+teconomix@users.noreply.github.com>
Co-authored-by: mukhtharcm <56378562+mukhtharcm@users.noreply.github.com>
Reviewed-by: @mukhtharcm
2026-03-10 14:22:24 +05:30

125 lines
3.6 KiB
TypeScript

import { beforeEach, describe, expect, it, vi } from "vitest";
import type { ChannelDirectoryEntry } from "../../channels/plugins/types.js";
import type { OpenClawConfig } from "../../config/config.js";
import { resetDirectoryCache, resolveMessagingTarget } from "./target-resolver.js";
const mocks = vi.hoisted(() => ({
listGroups: vi.fn(),
listGroupsLive: vi.fn(),
resolveTarget: vi.fn(),
getChannelPlugin: vi.fn(),
}));
vi.mock("../../channels/plugins/index.js", () => ({
getChannelPlugin: (...args: unknown[]) => mocks.getChannelPlugin(...args),
normalizeChannelId: (value: string) => value,
}));
describe("resolveMessagingTarget (directory fallback)", () => {
const cfg = {} as OpenClawConfig;
beforeEach(() => {
mocks.listGroups.mockClear();
mocks.listGroupsLive.mockClear();
mocks.resolveTarget.mockClear();
mocks.getChannelPlugin.mockClear();
resetDirectoryCache();
mocks.getChannelPlugin.mockReturnValue({
directory: {
listGroups: mocks.listGroups,
listGroupsLive: mocks.listGroupsLive,
},
messaging: {
targetResolver: {
resolveTarget: mocks.resolveTarget,
},
},
});
});
it("uses live directory fallback and caches the result", async () => {
const entry: ChannelDirectoryEntry = { kind: "group", id: "123456789", name: "support" };
mocks.listGroups.mockResolvedValue([]);
mocks.listGroupsLive.mockResolvedValue([entry]);
const first = await resolveMessagingTarget({
cfg,
channel: "discord",
input: "support",
});
expect(first.ok).toBe(true);
if (first.ok) {
expect(first.target.source).toBe("directory");
expect(first.target.to).toBe("123456789");
}
expect(mocks.listGroups).toHaveBeenCalledTimes(1);
expect(mocks.listGroupsLive).toHaveBeenCalledTimes(1);
const second = await resolveMessagingTarget({
cfg,
channel: "discord",
input: "support",
});
expect(second.ok).toBe(true);
expect(mocks.listGroups).toHaveBeenCalledTimes(1);
expect(mocks.listGroupsLive).toHaveBeenCalledTimes(1);
});
it("skips directory lookup for direct ids", async () => {
const result = await resolveMessagingTarget({
cfg,
channel: "discord",
input: "123456789",
});
expect(result.ok).toBe(true);
if (result.ok) {
expect(result.target.source).toBe("normalized");
expect(result.target.to).toBe("123456789");
}
expect(mocks.listGroups).not.toHaveBeenCalled();
expect(mocks.listGroupsLive).not.toHaveBeenCalled();
});
it("lets plugins override id-like target resolution before falling back to raw ids", async () => {
mocks.getChannelPlugin.mockReturnValue({
messaging: {
targetResolver: {
looksLikeId: () => true,
resolveTarget: mocks.resolveTarget,
},
},
});
mocks.resolveTarget.mockResolvedValue({
to: "user:dm-user-id",
kind: "user",
source: "directory",
});
const result = await resolveMessagingTarget({
cfg,
channel: "mattermost",
input: "dthcxgoxhifn3pwh65cut3ud3w",
});
expect(result.ok).toBe(true);
if (result.ok) {
expect(result.target).toEqual({
to: "user:dm-user-id",
kind: "user",
source: "directory",
display: undefined,
});
}
expect(mocks.resolveTarget).toHaveBeenCalledWith(
expect.objectContaining({
input: "dthcxgoxhifn3pwh65cut3ud3w",
}),
);
expect(mocks.listGroups).not.toHaveBeenCalled();
expect(mocks.listGroupsLive).not.toHaveBeenCalled();
});
});