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

推荐订阅源

Hacker News: Ask HN
Hacker News: Ask HN
Last Week in AI
Last Week in AI
G
Google Developers Blog
腾讯CDC
OSCHINA 社区最新新闻
OSCHINA 社区最新新闻
博客园 - 司徒正美
IT之家
IT之家
博客园 - 聂微东
Google DeepMind News
Google DeepMind News
M
Microsoft Research Blog - Microsoft Research
Blog — PlanetScale
Blog — PlanetScale
D
Docker
F
Fortinet All Blogs
A
About on SuperTechFans
J
Java Code Geeks
Microsoft Azure Blog
Microsoft Azure Blog
奇客Solidot–传递最新科技情报
奇客Solidot–传递最新科技情报
C
Cyber Attacks, Cyber Crime and Cyber Security
钛媒体:引领未来商业与生活新知
钛媒体:引领未来商业与生活新知
小众软件
小众软件
PCI Perspectives
PCI Perspectives
GbyAI
GbyAI
Recorded Future
Recorded Future
E
Exploit-DB.com RSS Feed
V
V2EX - 技术
S
Schneier on Security
S
Security Archives - TechRepublic
I
InfoQ
Hacker News - Newest:
Hacker News - Newest: "LLM"
L
LINUX DO - 最新话题
W
WeLiveSecurity
Security Latest
Security Latest
博客园 - 三生石上(FineUI控件)
T
The Blog of Author Tim Ferriss
Stack Overflow Blog
Stack Overflow Blog
Stack Overflow Blog
Stack Overflow Blog
Hugging Face - Blog
Hugging Face - Blog
B
Blog
Apple Machine Learning Research
Apple Machine Learning Research
Recent Commits to openclaw:main
Recent Commits to openclaw:main
S
Secure Thoughts
B
Blog RSS Feed
N
Netflix TechBlog - Medium
C
Comments on: Blog
SecWiki News
SecWiki News
C
Cybersecurity and Infrastructure Security Agency CISA
Microsoft Security Blog
Microsoft Security Blog
雷峰网
雷峰网
P
Proofpoint News Feed
I
Intezer

Recent Commits to openclaw:main

