惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

博客园 - 司徒正美
大猫的无限游戏
大猫的无限游戏
Scott Helme
Scott Helme
CTFtime.org: upcoming CTF events
CTFtime.org: upcoming CTF events
S
Secure Thoughts
Google DeepMind News
Google DeepMind News
博客园_首页
Hacker News: Ask HN
Hacker News: Ask HN
量子位
Jina AI
Jina AI
I
InfoQ
V
V2EX
Martin Fowler
Martin Fowler
Y
Y Combinator Blog
H
Hackread – Cybersecurity News, Data Breaches, AI and More
人人都是产品经理
人人都是产品经理
B
Blog
IT之家
IT之家
云风的 BLOG
云风的 BLOG
奇客Solidot–传递最新科技情报
奇客Solidot–传递最新科技情报
博客园 - Franky
博客园 - 【当耐特】
N
Netflix TechBlog - Medium
Cloudbric
Cloudbric
H
Heimdal Security Blog
TaoSecurity Blog
TaoSecurity Blog
S
Security @ Cisco Blogs
U
Unit 42
Project Zero
Project Zero
Webroot Blog
Webroot Blog
The Register - Security
The Register - Security
N
News | PayPal Newsroom
Microsoft Security Blog
Microsoft Security Blog
H
Help Net Security
Forbes - Security
Forbes - Security
宝玉的分享
宝玉的分享
Last Week in AI
Last Week in AI
C
Check Point Blog
博客园 - 聂微东
M
MIT News - Artificial intelligence
有赞技术团队
有赞技术团队
D
DataBreaches.Net
Cyberwarzone
Cyberwarzone
N
News and Events Feed by Topic
N
News and Events Feed by Topic
Simon Willison's Weblog
Simon Willison's Weblog
J
Java Code Geeks
G
Google Developers Blog
GbyAI
GbyAI
T
Threatpost

Recent Commits to openclaw:main

