* feat(zalouser): add markdown-to-Zalo text style parsing Parse markdown formatting (bold, italic, strikethrough, headings, lists, code blocks, blockquotes, custom color/style tags) into Zalo native TextStyle ranges so outbound messages render with rich formatting. - Add text-styles.ts with parseZalouserTextStyles() converter - Wire markdown mode into send pipeline (sendMessageZalouser) - Export TextStyle enum and Style type from zca-client - Add textMode/textStyles to ZaloSendOptions - Pass textStyles through sendZaloTextMessage to zca-js API - Enable textMode:"markdown" in outbound sendText/sendMedia and monitor - Add comprehensive tests for parsing, send, and channel integration * fix(zalouser): harden markdown text parsing * fix(zalouser): mirror zca-js text style types * fix(zalouser): support tilde fenced code blocks * fix(zalouser): handle quoted fenced code blocks * fix(zalouser): preserve literal quote lines in code fences * fix(zalouser): support indented quoted fences * fix(zalouser): preserve quoted markdown blocks * fix(zalouser): rechunk formatted messages * fix(zalouser): preserve markdown structure across chunks * fix(zalouser): honor chunk limits and CRLF fences
168 lines
5.4 KiB
TypeScript
168 lines
5.4 KiB
TypeScript
import type { ReplyPayload } from "openclaw/plugin-sdk/zalouser";
|
|
import { beforeEach, describe, expect, it, vi } from "vitest";
|
|
import {
|
|
installSendPayloadContractSuite,
|
|
primeSendMock,
|
|
} from "../../../src/test-utils/send-payload-contract.js";
|
|
import { zalouserPlugin } from "./channel.js";
|
|
import { setZalouserRuntime } from "./runtime.js";
|
|
|
|
vi.mock("./send.js", () => ({
|
|
sendMessageZalouser: vi.fn().mockResolvedValue({ ok: true, messageId: "zlu-1" }),
|
|
sendReactionZalouser: vi.fn().mockResolvedValue({ ok: true }),
|
|
}));
|
|
|
|
vi.mock("./accounts.js", async (importOriginal) => {
|
|
const actual = (await importOriginal()) as Record<string, unknown>;
|
|
return {
|
|
...actual,
|
|
resolveZalouserAccountSync: () => ({
|
|
accountId: "default",
|
|
profile: "default",
|
|
name: "test",
|
|
enabled: true,
|
|
config: {},
|
|
}),
|
|
};
|
|
});
|
|
|
|
function baseCtx(payload: ReplyPayload) {
|
|
return {
|
|
cfg: {},
|
|
to: "user:987654321",
|
|
text: "",
|
|
payload,
|
|
};
|
|
}
|
|
|
|
describe("zalouserPlugin outbound sendPayload", () => {
|
|
let mockedSend: ReturnType<typeof vi.mocked<(typeof import("./send.js"))["sendMessageZalouser"]>>;
|
|
|
|
beforeEach(async () => {
|
|
setZalouserRuntime({
|
|
channel: {
|
|
text: {
|
|
resolveChunkMode: vi.fn(() => "length"),
|
|
resolveTextChunkLimit: vi.fn(() => 1200),
|
|
},
|
|
},
|
|
} as never);
|
|
const mod = await import("./send.js");
|
|
mockedSend = vi.mocked(mod.sendMessageZalouser);
|
|
mockedSend.mockClear();
|
|
mockedSend.mockResolvedValue({ ok: true, messageId: "zlu-1" });
|
|
});
|
|
|
|
it("group target delegates with isGroup=true and stripped threadId", async () => {
|
|
mockedSend.mockResolvedValue({ ok: true, messageId: "zlu-g1" });
|
|
|
|
const result = await zalouserPlugin.outbound!.sendPayload!({
|
|
...baseCtx({ text: "hello group" }),
|
|
to: "group:1471383327500481391",
|
|
});
|
|
|
|
expect(mockedSend).toHaveBeenCalledWith(
|
|
"1471383327500481391",
|
|
"hello group",
|
|
expect.objectContaining({ isGroup: true, textMode: "markdown" }),
|
|
);
|
|
expect(result).toMatchObject({ channel: "zalouser", messageId: "zlu-g1" });
|
|
});
|
|
|
|
it("treats bare numeric targets as direct chats for backward compatibility", async () => {
|
|
mockedSend.mockResolvedValue({ ok: true, messageId: "zlu-d1" });
|
|
|
|
const result = await zalouserPlugin.outbound!.sendPayload!({
|
|
...baseCtx({ text: "hello" }),
|
|
to: "987654321",
|
|
});
|
|
|
|
expect(mockedSend).toHaveBeenCalledWith(
|
|
"987654321",
|
|
"hello",
|
|
expect.objectContaining({ isGroup: false, textMode: "markdown" }),
|
|
);
|
|
expect(result).toMatchObject({ channel: "zalouser", messageId: "zlu-d1" });
|
|
});
|
|
|
|
it("preserves provider-native group ids when sending to raw g- targets", async () => {
|
|
mockedSend.mockResolvedValue({ ok: true, messageId: "zlu-g-native" });
|
|
|
|
const result = await zalouserPlugin.outbound!.sendPayload!({
|
|
...baseCtx({ text: "hello native group" }),
|
|
to: "g-1471383327500481391",
|
|
});
|
|
|
|
expect(mockedSend).toHaveBeenCalledWith(
|
|
"g-1471383327500481391",
|
|
"hello native group",
|
|
expect.objectContaining({ isGroup: true, textMode: "markdown" }),
|
|
);
|
|
expect(result).toMatchObject({ channel: "zalouser", messageId: "zlu-g-native" });
|
|
});
|
|
|
|
it("passes long markdown through once so formatting happens before chunking", async () => {
|
|
const text = `**${"a".repeat(2501)}**`;
|
|
mockedSend.mockResolvedValue({ ok: true, messageId: "zlu-code" });
|
|
|
|
const result = await zalouserPlugin.outbound!.sendPayload!({
|
|
...baseCtx({ text }),
|
|
to: "987654321",
|
|
});
|
|
|
|
expect(mockedSend).toHaveBeenCalledTimes(1);
|
|
expect(mockedSend).toHaveBeenCalledWith(
|
|
"987654321",
|
|
text,
|
|
expect.objectContaining({
|
|
isGroup: false,
|
|
textMode: "markdown",
|
|
textChunkMode: "length",
|
|
textChunkLimit: 1200,
|
|
}),
|
|
);
|
|
expect(result).toMatchObject({ channel: "zalouser", messageId: "zlu-code" });
|
|
});
|
|
|
|
installSendPayloadContractSuite({
|
|
channel: "zalouser",
|
|
chunking: { mode: "passthrough", longTextLength: 3000 },
|
|
createHarness: ({ payload, sendResults }) => {
|
|
primeSendMock(mockedSend, { ok: true, messageId: "zlu-1" }, sendResults);
|
|
return {
|
|
run: async () => await zalouserPlugin.outbound!.sendPayload!(baseCtx(payload)),
|
|
sendMock: mockedSend,
|
|
to: "987654321",
|
|
};
|
|
},
|
|
});
|
|
});
|
|
|
|
describe("zalouserPlugin messaging target normalization", () => {
|
|
it("normalizes user/group aliases to canonical targets", () => {
|
|
const normalize = zalouserPlugin.messaging?.normalizeTarget;
|
|
expect(normalize).toBeTypeOf("function");
|
|
if (!normalize) {
|
|
return;
|
|
}
|
|
expect(normalize("zlu:g:30003")).toBe("group:30003");
|
|
expect(normalize("zalouser:u:20002")).toBe("user:20002");
|
|
expect(normalize("zlu:g-30003")).toBe("group:g-30003");
|
|
expect(normalize("zalouser:u-20002")).toBe("user:u-20002");
|
|
expect(normalize("20002")).toBe("20002");
|
|
});
|
|
|
|
it("treats canonical and provider-native user/group targets as ids", () => {
|
|
const looksLikeId = zalouserPlugin.messaging?.targetResolver?.looksLikeId;
|
|
expect(looksLikeId).toBeTypeOf("function");
|
|
if (!looksLikeId) {
|
|
return;
|
|
}
|
|
expect(looksLikeId("user:20002")).toBe(true);
|
|
expect(looksLikeId("group:30003")).toBe(true);
|
|
expect(looksLikeId("g-30003")).toBe(true);
|
|
expect(looksLikeId("u-20002")).toBe(true);
|
|
expect(looksLikeId("Alice Nguyen")).toBe(false);
|
|
});
|
|
});
|