fix(sessions): stop doctor OOM on large session stores and reclaim st… fix(ollama): strip inline kimi cloud reasoning leak (#86515) fix(utils): clamp fetch timeout timers (#85985) fix(ui): preserve user code block rendering (#85942) fix(memory): prevent silent vector index degradation when embedding p… docs: clarify agent transcript placeholders test(qa): annotate live transport RTT measurements · openclaw/openclaw@bb6f37e fix(qqbot): derive outbound watchdog from configured timeouts (#85267… · openclaw/openclaw@aa702cf fix(test): clean kitchen sink rpc temp state · openclaw/openclaw@6f695c1 fix: quiet missing daily memory reads fix: tighten empty plugin registry reuse · openclaw/openclaw@026cfb6 perf: speed up agent transcript lookup · openclaw/openclaw@e7ad116 fix: guard QMD session stem fallback (#86482) · openclaw/openclaw@2e3b59b Guard OpenAI chat payload turns (#86497) · openclaw/openclaw@489e415 fix(gateway): keep session tool mirrors under pressure · openclaw/openclaw@459e89a docs: route github creation through agent transcript test(tools): add unmocked image custom-provider auth regression (#85733) · openclaw/openclaw@f0bfb3f refactor(plugin-sdk): rename plain text tool-call compat wrapper docs(skills): defer private release locators · openclaw/openclaw@23d38e4 Replace Sharp image backend with Photon (#86437) · openclaw/openclaw@b9f975b fix(agents): release embedded-attempt session lock on every exit path… · openclaw/openclaw@32ddfc2 fix: accept OpenClaw voice wake confusions (#86507) fix(crabbox): bootstrap macos js toolchain chore: add agent transcript skill · openclaw/openclaw@d63e8d4 fix(gateway): dedupe session tool fanout · openclaw/openclaw@89a21db fix: Hook ingress token unlocks password-mode gateway auth (#86453) · openclaw/openclaw@d51f268 fix #86077: keep fallback errors candidate scoped (#86134) · openclaw/openclaw@d6b7fe8 fix(diagnostics): reclaim wedged session lanes with a stale leaked ac… · openclaw/openclaw@6f76d9f fix: derive plugin media trust from metadata (#86410) · openclaw/openclaw@e761eb8 fix(media-understanding): normalize HEIC before image descriptions (#… · openclaw/openclaw@75c7236 fix: accept leading fuzzy Discord voice wake names (#86484) · openclaw/openclaw@8fe4f34 feat: promote provider tool call stream wrapper (#86489) fix(test): dedupe kitchen sink command assertions test: derive deprecated sdk usage guard (#86403) fix(qa): extend memory fallback Windows budget fix(ui): move control ui chunking helper out of runtime source · openclaw/openclaw@968c87d fix: quiet retained lost task noise (#86475) fix(build): keep control ui chunking out of deadcode · openclaw/openclaw@dc26069 fix: rotate realtime voice sessions on max duration · openclaw/openclaw@dc2c4aa fix(test): stream bundled plugin sweep logs · openclaw/openclaw@fc3cd49 docs: add bugfix changelog credits · openclaw/openclaw@2e7e4bc fix(models): show oauth marker auth status (#86378) · openclaw/openclaw@a6df39d fix: seed cron task progress summaries (#86313) · openclaw/openclaw@92afd8b fix(update): exclude prerelease tags from stable git channel (#86260) · openclaw/openclaw@28f169b fix(doctor): warn and continue when cron job store is unreadable (#86… fix(gateway): clear runtime config snapshot before in-process restart… · openclaw/openclaw@90caa3b fix(scripts): restore sparse crabbox changed gates · openclaw/openclaw@d270879 fix(build): support Windows UI builds · openclaw/openclaw@0bb9b42 Fix local embedding worker safety (#85348) · openclaw/openclaw@7ff29a9 fix(ui): scope chat session picker to active agent (#85965) · openclaw/openclaw@70c7d6f [codex] improve iOS realtime talk mode (#86355) · openclaw/openclaw@9ca52ce fix(scripts): dedupe docker lane resources · openclaw/openclaw@5e94469 docs: add code size guidance · openclaw/openclaw@9a60fcf fix(test): avoid source gateway import in rpc walk · openclaw/openclaw@e9b8a6e docs: add bugfix changelog entries · openclaw/openclaw@f950132 Fix heartbeat response loop guard (#86324) (#86357) · openclaw/openclaw@e2c174e fix(memory-core): filter REM dreaming candidates to light-staged entr… · openclaw/openclaw@8b42771 fix(telegram): propagate forum topic names into agent context (#86299) fix(slack): keep downloaded files out of reply media (#86318) · openclaw/openclaw@2fcd481 fix(cron): accept plus durations for one-shot jobs (#86341) · openclaw/openclaw@9239f94 fix(plugins): clear metadata memo at lifecycle boundaries · openclaw/openclaw@e7c696a chore(skills): normalize release skill routing · openclaw/openclaw@4737e19 docs(release): require early performance regression check · openclaw/openclaw@0336938 fix(qa): capture Windows gateway metrics · openclaw/openclaw@9afbfc1 feat(qa): add coverage scenario matching · openclaw/openclaw@a1fe86a fix(perf): avoid duplicate docker package ui build build: enable modern TypeScript module syntax · openclaw/openclaw@bbc1772 ci: include performance evidence in release validation fix(providers): stream ordinary tool-like prose promptly fix(perf): harden gateway restart bench exits · openclaw/openclaw@82bbcf6 fix(gateway): gate talk secret bootstrap handoff (#85690) · openclaw/openclaw@c791e42 fix: suppress async media incomplete-turn errors (#85933) · openclaw/openclaw@35dcd42 migrate auth credentials · openclaw/openclaw@f036bac fix migrate auth lint · openclaw/openclaw@50e6cb0 fix migrate supported auth imports · openclaw/openclaw@44bb2be fix migrate auth opt-out precedence · openclaw/openclaw@2016a51 honor migrate auth opt-out in plan · openclaw/openclaw@17edec7 address migrate auth review comments · openclaw/openclaw@0a98c2d fix ci blockers for migrate auth docs: add migrate auth changelog (#85667) · openclaw/openclaw@f7fcbdb fix(scripts): avoid duplicate install smoke ui build · openclaw/openclaw@b1b2841 fix(telegram): preserve inbound text entities (#83873) · openclaw/openclaw@b552919 chore: ignore Python bytecode caches · openclaw/openclaw@b6b2755 fix: make autoreview progress visible · openclaw/openclaw@236edb2 ci(release): fix plugin prerelease extension batch invocation test(telegram): provide topic cache store in message context harness · openclaw/openclaw@ff1fde1 test(agents): complete provider runtime test mocks · openclaw/openclaw@be8cd12 test(telegram): type topic cache harness store · openclaw/openclaw@84ab206 test(agents): sync provider runtime mocks · openclaw/openclaw@a289dd9 refactor: keep plain text tool-call promotion private (#86374) · openclaw/openclaw@c3ab2de fix(discord): suppress self-reply prompt echoes (#86238) docs: clarify config migration policy · openclaw/openclaw@c44367f fix(perf): fail startup bench on early gateway exit · openclaw/openclaw@a8fc28c fix: prevent plain text tool call leaks (#86222) · openclaw/openclaw@cd62780 fix: handle npm min-release-age in installers · openclaw/openclaw@316d97c fix(scripts): include ui:build in build-all full and ciArtifacts prof… · openclaw/openclaw@6704d0a fix(e2e): sample Windows kitchen sink gateway RSS · openclaw/openclaw@73189e3 fix(cron): respect isolated target and error on missing remove id (#8… · openclaw/openclaw@6709f4e fix(pi-embedded-runner): propagate trigger-derived priority to the gl… · openclaw/openclaw@0580f57 fix(cli): suppress self-update version warnings · openclaw/openclaw@e2bd20f
fix(discord): merge media captions into one message (#86487) · openclaw/openclaw@bc10fad
neeravmakwan · 2026-05-25 · via Recent Commits to openclaw:main

@@ -795,6 +795,19 @@ describe("block reply coalescer", () => {

795795

return { flushes, coalescer };

796796

}

797797798+

type FlushedPayload = Parameters<Parameters<typeof createBlockReplyCoalescer>[0]["onFlush"]>[0];

799+

function createPayloadCoalescerHarness<T>(pick: (payload: FlushedPayload) => T) {

800+

const flushes: T[] = [];

801+

const coalescer = createBlockReplyCoalescer({

802+

config: { minChars: 1, maxChars: 200, idleMs: 0, joiner: " " },

803+

shouldAbort: () => false,

804+

onFlush: (payload) => {

805+

flushes.push(pick(payload));

806+

},

807+

});

808+

return { flushes, coalescer };

809+

}

810+798811

it("coalesces chunks within the idle window", async () => {

799812

vi.useFakeTimers();

800813

const { flushes, coalescer } = createBlockCoalescerHarness({

@@ -967,26 +980,107 @@ describe("block reply coalescer", () => {

967980

}

968981

});

969982970-

it("flushes buffered text before media payloads", () => {

971-

const flushes: Array<{ text?: string; mediaUrls?: string[] }> = [];

972-

const coalescer = createBlockReplyCoalescer({

973-

config: { minChars: 1, maxChars: 200, idleMs: 0, joiner: " " },

974-

shouldAbort: () => false,

975-

onFlush: (payload) => {

976-

flushes.push({

977-

text: payload.text,

978-

mediaUrls: payload.mediaUrls,

979-

});

983+

it("merges compatible buffered text into following media payloads", async () => {

984+

const { flushes, coalescer } = createPayloadCoalescerHarness<{

985+

text?: string;

986+

mediaUrls?: string[];

987+

replyToId?: string;

988+

}>((payload) => ({

989+

text: payload.text,

990+

mediaUrls: payload.mediaUrls,

991+

replyToId: payload.replyToId,

992+

}));

993+994+

coalescer.enqueue({ text: "Hello", replyToId: "thread-1" });

995+

coalescer.enqueue({ text: "world" });

996+

coalescer.enqueue({ mediaUrls: ["https://example.com/a.png"] });

997+

await coalescer.flush({ force: true });

998+999+

expect(flushes).toEqual([

1000+

{

1001+

text: "Hello world",

1002+

mediaUrls: ["https://example.com/a.png"],

1003+

replyToId: "thread-1",

9801004

},

981-

});

1005+

]);

1006+

coalescer.stop();

1007+

});

9821008983-

coalescer.enqueue({ text: "Hello" });

984-

coalescer.enqueue({ text: "world" });

1009+

it("keeps reasoning text separate from media payloads", async () => {

1010+

const { flushes, coalescer } = createPayloadCoalescerHarness<{

1011+

text?: string;

1012+

mediaUrls?: string[];

1013+

isReasoning?: boolean;

1014+

}>((payload) => ({

1015+

text: payload.text,

1016+

mediaUrls: payload.mediaUrls,

1017+

isReasoning: payload.isReasoning,

1018+

}));

1019+1020+

coalescer.enqueue({ text: "hidden", isReasoning: true });

9851021

coalescer.enqueue({ mediaUrls: ["https://example.com/a.png"] });

986-

void coalescer.flush({ force: true });

1022+

await coalescer.flush({ force: true });

1023+1024+

expect(flushes).toEqual([

1025+

{ text: "hidden", mediaUrls: undefined, isReasoning: true },

1026+

{

1027+

text: undefined,

1028+

mediaUrls: ["https://example.com/a.png"],

1029+

isReasoning: undefined,

1030+

},

1031+

]);

1032+

coalescer.stop();

1033+

});

1034+1035+

it("keeps buffered text separate when media changes reply target", async () => {

1036+

const { flushes, coalescer } = createPayloadCoalescerHarness<{

1037+

text?: string;

1038+

mediaUrls?: string[];

1039+

replyToId?: string;

1040+

}>((payload) => ({

1041+

text: payload.text,

1042+

mediaUrls: payload.mediaUrls,

1043+

replyToId: payload.replyToId,

1044+

}));

1045+1046+

coalescer.enqueue({ text: "Unthreaded caption" });

1047+

coalescer.enqueue({ mediaUrls: ["https://example.com/a.png"], replyToId: "thread-2" });

1048+

await coalescer.flush({ force: true });

9871049988-

expect(flushes[0].text).toBe("Hello world");

989-

expect(flushes[1].mediaUrls).toEqual(["https://example.com/a.png"]);

1050+

expect(flushes).toEqual([

1051+

{ text: "Unthreaded caption", mediaUrls: undefined, replyToId: undefined },

1052+

{

1053+

text: undefined,

1054+

mediaUrls: ["https://example.com/a.png"],

1055+

replyToId: "thread-2",

1056+

},

1057+

]);

1058+

coalescer.stop();

1059+

});

1060+1061+

it("keeps text separate from voice media payloads", async () => {

1062+

const { flushes, coalescer } = createPayloadCoalescerHarness<{

1063+

text?: string;

1064+

mediaUrls?: string[];

1065+

audioAsVoice?: boolean;

1066+

}>((payload) => ({

1067+

text: payload.text,

1068+

mediaUrls: payload.mediaUrls,

1069+

audioAsVoice: payload.audioAsVoice,

1070+

}));

1071+1072+

coalescer.enqueue({ text: "Listen to this" });

1073+

coalescer.enqueue({ mediaUrls: ["https://example.com/a.ogg"], audioAsVoice: true });

1074+

await coalescer.flush({ force: true });

1075+1076+

expect(flushes).toEqual([

1077+

{ text: "Listen to this", mediaUrls: undefined, audioAsVoice: undefined },

1078+

{

1079+

text: undefined,

1080+

mediaUrls: ["https://example.com/a.ogg"],

1081+

audioAsVoice: true,

1082+

},

1083+

]);

9901084

coalescer.stop();

9911085

});

9921086

});