fix(web): unwrap ephemeral/view-once and keep mentions
This commit is contained in:
@@ -166,9 +166,10 @@ export async function monitorWebInbox(options: {
|
|||||||
const timestamp = msg.messageTimestamp
|
const timestamp = msg.messageTimestamp
|
||||||
? Number(msg.messageTimestamp) * 1000
|
? Number(msg.messageTimestamp) * 1000
|
||||||
: undefined;
|
: undefined;
|
||||||
|
const unwrapped = unwrapMessage(msg.message as proto.IMessage | undefined);
|
||||||
const mentionedJids =
|
const mentionedJids =
|
||||||
msg.message?.extendedTextMessage?.contextInfo?.mentionedJid ??
|
unwrapped?.extendedTextMessage?.contextInfo?.mentionedJid ??
|
||||||
msg.message?.extendedTextMessage?.contextInfo?.quotedMessage
|
unwrapped?.extendedTextMessage?.contextInfo?.quotedMessage
|
||||||
?.extendedTextMessage?.contextInfo?.mentionedJid;
|
?.extendedTextMessage?.contextInfo?.mentionedJid;
|
||||||
const senderName = msg.pushName ?? undefined;
|
const senderName = msg.pushName ?? undefined;
|
||||||
inboundLogger.info(
|
inboundLogger.info(
|
||||||
@@ -302,9 +303,24 @@ export async function monitorWebInbox(options: {
|
|||||||
} as const;
|
} as const;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function unwrapMessage(message: proto.IMessage | undefined): proto.IMessage | undefined {
|
||||||
|
if (!message) return undefined;
|
||||||
|
if (message.ephemeralMessage?.message) {
|
||||||
|
return unwrapMessage(message.ephemeralMessage.message as proto.IMessage);
|
||||||
|
}
|
||||||
|
if (message.viewOnceMessage?.message) {
|
||||||
|
return unwrapMessage(message.viewOnceMessage.message as proto.IMessage);
|
||||||
|
}
|
||||||
|
if (message.viewOnceMessageV2?.message) {
|
||||||
|
return unwrapMessage(message.viewOnceMessageV2.message as proto.IMessage);
|
||||||
|
}
|
||||||
|
return message;
|
||||||
|
}
|
||||||
|
|
||||||
export function extractText(
|
export function extractText(
|
||||||
message: proto.IMessage | undefined,
|
rawMessage: proto.IMessage | undefined,
|
||||||
): string | undefined {
|
): string | undefined {
|
||||||
|
const message = unwrapMessage(rawMessage);
|
||||||
if (!message) return undefined;
|
if (!message) return undefined;
|
||||||
if (typeof message.conversation === "string" && message.conversation.trim()) {
|
if (typeof message.conversation === "string" && message.conversation.trim()) {
|
||||||
return message.conversation.trim();
|
return message.conversation.trim();
|
||||||
@@ -318,8 +334,9 @@ export function extractText(
|
|||||||
}
|
}
|
||||||
|
|
||||||
export function extractMediaPlaceholder(
|
export function extractMediaPlaceholder(
|
||||||
message: proto.IMessage | undefined,
|
rawMessage: proto.IMessage | undefined,
|
||||||
): string | undefined {
|
): string | undefined {
|
||||||
|
const message = unwrapMessage(rawMessage);
|
||||||
if (!message) return undefined;
|
if (!message) return undefined;
|
||||||
if (message.imageMessage) return "<media:image>";
|
if (message.imageMessage) return "<media:image>";
|
||||||
if (message.videoMessage) return "<media:video>";
|
if (message.videoMessage) return "<media:video>";
|
||||||
@@ -333,7 +350,7 @@ async function downloadInboundMedia(
|
|||||||
msg: proto.IWebMessageInfo,
|
msg: proto.IWebMessageInfo,
|
||||||
sock: Awaited<ReturnType<typeof createWaSocket>>,
|
sock: Awaited<ReturnType<typeof createWaSocket>>,
|
||||||
): Promise<{ buffer: Buffer; mimetype?: string } | undefined> {
|
): Promise<{ buffer: Buffer; mimetype?: string } | undefined> {
|
||||||
const message = msg.message;
|
const message = unwrapMessage(msg.message as proto.IMessage | undefined);
|
||||||
if (!message) return undefined;
|
if (!message) return undefined;
|
||||||
const mimetype =
|
const mimetype =
|
||||||
message.imageMessage?.mimetype ??
|
message.imageMessage?.mimetype ??
|
||||||
|
|||||||
@@ -270,6 +270,51 @@ describe("web monitor inbox", () => {
|
|||||||
await listener.close();
|
await listener.close();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it("unwraps ephemeral messages, preserves mentions, and still delivers group pings", async () => {
|
||||||
|
const onMessage = vi.fn();
|
||||||
|
const listener = await monitorWebInbox({ verbose: false, onMessage });
|
||||||
|
const sock = await createWaSocket();
|
||||||
|
const upsert = {
|
||||||
|
type: "notify",
|
||||||
|
messages: [
|
||||||
|
{
|
||||||
|
key: {
|
||||||
|
id: "grp-ephem",
|
||||||
|
fromMe: false,
|
||||||
|
remoteJid: "424242@g.us",
|
||||||
|
participant: "888@s.whatsapp.net",
|
||||||
|
},
|
||||||
|
message: {
|
||||||
|
ephemeralMessage: {
|
||||||
|
message: {
|
||||||
|
extendedTextMessage: {
|
||||||
|
text: "oh hey @Clawd UK !",
|
||||||
|
contextInfo: { mentionedJid: ["123@s.whatsapp.net"] },
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
};
|
||||||
|
|
||||||
|
sock.ev.emit("messages.upsert", upsert);
|
||||||
|
await new Promise((resolve) => setImmediate(resolve));
|
||||||
|
|
||||||
|
expect(onMessage).toHaveBeenCalledTimes(1);
|
||||||
|
expect(onMessage).toHaveBeenCalledWith(
|
||||||
|
expect.objectContaining({
|
||||||
|
chatType: "group",
|
||||||
|
conversationId: "424242@g.us",
|
||||||
|
body: "oh hey @Clawd UK !",
|
||||||
|
mentionedJids: ["123@s.whatsapp.net"],
|
||||||
|
senderE164: "+888",
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
|
||||||
|
await listener.close();
|
||||||
|
});
|
||||||
|
|
||||||
it("still forwards group messages (with sender info) even when allowFrom is restrictive", async () => {
|
it("still forwards group messages (with sender info) even when allowFrom is restrictive", async () => {
|
||||||
mockLoadConfig.mockReturnValue({
|
mockLoadConfig.mockReturnValue({
|
||||||
inbound: {
|
inbound: {
|
||||||
|
|||||||
Reference in New Issue
Block a user