test: merge chat side-result checks · openclaw/openclaw@ddd2c2a test: merge cron history checks · openclaw/openclaw@f7eb746 test: merge responsive navigation shell checks · openclaw/openclaw@c2e4b47 docs(changelog): add codex oauth fixes · openclaw/openclaw@628e6cd test: merge navigation routing cases · openclaw/openclaw@5d8cecb Tests: mock channel registry bundled fallback · openclaw/openclaw@2b08233 Secrets: avoid broad web search discovery for single plugin config · openclaw/openclaw@a464f59 test: merge config view browser checks · openclaw/openclaw@20cf511 fix(status): align oauth health with runtime · openclaw/openclaw@eed7116 feat: add macOS screen snapshots for monitor preview (#67954) thanks … · openclaw/openclaw@f377db1 fix: report shared auth scopes in hello-ok (#67810) thanks @BunsDev · openclaw/openclaw@0b6c39b Auto-reply: avoid eager bundled route fallback · openclaw/openclaw@3ea1bf4 Tests: narrow session binding contract setup · openclaw/openclaw@54e4e16 fix(macOS): enable undo/redo in webchat composer text input (#34962) · openclaw/openclaw@00951dc Tests: speed up channel setup promotion · openclaw/openclaw@82b529a Docs: refresh agent instructions · openclaw/openclaw@5775fe2 fix(auth): serialize OAuth refresh across agents to fix #26322 (#67876) · openclaw/openclaw@8e79080 test: allow ollama public surface boundary test · openclaw/openclaw@7d4f1a6 Docs: add test performance guardrails · openclaw/openclaw@89706d3 Tests: restore context-engine usage proof · openclaw/openclaw@e4c4f95 Tests: slim context engine runtime coverage · openclaw/openclaw@74c198f ci: retry failed custom checkouts · openclaw/openclaw@0ee5baf test: trim duplicate provider auth onboarding cases · openclaw/openclaw@1ffc02e matrix: fix sessions_spawn --thread subagent session spawning (#67643) · openclaw/openclaw@1ce2596 test: reduce auth choice fixture churn · openclaw/openclaw@857b9cd test: mock health status config boundaries · openclaw/openclaw@9d5ab4a test: mock onboard config io boundary · openclaw/openclaw@299694d test: mock legacy state plugin boundaries · openclaw/openclaw@2713089 test: mock channel install boundaries · openclaw/openclaw@b945248 test: mock doctor preview channel boundaries · openclaw/openclaw@b1a3ad4 test: trim doctor command hotspots · openclaw/openclaw@c66f16a test: isolate agent auth and spawn hotspots · openclaw/openclaw@9285935 test: stabilize MCP startup disposal race · openclaw/openclaw@dd9d2eb test: merge browser contract server suites · openclaw/openclaw@5817a76 test: narrow ollama provider discovery setup · openclaw/openclaw@a0d9598 build: declare qa-lab aimock runtime dependency · openclaw/openclaw@24431e5 test: speed up safe-bins exec harness · openclaw/openclaw@ee856ab test: preserve tool helpers in embedded runner mocks · openclaw/openclaw@acd86a0 refactor: move memory embeddings into provider plugins · openclaw/openclaw@77e6e4c test: reuse system-run temp fixtures · openclaw/openclaw@7e9ff0f test: trim hotspot wait overhead · openclaw/openclaw@12a59b0 Check: avoid duplicate boundary prep · openclaw/openclaw@baf11b8 test: reduce hotspot fixture overhead · openclaw/openclaw@3a59edd feat(ui): overhaul settings and slash command UX (#67819) thanks @Bun… · openclaw/openclaw@2cfb660 QA Matrix: exit cleanly on failure · openclaw/openclaw@42805d2 QA Matrix: isolate scenario coverage · openclaw/openclaw@7e659e1 Matrix: refresh crypto bootstrap state · openclaw/openclaw@94081d8 QA Lab: add provider registry · openclaw/openclaw@bb7e982 Matrix: add plugin changelog · openclaw/openclaw@4acab55 test: trim more hotspot overhead · openclaw/openclaw@f485311 test: trim remaining hotspot tests · openclaw/openclaw@6ba8626 test: narrow hotspot mocks · openclaw/openclaw@dbc8179 test: isolate gemini embedding request helpers · openclaw/openclaw@cd330f5 test: trim memory and mcp hotspots · openclaw/openclaw@fd48dfa test: slim provider registry mocks · openclaw/openclaw@2e08c77 test: harden Parallels update smoke · openclaw/openclaw@1a98090 feat: default Anthropic to Opus 4.7 · openclaw/openclaw@628b454 fix: harden node-host shell payload mutability checks · openclaw/openclaw@75c551e fix: land node-host approval binding for native binaries (#66731) (th… · openclaw/openclaw@29919bb CI: add daily schedule to CodeQL workflow (#67645) · openclaw/openclaw@69d25f5 fix(gateway): capture config hash after plugin auto-enable to prevent… · openclaw/openclaw@8c11210 fix: repair sanitized replay tool results before send (#67620) (thank… · openclaw/openclaw@c3c7a99 fix: restrict HTML timeout short-circuit to transient statuses · openclaw/openclaw@de129a6 fix: keep TUI watchdog bound to active run (#67401) (thanks @xantorres) · openclaw/openclaw@3525273 Gateway/skills: dedupe skills prefix-match + drop dead fallback on log · openclaw/openclaw@d7f489f Extensions/lmstudio: back off inference preload after consecutive fai… · openclaw/openclaw@b555214 TUI/streaming: add watchdog that resets the activity indicator after … · openclaw/openclaw@f44ab20 Agents/tool-loop: enable unknown-tool stream guard by default · openclaw/openclaw@36ed367 Gateway/skills: invalidate session skills snapshot on config write · openclaw/openclaw@b23d59a fix: classify HTML provider error pages correctly (#67642) (thanks @s… · openclaw/openclaw@e588e90 fix(skills): remove unused model-usage import (#67641) · openclaw/openclaw@55f05df docs(changelog): credit codex fix superseded PRs · openclaw/openclaw@e485f24 fix(openai-codex): normalize stale transport metadata in resolution a… · openclaw/openclaw@90801ba CI: pin Docker-related GitHub Actions (#67632) · openclaw/openclaw@f697b01 Android: modernize WebView and discovery API usage (#67627) · openclaw/openclaw@44a6e50 fix(deps): bump hono to 4.12.14 and @hono/node-server to 1.19.14 (GHS… · openclaw/openclaw@fbccc18 fix(deps): bump dompurify to 3.4.0 (#67614) · openclaw/openclaw@2c2dc00 CI: add explicit permissions to all workflow jobs (fixes code-scannin… · openclaw/openclaw@01b7516 fix: register bundled TTS providers and route overrides correctly (#6… · openclaw/openclaw@6ea3cdd fix: align host tilde paths with OS home (#62804) (thanks @stainlu) · openclaw/openclaw@ecfaf64 fix: flush creds queue before reconnect socket open (#67464) (thanks … · openclaw/openclaw@405c63f fix: strip standalone <function> tool call tags from visible text (#6… · openclaw/openclaw@78df859 fix(agents): preserve cli session metadata before transcript persist … · openclaw/openclaw@898fd04 docs(changelog): move cli transcript entry · openclaw/openclaw@c1817c6 fix(agents): normalize cli transcript api field · openclaw/openclaw@3a3fae0 docs(changelog): note cli transcript persistence · openclaw/openclaw@6c343f1 fix(agents): persist cli transcript turns · openclaw/openclaw@b8ef507 fix(msteams): harden security-sensitive flows (#65841) · openclaw/openclaw@c56b56e [Dashboard] Fix exec approval modal overflow for long command content… · openclaw/openclaw@053c5b0 Docs: remove QA changelog entry · openclaw/openclaw@7fd5771 QA: fix private runtime source loading (#67428) · openclaw/openclaw@d5933af docs(gateway): correct protocol.md schema path, hello-ok example, aut… · openclaw/openclaw@489404d CI: pin Node 22 runners to 22.18.0 · openclaw/openclaw@4ffa621 models.authStatus: normalize provider ids + tighten env-backed escape… · openclaw/openclaw@f2fdb9d Update CHANGELOG.md · openclaw/openclaw@7694a92 test(parallels): clean up npm update guard jobs · openclaw/openclaw@045ea7b Plugins: prefer scanDir override paths · openclaw/openclaw@b2974da fix(dreaming): default storage.mode to "separate" so phase blocks sto… · openclaw/openclaw@8c392f0 fix(memory-core): skip dreaming transcript ingestion via session stor… · openclaw/openclaw@a1b01f0 fix: dedupe replayed exec.finished node events (#67281) · openclaw/openclaw@5dcf526
perf: slim slack media test imports · openclaw/openclaw@0999fec
steipete · 2026-04-24 · via Recent Commits to openclaw:main

@@ -1,11 +1,4 @@

1-

import * as ssrf from "openclaw/plugin-sdk/infra-runtime";

2-

import * as mediaFetch from "openclaw/plugin-sdk/media-runtime";

3-

import type { SavedMedia } from "openclaw/plugin-sdk/media-runtime";

4-

import * as mediaStore from "openclaw/plugin-sdk/media-runtime";

5-

import { logVerbose } from "openclaw/plugin-sdk/runtime-env";

6-

import { type FetchMock, withFetchPreconnect } from "openclaw/plugin-sdk/testing";

71

import { afterEach, beforeEach, describe, expect, it, vi } from "vitest";

8-

import { mockPinnedHostnameResolution } from "../../../../src/test-helpers/ssrf.js";

92

import {

103

fetchWithSlackAuth,

114

resolveSlackAttachmentContent,

@@ -14,16 +7,84 @@ import {

147

resolveSlackThreadStarter,

158

resetSlackThreadStarterCacheForTest,

169

} from "./media.js";

17-18-

vi.mock("openclaw/plugin-sdk/runtime-env", () => ({

19-

logVerbose: vi.fn(),

20-

danger: (message: string) => message,

21-

shouldLogVerbose: () => false,

10+

import type { FetchLike, SavedMedia } from "./media.runtime.js";

11+

import * as mediaRuntime from "./media.runtime.js";

12+

import { logVerbose } from "./media.runtime.js";

13+14+

type FetchMock = (input: RequestInfo | URL, init?: RequestInit) => Promise<Response>;

15+16+

const fetchRemoteMediaMock = vi.hoisted(() =>

17+

vi.fn(

18+

async (params: {

19+

url: string;

20+

fetchImpl: FetchLike;

21+

filePathHint?: string;

22+

requestInit?: RequestInit;

23+

}) => {

24+

let response = await params.fetchImpl(params.url, {

25+

...params.requestInit,

26+

dispatcher: {},

27+

} as RequestInit & { dispatcher: unknown });

28+

if (response.status >= 300 && response.status < 400) {

29+

const location = response.headers.get("location");

30+

if (location) {

31+

const source = new URL(params.url);

32+

const redirect = new URL(location, source);

33+

const sameOrigin = redirect.origin === source.origin;

34+

response = await params.fetchImpl(redirect.toString(), {

35+

...(sameOrigin ? params.requestInit : {}),

36+

redirect: "follow",

37+

dispatcher: {},

38+

} as RequestInit & { dispatcher: unknown });

39+

}

40+

}

41+

if (response.status < 200 || response.status >= 300) {

42+

throw new Error(`fetch failed: ${response.status}`);

43+

}

44+

return {

45+

buffer: Buffer.from(await response.arrayBuffer()),

46+

contentType: response.headers.get("content-type") ?? undefined,

47+

fileName: params.filePathHint ?? new URL(params.url).pathname.split("/").at(-1),

48+

};

49+

},

50+

),

51+

);

52+

const saveMediaBufferMock = vi.hoisted(() =>

53+

vi.fn(async (_buffer: Buffer, contentType?: string) => ({

54+

id: "saved-media-id",

55+

path: "/tmp/test.bin",

56+

size: _buffer.byteLength,

57+

contentType,

58+

})),

59+

);

60+

const fetchWithRuntimeDispatcherMock = vi.hoisted(() => vi.fn());

61+

const logVerboseMock = vi.hoisted(() => vi.fn());

62+63+

vi.mock("./media.runtime.js", () => ({

64+

fetchRemoteMedia: fetchRemoteMediaMock,

65+

fetchWithRuntimeDispatcher: fetchWithRuntimeDispatcherMock,

66+

logVerbose: logVerboseMock,

67+

saveMediaBuffer: saveMediaBufferMock,

2268

}));

236970+

function withFetchPreconnect(fetchMock: ReturnType<typeof vi.fn<FetchMock>>): typeof fetch {

71+

return Object.assign(

72+

((input: RequestInfo | URL, init?: RequestInit) => fetchMock(input, init)) as typeof fetch,

73+

{ mock: fetchMock.mock },

74+

);

75+

}

76+2477

// Store original fetch

2578

const originalFetch = globalThis.fetch;

2679

let mockFetch: ReturnType<typeof vi.fn<FetchMock>>;

80+81+

beforeEach(() => {

82+

fetchRemoteMediaMock.mockClear();

83+

fetchWithRuntimeDispatcherMock.mockClear();

84+

logVerboseMock.mockClear();

85+

saveMediaBufferMock.mockClear();

86+

});

87+2788

const createSavedMedia = (filePath: string, contentType: string): SavedMedia => ({

2889

id: "saved-media-id",

2990

path: filePath,

@@ -41,7 +102,7 @@ async function expectPrivateDownloadRedirect(params: {

41102

redirectedUrl: string;

42103

secondAuthorization: string | null;

43104

}) {

44-

vi.spyOn(mediaStore, "saveMediaBuffer").mockResolvedValue(

105+

vi.spyOn(mediaRuntime, "saveMediaBuffer").mockResolvedValue(

45106

createSavedMedia("/tmp/test.jpg", "image/jpeg"),

46107

);

47108

@@ -225,7 +286,6 @@ describe("resolveSlackMedia", () => {

225286

beforeEach(() => {

226287

mockFetch = vi.fn();

227288

globalThis.fetch = mockFetch as unknown as typeof fetch;

228-

mockPinnedHostnameResolution();

229289

});

230290231291

afterEach(() => {

@@ -234,7 +294,7 @@ describe("resolveSlackMedia", () => {

234294

});

235295236296

it("prefers url_private_download over url_private", async () => {

237-

vi.spyOn(mediaStore, "saveMediaBuffer").mockResolvedValue(

297+

vi.spyOn(mediaRuntime, "saveMediaBuffer").mockResolvedValue(

238298

createSavedMedia("/tmp/test.jpg", "image/jpeg"),

239299

);

240300

@@ -313,7 +373,7 @@ describe("resolveSlackMedia", () => {

313373

});

314374315375

it("rejects HTML auth pages for non-HTML files", async () => {

316-

const saveMediaBufferMock = vi.spyOn(mediaStore, "saveMediaBuffer");

376+

const saveMediaBufferMock = vi.spyOn(mediaRuntime, "saveMediaBuffer");

317377

mockFetch.mockResolvedValueOnce(

318378

new Response("<!DOCTYPE html><html><body>login</body></html>", {

319379

status: 200,

@@ -332,7 +392,7 @@ describe("resolveSlackMedia", () => {

332392

});

333393334394

it("allows expected HTML uploads", async () => {

335-

vi.spyOn(mediaStore, "saveMediaBuffer").mockResolvedValue(

395+

vi.spyOn(mediaRuntime, "saveMediaBuffer").mockResolvedValue(

336396

createSavedMedia("/tmp/page.html", "text/html"),

337397

);

338398

mockFetch.mockResolvedValueOnce(

@@ -363,7 +423,7 @@ describe("resolveSlackMedia", () => {

363423

// video/mp4 for MP4 containers. Verify resolveSlackMedia preserves

364424

// the overridden audio/* type in its return value despite this.

365425

const saveMediaBufferMock = vi

366-

.spyOn(mediaStore, "saveMediaBuffer")

426+

.spyOn(mediaRuntime, "saveMediaBuffer")

367427

.mockResolvedValue(createSavedMedia("/tmp/voice.mp4", "video/mp4"));

368428369429

const mockResponse = new Response(Buffer.from("audio data"), {

@@ -401,7 +461,7 @@ describe("resolveSlackMedia", () => {

401461402462

it("preserves original MIME for non-voice Slack files", async () => {

403463

const saveMediaBufferMock = vi

404-

.spyOn(mediaStore, "saveMediaBuffer")

464+

.spyOn(mediaRuntime, "saveMediaBuffer")

405465

.mockResolvedValue(createSavedMedia("/tmp/video.mp4", "video/mp4"));

406466407467

const mockResponse = new Response(Buffer.from("video data"), {

@@ -434,7 +494,7 @@ describe("resolveSlackMedia", () => {

434494

});

435495436496

it("falls through to next file when first file returns error", async () => {

437-

vi.spyOn(mediaStore, "saveMediaBuffer").mockResolvedValue(

497+

vi.spyOn(mediaRuntime, "saveMediaBuffer").mockResolvedValue(

438498

createSavedMedia("/tmp/test.jpg", "image/jpeg"),

439499

);

440500

@@ -463,7 +523,7 @@ describe("resolveSlackMedia", () => {

463523

});

464524465525

it("returns all successfully downloaded files as an array", async () => {

466-

vi.spyOn(mediaStore, "saveMediaBuffer").mockImplementation(async (buffer, _contentType) => {

526+

vi.spyOn(mediaRuntime, "saveMediaBuffer").mockImplementation(async (buffer, _contentType) => {

467527

const text = Buffer.from(buffer).toString("utf8");

468528

if (text.includes("image a")) {

469529

return createSavedMedia("/tmp/a.jpg", "image/jpeg");

@@ -510,7 +570,7 @@ describe("resolveSlackMedia", () => {

510570511571

it("caps downloads to 8 files for large multi-attachment messages", async () => {

512572

const saveMediaBufferMock = vi

513-

.spyOn(mediaStore, "saveMediaBuffer")

573+

.spyOn(mediaRuntime, "saveMediaBuffer")

514574

.mockResolvedValue(createSavedMedia("/tmp/x.jpg", "image/jpeg"));

515575516576

mockFetch.mockImplementation(async () => {

@@ -539,14 +599,14 @@ describe("resolveSlackMedia", () => {

539599

});

540600541601

it("routes dispatcher-backed Slack media requests through runtime fetch", async () => {

542-

vi.spyOn(mediaStore, "saveMediaBuffer").mockResolvedValue(

602+

vi.spyOn(mediaRuntime, "saveMediaBuffer").mockResolvedValue(

543603

createSavedMedia("/tmp/test.jpg", "image/jpeg"),

544604

);

545605

globalThis.fetch = (async () => {

546606

throw new Error("global fetch should not receive dispatcher-backed Slack media requests");

547607

}) as typeof fetch;

548608

const runtimeFetchSpy = vi

549-

.spyOn(ssrf, "fetchWithRuntimeDispatcher")

609+

.spyOn(mediaRuntime, "fetchWithRuntimeDispatcher")

550610

.mockImplementation(async () => {

551611

return new Response(Buffer.from("image data"), {

552612

status: 200,

@@ -578,7 +638,6 @@ describe("Slack media SSRF policy", () => {

578638

beforeEach(() => {

579639

mockFetch = vi.fn();

580640

globalThis.fetch = withFetchPreconnect(mockFetch);

581-

mockPinnedHostnameResolution();

582641

});

583642584643

afterEach(() => {

@@ -587,14 +646,14 @@ describe("Slack media SSRF policy", () => {

587646

});

588647589648

it("passes ssrfPolicy with Slack CDN allowedHostnames and allowRfc2544BenchmarkRange to file downloads", async () => {

590-

vi.spyOn(mediaStore, "saveMediaBuffer").mockResolvedValue(

649+

vi.spyOn(mediaRuntime, "saveMediaBuffer").mockResolvedValue(

591650

createSavedMedia("/tmp/test.jpg", "image/jpeg"),

592651

);

593652

mockFetch.mockResolvedValueOnce(

594653

new Response(Buffer.from("img"), { status: 200, headers: { "content-type": "image/jpeg" } }),

595654

);

596655597-

const spy = vi.spyOn(mediaFetch, "fetchRemoteMedia");

656+

const spy = vi.spyOn(mediaRuntime, "fetchRemoteMedia");

598657599658

await resolveSlackMedia({

600659

files: [{ url_private: "https://files.slack.com/test.jpg", name: "test.jpg" }],

@@ -615,22 +674,14 @@ describe("Slack media SSRF policy", () => {

615674

});

616675617676

it("passes ssrfPolicy to forwarded attachment image downloads", async () => {

618-

vi.spyOn(mediaStore, "saveMediaBuffer").mockResolvedValue(

677+

vi.spyOn(mediaRuntime, "saveMediaBuffer").mockResolvedValue(

619678

createSavedMedia("/tmp/fwd.jpg", "image/jpeg"),

620679

);

621-

vi.spyOn(ssrf, "resolvePinnedHostnameWithPolicy").mockImplementation(async (hostname) => {

622-

const normalized = hostname.trim().toLowerCase().replace(/\.$/, "");

623-

return {

624-

hostname: normalized,

625-

addresses: ["93.184.216.34"],

626-

lookup: ssrf.createPinnedLookup({ hostname: normalized, addresses: ["93.184.216.34"] }),

627-

};

628-

});

629680

mockFetch.mockResolvedValueOnce(

630681

new Response(Buffer.from("fwd"), { status: 200, headers: { "content-type": "image/jpeg" } }),

631682

);

632683633-

const spy = vi.spyOn(mediaFetch, "fetchRemoteMedia");

684+

const spy = vi.spyOn(mediaRuntime, "fetchRemoteMedia");

634685635686

await resolveSlackAttachmentContent({

636687

attachments: [{ is_share: true, image_url: "https://files.slack.com/forwarded.jpg" }],

@@ -650,7 +701,6 @@ describe("resolveSlackAttachmentContent", () => {

650701

beforeEach(() => {

651702

mockFetch = vi.fn();

652703

globalThis.fetch = mockFetch as unknown as typeof fetch;

653-

mockPinnedHostnameResolution();

654704

});

655705656706

afterEach(() => {

@@ -696,7 +746,7 @@ describe("resolveSlackAttachmentContent", () => {

696746

});

697747698748

it("skips forwarded image URLs on non-Slack hosts", async () => {

699-

const saveMediaBufferMock = vi.spyOn(mediaStore, "saveMediaBuffer");

749+

const saveMediaBufferMock = vi.spyOn(mediaRuntime, "saveMediaBuffer");

700750701751

const result = await resolveSlackAttachmentContent({

702752

attachments: [{ is_share: true, image_url: "https://example.com/forwarded.jpg" }],

@@ -710,7 +760,7 @@ describe("resolveSlackAttachmentContent", () => {

710760

});

711761712762

it("downloads Slack-hosted images from forwarded shared attachments", async () => {

713-

vi.spyOn(mediaStore, "saveMediaBuffer").mockResolvedValue(

763+

vi.spyOn(mediaRuntime, "saveMediaBuffer").mockResolvedValue(

714764

createSavedMedia("/tmp/forwarded.jpg", "image/jpeg"),

715765

);

716766