fix: normalize outbound media payloads
This commit is contained in:
@@ -18,6 +18,7 @@ export const SendParamsSchema = Type.Object(
|
|||||||
to: NonEmptyString,
|
to: NonEmptyString,
|
||||||
message: NonEmptyString,
|
message: NonEmptyString,
|
||||||
mediaUrl: Type.Optional(Type.String()),
|
mediaUrl: Type.Optional(Type.String()),
|
||||||
|
mediaUrls: Type.Optional(Type.Array(Type.String())),
|
||||||
gifPlayback: Type.Optional(Type.Boolean()),
|
gifPlayback: Type.Optional(Type.Boolean()),
|
||||||
channel: Type.Optional(Type.String()),
|
channel: Type.Optional(Type.String()),
|
||||||
accountId: Type.Optional(Type.String()),
|
accountId: Type.Optional(Type.String()),
|
||||||
|
|||||||
@@ -104,4 +104,34 @@ describe("gateway send mirroring", () => {
|
|||||||
}),
|
}),
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it("mirrors MEDIA tags as attachments", async () => {
|
||||||
|
mocks.deliverOutboundPayloads.mockResolvedValue([{ messageId: "m2", channel: "slack" }]);
|
||||||
|
|
||||||
|
const respond = vi.fn();
|
||||||
|
await sendHandlers.send({
|
||||||
|
params: {
|
||||||
|
to: "channel:C1",
|
||||||
|
message: "Here\nMEDIA:https://example.com/image.png",
|
||||||
|
channel: "slack",
|
||||||
|
idempotencyKey: "idem-3",
|
||||||
|
sessionKey: "agent:main:main",
|
||||||
|
},
|
||||||
|
respond,
|
||||||
|
context: makeContext(),
|
||||||
|
req: { type: "req", id: "1", method: "send" },
|
||||||
|
client: null,
|
||||||
|
isWebchatConnect: () => false,
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(mocks.deliverOutboundPayloads).toHaveBeenCalledWith(
|
||||||
|
expect.objectContaining({
|
||||||
|
mirror: expect.objectContaining({
|
||||||
|
sessionKey: "agent:main:main",
|
||||||
|
text: "Here",
|
||||||
|
mediaUrls: ["https://example.com/image.png"],
|
||||||
|
}),
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ import { DEFAULT_CHAT_CHANNEL } from "../../channels/registry.js";
|
|||||||
import { loadConfig } from "../../config/config.js";
|
import { loadConfig } from "../../config/config.js";
|
||||||
import { createOutboundSendDeps } from "../../cli/deps.js";
|
import { createOutboundSendDeps } from "../../cli/deps.js";
|
||||||
import { deliverOutboundPayloads } from "../../infra/outbound/deliver.js";
|
import { deliverOutboundPayloads } from "../../infra/outbound/deliver.js";
|
||||||
|
import { normalizeReplyPayloadsForDelivery } from "../../infra/outbound/payloads.js";
|
||||||
import { resolveSessionAgentId } from "../../agents/agent-scope.js";
|
import { resolveSessionAgentId } from "../../agents/agent-scope.js";
|
||||||
import type { OutboundChannel } from "../../infra/outbound/targets.js";
|
import type { OutboundChannel } from "../../infra/outbound/targets.js";
|
||||||
import { resolveOutboundTarget } from "../../infra/outbound/targets.js";
|
import { resolveOutboundTarget } from "../../infra/outbound/targets.js";
|
||||||
@@ -57,6 +58,7 @@ export const sendHandlers: GatewayRequestHandlers = {
|
|||||||
to: string;
|
to: string;
|
||||||
message: string;
|
message: string;
|
||||||
mediaUrl?: string;
|
mediaUrl?: string;
|
||||||
|
mediaUrls?: string[];
|
||||||
gifPlayback?: boolean;
|
gifPlayback?: boolean;
|
||||||
channel?: string;
|
channel?: string;
|
||||||
accountId?: string;
|
accountId?: string;
|
||||||
@@ -82,6 +84,7 @@ export const sendHandlers: GatewayRequestHandlers = {
|
|||||||
}
|
}
|
||||||
const to = request.to.trim();
|
const to = request.to.trim();
|
||||||
const message = request.message.trim();
|
const message = request.message.trim();
|
||||||
|
const mediaUrls = Array.isArray(request.mediaUrls) ? request.mediaUrls : undefined;
|
||||||
const channelInput = typeof request.channel === "string" ? request.channel : undefined;
|
const channelInput = typeof request.channel === "string" ? request.channel : undefined;
|
||||||
const normalizedChannel = channelInput ? normalizeChannelId(channelInput) : null;
|
const normalizedChannel = channelInput ? normalizeChannelId(channelInput) : null;
|
||||||
if (channelInput && !normalizedChannel) {
|
if (channelInput && !normalizedChannel) {
|
||||||
@@ -126,12 +129,22 @@ export const sendHandlers: GatewayRequestHandlers = {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
const outboundDeps = context.deps ? createOutboundSendDeps(context.deps) : undefined;
|
const outboundDeps = context.deps ? createOutboundSendDeps(context.deps) : undefined;
|
||||||
|
const mirrorPayloads = normalizeReplyPayloadsForDelivery([
|
||||||
|
{ text: message, mediaUrl: request.mediaUrl, mediaUrls },
|
||||||
|
]);
|
||||||
|
const mirrorText = mirrorPayloads
|
||||||
|
.map((payload) => payload.text)
|
||||||
|
.filter(Boolean)
|
||||||
|
.join("\n");
|
||||||
|
const mirrorMediaUrls = mirrorPayloads.flatMap(
|
||||||
|
(payload) => payload.mediaUrls ?? (payload.mediaUrl ? [payload.mediaUrl] : []),
|
||||||
|
);
|
||||||
const results = await deliverOutboundPayloads({
|
const results = await deliverOutboundPayloads({
|
||||||
cfg,
|
cfg,
|
||||||
channel: outboundChannel,
|
channel: outboundChannel,
|
||||||
to: resolved.to,
|
to: resolved.to,
|
||||||
accountId,
|
accountId,
|
||||||
payloads: [{ text: message, mediaUrl: request.mediaUrl }],
|
payloads: [{ text: message, mediaUrl: request.mediaUrl, mediaUrls }],
|
||||||
gifPlayback: request.gifPlayback,
|
gifPlayback: request.gifPlayback,
|
||||||
deps: outboundDeps,
|
deps: outboundDeps,
|
||||||
mirror:
|
mirror:
|
||||||
@@ -142,8 +155,8 @@ export const sendHandlers: GatewayRequestHandlers = {
|
|||||||
sessionKey: request.sessionKey.trim(),
|
sessionKey: request.sessionKey.trim(),
|
||||||
config: cfg,
|
config: cfg,
|
||||||
}),
|
}),
|
||||||
text: message,
|
text: mirrorText || message,
|
||||||
mediaUrls: request.mediaUrl ? [request.mediaUrl] : undefined,
|
mediaUrls: mirrorMediaUrls.length > 0 ? mirrorMediaUrls : undefined,
|
||||||
}
|
}
|
||||||
: undefined,
|
: undefined,
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -201,13 +201,12 @@ describe("deliverOutboundPayloads", () => {
|
|||||||
it("normalizes payloads and drops empty entries", () => {
|
it("normalizes payloads and drops empty entries", () => {
|
||||||
const normalized = normalizeOutboundPayloads([
|
const normalized = normalizeOutboundPayloads([
|
||||||
{ text: "hi" },
|
{ text: "hi" },
|
||||||
{ mediaUrl: "https://x.test/a.jpg" },
|
{ text: "MEDIA:https://x.test/a.jpg" },
|
||||||
{ text: " ", mediaUrls: [] },
|
{ text: " ", mediaUrls: [] },
|
||||||
]);
|
]);
|
||||||
expect(normalized).toEqual([
|
expect(normalized).toEqual([
|
||||||
{ text: "hi", mediaUrls: [] },
|
{ text: "hi", mediaUrls: [] },
|
||||||
{ text: "", mediaUrls: ["https://x.test/a.jpg"] },
|
{ text: "", mediaUrls: ["https://x.test/a.jpg"] },
|
||||||
{ text: " ", mediaUrls: [] },
|
|
||||||
]);
|
]);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
@@ -586,12 +586,24 @@ async function handleSendAction(ctx: ResolvedActionContext): Promise<MessageActi
|
|||||||
}) ?? "";
|
}) ?? "";
|
||||||
|
|
||||||
const parsed = parseReplyDirectives(message);
|
const parsed = parseReplyDirectives(message);
|
||||||
|
const mergedMediaUrls: string[] = [];
|
||||||
|
const seenMedia = new Set<string>();
|
||||||
|
const pushMedia = (value?: string | null) => {
|
||||||
|
const trimmed = value?.trim();
|
||||||
|
if (!trimmed) return;
|
||||||
|
if (seenMedia.has(trimmed)) return;
|
||||||
|
seenMedia.add(trimmed);
|
||||||
|
mergedMediaUrls.push(trimmed);
|
||||||
|
};
|
||||||
|
pushMedia(mediaHint);
|
||||||
|
for (const url of parsed.mediaUrls ?? []) pushMedia(url);
|
||||||
|
pushMedia(parsed.mediaUrl);
|
||||||
message = parsed.text;
|
message = parsed.text;
|
||||||
params.message = message;
|
params.message = message;
|
||||||
if (!params.replyTo && parsed.replyToId) params.replyTo = parsed.replyToId;
|
if (!params.replyTo && parsed.replyToId) params.replyTo = parsed.replyToId;
|
||||||
if (!params.media) {
|
if (!params.media) {
|
||||||
// Use path/filePath if media not set, then fall back to parsed directives
|
// Use path/filePath if media not set, then fall back to parsed directives
|
||||||
params.media = mediaHint || parsed.mediaUrls?.[0] || parsed.mediaUrl || undefined;
|
params.media = mergedMediaUrls[0] || undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
message = await maybeApplyCrossContextMarker({
|
message = await maybeApplyCrossContextMarker({
|
||||||
@@ -630,6 +642,7 @@ async function handleSendAction(ctx: ResolvedActionContext): Promise<MessageActi
|
|||||||
to,
|
to,
|
||||||
message,
|
message,
|
||||||
mediaUrl: mediaUrl || undefined,
|
mediaUrl: mediaUrl || undefined,
|
||||||
|
mediaUrls: mergedMediaUrls.length ? mergedMediaUrls : undefined,
|
||||||
gifPlayback,
|
gifPlayback,
|
||||||
bestEffort: bestEffort ?? undefined,
|
bestEffort: bestEffort ?? undefined,
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -17,6 +17,7 @@ import {
|
|||||||
type OutboundDeliveryResult,
|
type OutboundDeliveryResult,
|
||||||
type OutboundSendDeps,
|
type OutboundSendDeps,
|
||||||
} from "./deliver.js";
|
} from "./deliver.js";
|
||||||
|
import { normalizeReplyPayloadsForDelivery } from "./payloads.js";
|
||||||
import type { OutboundChannel } from "./targets.js";
|
import type { OutboundChannel } from "./targets.js";
|
||||||
import { resolveOutboundTarget } from "./targets.js";
|
import { resolveOutboundTarget } from "./targets.js";
|
||||||
|
|
||||||
@@ -34,6 +35,7 @@ type MessageSendParams = {
|
|||||||
content: string;
|
content: string;
|
||||||
channel?: string;
|
channel?: string;
|
||||||
mediaUrl?: string;
|
mediaUrl?: string;
|
||||||
|
mediaUrls?: string[];
|
||||||
gifPlayback?: boolean;
|
gifPlayback?: boolean;
|
||||||
accountId?: string;
|
accountId?: string;
|
||||||
dryRun?: boolean;
|
dryRun?: boolean;
|
||||||
@@ -53,6 +55,7 @@ export type MessageSendResult = {
|
|||||||
to: string;
|
to: string;
|
||||||
via: "direct" | "gateway";
|
via: "direct" | "gateway";
|
||||||
mediaUrl: string | null;
|
mediaUrl: string | null;
|
||||||
|
mediaUrls?: string[];
|
||||||
result?: OutboundDeliveryResult | { messageId: string };
|
result?: OutboundDeliveryResult | { messageId: string };
|
||||||
dryRun?: boolean;
|
dryRun?: boolean;
|
||||||
};
|
};
|
||||||
@@ -115,13 +118,29 @@ export async function sendMessage(params: MessageSendParams): Promise<MessageSen
|
|||||||
throw new Error(`Unknown channel: ${channel}`);
|
throw new Error(`Unknown channel: ${channel}`);
|
||||||
}
|
}
|
||||||
const deliveryMode = plugin.outbound?.deliveryMode ?? "direct";
|
const deliveryMode = plugin.outbound?.deliveryMode ?? "direct";
|
||||||
|
const normalizedPayloads = normalizeReplyPayloadsForDelivery([
|
||||||
|
{
|
||||||
|
text: params.content,
|
||||||
|
mediaUrl: params.mediaUrl,
|
||||||
|
mediaUrls: params.mediaUrls,
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
const mirrorText = normalizedPayloads
|
||||||
|
.map((payload) => payload.text)
|
||||||
|
.filter(Boolean)
|
||||||
|
.join("\n");
|
||||||
|
const mirrorMediaUrls = normalizedPayloads.flatMap(
|
||||||
|
(payload) => payload.mediaUrls ?? (payload.mediaUrl ? [payload.mediaUrl] : []),
|
||||||
|
);
|
||||||
|
const primaryMediaUrl = mirrorMediaUrls[0] ?? params.mediaUrl ?? null;
|
||||||
|
|
||||||
if (params.dryRun) {
|
if (params.dryRun) {
|
||||||
return {
|
return {
|
||||||
channel,
|
channel,
|
||||||
to: params.to,
|
to: params.to,
|
||||||
via: deliveryMode === "gateway" ? "gateway" : "direct",
|
via: deliveryMode === "gateway" ? "gateway" : "direct",
|
||||||
mediaUrl: params.mediaUrl ?? null,
|
mediaUrl: primaryMediaUrl,
|
||||||
|
mediaUrls: mirrorMediaUrls.length ? mirrorMediaUrls : undefined,
|
||||||
dryRun: true,
|
dryRun: true,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
@@ -142,15 +161,15 @@ export async function sendMessage(params: MessageSendParams): Promise<MessageSen
|
|||||||
channel: outboundChannel,
|
channel: outboundChannel,
|
||||||
to: resolvedTarget.to,
|
to: resolvedTarget.to,
|
||||||
accountId: params.accountId,
|
accountId: params.accountId,
|
||||||
payloads: [{ text: params.content, mediaUrl: params.mediaUrl }],
|
payloads: normalizedPayloads,
|
||||||
gifPlayback: params.gifPlayback,
|
gifPlayback: params.gifPlayback,
|
||||||
deps: params.deps,
|
deps: params.deps,
|
||||||
bestEffort: params.bestEffort,
|
bestEffort: params.bestEffort,
|
||||||
mirror: params.mirror
|
mirror: params.mirror
|
||||||
? {
|
? {
|
||||||
...params.mirror,
|
...params.mirror,
|
||||||
text: params.content,
|
text: mirrorText || params.content,
|
||||||
mediaUrls: params.mediaUrl ? [params.mediaUrl] : undefined,
|
mediaUrls: mirrorMediaUrls.length ? mirrorMediaUrls : undefined,
|
||||||
}
|
}
|
||||||
: undefined,
|
: undefined,
|
||||||
});
|
});
|
||||||
@@ -159,7 +178,8 @@ export async function sendMessage(params: MessageSendParams): Promise<MessageSen
|
|||||||
channel,
|
channel,
|
||||||
to: params.to,
|
to: params.to,
|
||||||
via: "direct",
|
via: "direct",
|
||||||
mediaUrl: params.mediaUrl ?? null,
|
mediaUrl: primaryMediaUrl,
|
||||||
|
mediaUrls: mirrorMediaUrls.length ? mirrorMediaUrls : undefined,
|
||||||
result: results.at(-1),
|
result: results.at(-1),
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
@@ -173,6 +193,7 @@ export async function sendMessage(params: MessageSendParams): Promise<MessageSen
|
|||||||
to: params.to,
|
to: params.to,
|
||||||
message: params.content,
|
message: params.content,
|
||||||
mediaUrl: params.mediaUrl,
|
mediaUrl: params.mediaUrl,
|
||||||
|
mediaUrls: mirrorMediaUrls.length ? mirrorMediaUrls : params.mediaUrls,
|
||||||
gifPlayback: params.gifPlayback,
|
gifPlayback: params.gifPlayback,
|
||||||
accountId: params.accountId,
|
accountId: params.accountId,
|
||||||
channel,
|
channel,
|
||||||
@@ -189,7 +210,8 @@ export async function sendMessage(params: MessageSendParams): Promise<MessageSen
|
|||||||
channel,
|
channel,
|
||||||
to: params.to,
|
to: params.to,
|
||||||
via: "gateway",
|
via: "gateway",
|
||||||
mediaUrl: params.mediaUrl ?? null,
|
mediaUrl: primaryMediaUrl,
|
||||||
|
mediaUrls: mirrorMediaUrls.length ? mirrorMediaUrls : undefined,
|
||||||
result,
|
result,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -58,6 +58,7 @@ export async function executeSendAction(params: {
|
|||||||
to: string;
|
to: string;
|
||||||
message: string;
|
message: string;
|
||||||
mediaUrl?: string;
|
mediaUrl?: string;
|
||||||
|
mediaUrls?: string[];
|
||||||
gifPlayback?: boolean;
|
gifPlayback?: boolean;
|
||||||
bestEffort?: boolean;
|
bestEffort?: boolean;
|
||||||
}): Promise<{
|
}): Promise<{
|
||||||
@@ -91,6 +92,7 @@ export async function executeSendAction(params: {
|
|||||||
to: params.to,
|
to: params.to,
|
||||||
content: params.message,
|
content: params.message,
|
||||||
mediaUrl: params.mediaUrl || undefined,
|
mediaUrl: params.mediaUrl || undefined,
|
||||||
|
mediaUrls: params.mediaUrls,
|
||||||
channel: params.ctx.channel || undefined,
|
channel: params.ctx.channel || undefined,
|
||||||
accountId: params.ctx.accountId ?? undefined,
|
accountId: params.ctx.accountId ?? undefined,
|
||||||
gifPlayback: params.gifPlayback,
|
gifPlayback: params.gifPlayback,
|
||||||
|
|||||||
@@ -24,6 +24,22 @@ describe("normalizeOutboundPayloadsForJson", () => {
|
|||||||
},
|
},
|
||||||
]);
|
]);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it("keeps mediaUrl null for multi MEDIA tags", () => {
|
||||||
|
expect(
|
||||||
|
normalizeOutboundPayloadsForJson([
|
||||||
|
{
|
||||||
|
text: "MEDIA:https://x.test/a.png\nMEDIA:https://x.test/b.png",
|
||||||
|
},
|
||||||
|
]),
|
||||||
|
).toEqual([
|
||||||
|
{
|
||||||
|
text: "",
|
||||||
|
mediaUrl: null,
|
||||||
|
mediaUrls: ["https://x.test/a.png", "https://x.test/b.png"],
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe("formatOutboundPayloadLog", () => {
|
describe("formatOutboundPayloadLog", () => {
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
import { parseReplyDirectives } from "../../auto-reply/reply/reply-directives.js";
|
||||||
|
import { isRenderablePayload } from "../../auto-reply/reply/reply-payloads.js";
|
||||||
import type { ReplyPayload } from "../../auto-reply/types.js";
|
import type { ReplyPayload } from "../../auto-reply/types.js";
|
||||||
|
|
||||||
export type NormalizedOutboundPayload = {
|
export type NormalizedOutboundPayload = {
|
||||||
@@ -11,8 +13,51 @@ export type OutboundPayloadJson = {
|
|||||||
mediaUrls?: string[];
|
mediaUrls?: string[];
|
||||||
};
|
};
|
||||||
|
|
||||||
|
function mergeMediaUrls(...lists: Array<Array<string | undefined> | undefined>): string[] {
|
||||||
|
const seen = new Set<string>();
|
||||||
|
const merged: string[] = [];
|
||||||
|
for (const list of lists) {
|
||||||
|
if (!list) continue;
|
||||||
|
for (const entry of list) {
|
||||||
|
const trimmed = entry?.trim();
|
||||||
|
if (!trimmed) continue;
|
||||||
|
if (seen.has(trimmed)) continue;
|
||||||
|
seen.add(trimmed);
|
||||||
|
merged.push(trimmed);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return merged;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function normalizeReplyPayloadsForDelivery(payloads: ReplyPayload[]): ReplyPayload[] {
|
||||||
|
return payloads.flatMap((payload) => {
|
||||||
|
const parsed = parseReplyDirectives(payload.text ?? "");
|
||||||
|
const explicitMediaUrls = payload.mediaUrls ?? parsed.mediaUrls;
|
||||||
|
const explicitMediaUrl = payload.mediaUrl ?? parsed.mediaUrl;
|
||||||
|
const mergedMedia = mergeMediaUrls(
|
||||||
|
explicitMediaUrls,
|
||||||
|
explicitMediaUrl ? [explicitMediaUrl] : undefined,
|
||||||
|
);
|
||||||
|
const hasMultipleMedia = (explicitMediaUrls?.length ?? 0) > 1;
|
||||||
|
const resolvedMediaUrl = hasMultipleMedia ? undefined : explicitMediaUrl;
|
||||||
|
const next: ReplyPayload = {
|
||||||
|
...payload,
|
||||||
|
text: parsed.text ?? "",
|
||||||
|
mediaUrls: mergedMedia.length ? mergedMedia : undefined,
|
||||||
|
mediaUrl: resolvedMediaUrl,
|
||||||
|
replyToId: payload.replyToId ?? parsed.replyToId,
|
||||||
|
replyToTag: payload.replyToTag || parsed.replyToTag,
|
||||||
|
replyToCurrent: payload.replyToCurrent || parsed.replyToCurrent,
|
||||||
|
audioAsVoice: Boolean(payload.audioAsVoice || parsed.audioAsVoice),
|
||||||
|
};
|
||||||
|
if (parsed.isSilent && mergedMedia.length === 0) return [];
|
||||||
|
if (!isRenderablePayload(next)) return [];
|
||||||
|
return [next];
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
export function normalizeOutboundPayloads(payloads: ReplyPayload[]): NormalizedOutboundPayload[] {
|
export function normalizeOutboundPayloads(payloads: ReplyPayload[]): NormalizedOutboundPayload[] {
|
||||||
return payloads
|
return normalizeReplyPayloadsForDelivery(payloads)
|
||||||
.map((payload) => ({
|
.map((payload) => ({
|
||||||
text: payload.text ?? "",
|
text: payload.text ?? "",
|
||||||
mediaUrls: payload.mediaUrls ?? (payload.mediaUrl ? [payload.mediaUrl] : []),
|
mediaUrls: payload.mediaUrls ?? (payload.mediaUrl ? [payload.mediaUrl] : []),
|
||||||
@@ -21,7 +66,7 @@ export function normalizeOutboundPayloads(payloads: ReplyPayload[]): NormalizedO
|
|||||||
}
|
}
|
||||||
|
|
||||||
export function normalizeOutboundPayloadsForJson(payloads: ReplyPayload[]): OutboundPayloadJson[] {
|
export function normalizeOutboundPayloadsForJson(payloads: ReplyPayload[]): OutboundPayloadJson[] {
|
||||||
return payloads.map((payload) => ({
|
return normalizeReplyPayloadsForDelivery(payloads).map((payload) => ({
|
||||||
text: payload.text ?? "",
|
text: payload.text ?? "",
|
||||||
mediaUrl: payload.mediaUrl ?? null,
|
mediaUrl: payload.mediaUrl ?? null,
|
||||||
mediaUrls: payload.mediaUrls ?? (payload.mediaUrl ? [payload.mediaUrl] : undefined),
|
mediaUrls: payload.mediaUrls ?? (payload.mediaUrl ? [payload.mediaUrl] : undefined),
|
||||||
|
|||||||
Reference in New Issue
Block a user