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

推荐订阅源

V
Vulnerabilities – Threatpost
P
Proofpoint News Feed
The Hacker News
The Hacker News
Know Your Adversary
Know Your Adversary
Threat Intelligence Blog | Flashpoint
Threat Intelligence Blog | Flashpoint
T
Tenable Blog
AWS News Blog
AWS News Blog
S
Securelist
T
Threatpost
C
Cybersecurity and Infrastructure Security Agency CISA
IT之家
IT之家
腾讯CDC
WordPress大学
WordPress大学
Spread Privacy
Spread Privacy
C
Check Point Blog
cs.CL updates on arXiv.org
cs.CL updates on arXiv.org
Engineering at Meta
Engineering at Meta
Latest news
Latest news
A
About on SuperTechFans
The Register - Security
The Register - Security
L
LINUX DO - 热门话题
T
The Exploit Database - CXSecurity.com
C
Cisco Blogs
T
Tailwind CSS Blog
Simon Willison's Weblog
Simon Willison's Weblog
阮一峰的网络日志
阮一峰的网络日志
MyScale Blog
MyScale Blog
大猫的无限游戏
大猫的无限游戏
T
Tor Project blog
L
Lohrmann on Cybersecurity
G
GRAHAM CLULEY
B
Blog RSS Feed
Scott Helme
Scott Helme
让小产品的独立变现更简单 - ezindie.com
让小产品的独立变现更简单 - ezindie.com
NISL@THU
NISL@THU
P
Privacy International News Feed
Security Latest
Security Latest
Recorded Future
Recorded Future
L
LangChain Blog
Cyberwarzone
Cyberwarzone
C
Cyber Attacks, Cyber Crime and Cyber Security
C
CXSECURITY Database RSS Feed - CXSecurity.com
博客园 - 聂微东
Google DeepMind News
Google DeepMind News
Last Week in AI
Last Week in AI
Apple Machine Learning Research
Apple Machine Learning Research
F
Fortinet All Blogs
O
OpenAI News
T
Threat Research - Cisco Blogs
Blog — PlanetScale
Blog — PlanetScale

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
refactor(channels): route remaining turns through kernel · openclaw/openclaw@9a3a341
steipete · 2026-04-30 · via Recent Commits to openclaw:main

@@ -1714,177 +1714,192 @@ async function processMessageAfterDedupe(

17141714

},

17151715

},

17161716

});

1717-

await core.channel.turn.dispatchAssembled({

1718-

cfg: config,

1717+

await core.channel.turn.run({

17191718

channel: "bluebubbles",

17201719

accountId: account.accountId,

1721-

agentId: route.agentId,

1722-

routeSessionKey: route.sessionKey,

1723-

storePath,

1724-

ctxPayload,

1725-

recordInboundSession: core.channel.session.recordInboundSession,

1726-

dispatchReplyWithBufferedBlockDispatcher:

1727-

core.channel.reply.dispatchReplyWithBufferedBlockDispatcher,

1728-

delivery: {

1729-

deliver: async (payload, info) => {

1730-

const rawReplyToId =

1731-

privateApiEnabled && typeof payload.replyToId === "string"

1732-

? payload.replyToId.trim()

1733-

: "";

1734-

// Resolve short ID (e.g., "5") to full UUID, scoped to the chat

1735-

// this deliver path is already routing for (cross-chat guard).

1736-

const replyToMessageGuid = rawReplyToId

1737-

? resolveBlueBubblesMessageId(rawReplyToId, {

1738-

requireKnownShortId: true,

1739-

chatContext: {

1740-

chatGuid: chatGuidForActions ?? chatGuid,

1741-

chatIdentifier,

1742-

chatId,

1743-

},

1744-

})

1745-

: "";

1746-

const mediaList = resolveOutboundMediaUrls(payload);

1747-

if (mediaList.length > 0) {

1748-

const tableMode = core.channel.text.resolveMarkdownTableMode({

1749-

cfg: config,

1750-

channel: "bluebubbles",

1751-

accountId: account.accountId,

1752-

});

1753-

const text = sanitizeReplyDirectiveText(

1754-

core.channel.text.convertMarkdownTables(payload.text ?? "", tableMode),

1755-

);

1756-

await sendMediaWithLeadingCaption({

1757-

mediaUrls: mediaList,

1758-

caption: text,

1759-

send: async ({ mediaUrl, caption }) => {

1760-

const cachedBody = (caption ?? "").trim() || "<media:attachment>";

1720+

raw: ctxPayload,

1721+

adapter: {

1722+

ingest: () => ({

1723+

id: String(ctxPayload.MessageSid ?? message.messageId),

1724+

timestamp: message.timestamp,

1725+

rawText: rawBody,

1726+

textForAgent: rawBody,

1727+

textForCommands: commandBody,

1728+

raw: ctxPayload,

1729+

}),

1730+

resolveTurn: () => ({

1731+

cfg: config,

1732+

channel: "bluebubbles",

1733+

accountId: account.accountId,

1734+

agentId: route.agentId,

1735+

routeSessionKey: route.sessionKey,

1736+

storePath,

1737+

ctxPayload,

1738+

recordInboundSession: core.channel.session.recordInboundSession,

1739+

dispatchReplyWithBufferedBlockDispatcher:

1740+

core.channel.reply.dispatchReplyWithBufferedBlockDispatcher,

1741+

delivery: {

1742+

deliver: async (payload, info) => {

1743+

const rawReplyToId =

1744+

privateApiEnabled && typeof payload.replyToId === "string"

1745+

? payload.replyToId.trim()

1746+

: "";

1747+

// Resolve short ID (e.g., "5") to full UUID, scoped to the chat

1748+

// this deliver path is already routing for (cross-chat guard).

1749+

const replyToMessageGuid = rawReplyToId

1750+

? resolveBlueBubblesMessageId(rawReplyToId, {

1751+

requireKnownShortId: true,

1752+

chatContext: {

1753+

chatGuid: chatGuidForActions ?? chatGuid,

1754+

chatIdentifier,

1755+

chatId,

1756+

},

1757+

})

1758+

: "";

1759+

const mediaList = resolveOutboundMediaUrls(payload);

1760+

if (mediaList.length > 0) {

1761+

const tableMode = core.channel.text.resolveMarkdownTableMode({

1762+

cfg: config,

1763+

channel: "bluebubbles",

1764+

accountId: account.accountId,

1765+

});

1766+

const text = sanitizeReplyDirectiveText(

1767+

core.channel.text.convertMarkdownTables(payload.text ?? "", tableMode),

1768+

);

1769+

await sendMediaWithLeadingCaption({

1770+

mediaUrls: mediaList,

1771+

caption: text,

1772+

send: async ({ mediaUrl, caption }) => {

1773+

const cachedBody = (caption ?? "").trim() || "<media:attachment>";

1774+

const pendingId = rememberPendingOutboundMessageId({

1775+

accountId: account.accountId,

1776+

sessionKey: route.sessionKey,

1777+

outboundTarget,

1778+

chatGuid: chatGuidForActions ?? chatGuid,

1779+

chatIdentifier,

1780+

chatId,

1781+

snippet: cachedBody,

1782+

});

1783+

let result: Awaited<ReturnType<typeof sendBlueBubblesMedia>>;

1784+

try {

1785+

result = await sendBlueBubblesMedia({

1786+

cfg: config,

1787+

to: outboundTarget,

1788+

mediaUrl,

1789+

caption: caption ?? undefined,

1790+

replyToId: replyToMessageGuid || null,

1791+

accountId: account.accountId,

1792+

asVoice: payload.audioAsVoice === true,

1793+

});

1794+

} catch (err) {

1795+

forgetPendingOutboundMessageId(pendingId);

1796+

throw err;

1797+

}

1798+

if (maybeEnqueueOutboundMessageId(result.messageId, cachedBody)) {

1799+

forgetPendingOutboundMessageId(pendingId);

1800+

}

1801+

sentMessage = true;

1802+

statusSink?.({ lastOutboundAt: Date.now() });

1803+

if (info.kind === "block") {

1804+

restartTypingSoon();

1805+

}

1806+

},

1807+

});

1808+

return;

1809+

}

1810+1811+

const textLimit =

1812+

account.config.textChunkLimit && account.config.textChunkLimit > 0

1813+

? account.config.textChunkLimit

1814+

: DEFAULT_TEXT_LIMIT;

1815+

const chunkMode = account.config.chunkMode ?? "length";

1816+

const tableMode = core.channel.text.resolveMarkdownTableMode({

1817+

cfg: config,

1818+

channel: "bluebubbles",

1819+

accountId: account.accountId,

1820+

});

1821+

const text = sanitizeReplyDirectiveText(

1822+

core.channel.text.convertMarkdownTables(payload.text ?? "", tableMode),

1823+

);

1824+

const chunks =

1825+

chunkMode === "newline"

1826+

? resolveTextChunksWithFallback(

1827+

text,

1828+

core.channel.text.chunkTextWithMode(text, textLimit, chunkMode),

1829+

)

1830+

: resolveTextChunksWithFallback(

1831+

text,

1832+

core.channel.text.chunkMarkdownText(text, textLimit),

1833+

);

1834+

if (!chunks.length) {

1835+

return;

1836+

}

1837+

for (const chunk of chunks) {

17611838

const pendingId = rememberPendingOutboundMessageId({

17621839

accountId: account.accountId,

17631840

sessionKey: route.sessionKey,

17641841

outboundTarget,

17651842

chatGuid: chatGuidForActions ?? chatGuid,

17661843

chatIdentifier,

17671844

chatId,

1768-

snippet: cachedBody,

1845+

snippet: chunk,

17691846

});

1770-

let result: Awaited<ReturnType<typeof sendBlueBubblesMedia>>;

1847+

let result: Awaited<ReturnType<typeof sendMessageBlueBubbles>>;

17711848

try {

1772-

result = await sendBlueBubblesMedia({

1849+

result = await sendMessageBlueBubbles(outboundTarget, chunk, {

17731850

cfg: config,

1774-

to: outboundTarget,

1775-

mediaUrl,

1776-

caption: caption ?? undefined,

1777-

replyToId: replyToMessageGuid || null,

17781851

accountId: account.accountId,

1779-

asVoice: payload.audioAsVoice === true,

1852+

replyToMessageGuid: replyToMessageGuid || undefined,

17801853

});

17811854

} catch (err) {

17821855

forgetPendingOutboundMessageId(pendingId);

17831856

throw err;

17841857

}

1785-

if (maybeEnqueueOutboundMessageId(result.messageId, cachedBody)) {

1858+

if (maybeEnqueueOutboundMessageId(result.messageId, chunk)) {

17861859

forgetPendingOutboundMessageId(pendingId);

17871860

}

17881861

sentMessage = true;

17891862

statusSink?.({ lastOutboundAt: Date.now() });

17901863

if (info.kind === "block") {

17911864

restartTypingSoon();

17921865

}

1793-

},

1794-

});

1795-

return;

1796-

}

1797-1798-

const textLimit =

1799-

account.config.textChunkLimit && account.config.textChunkLimit > 0

1800-

? account.config.textChunkLimit

1801-

: DEFAULT_TEXT_LIMIT;

1802-

const chunkMode = account.config.chunkMode ?? "length";

1803-

const tableMode = core.channel.text.resolveMarkdownTableMode({

1804-

cfg: config,

1805-

channel: "bluebubbles",

1806-

accountId: account.accountId,

1807-

});

1808-

const text = sanitizeReplyDirectiveText(

1809-

core.channel.text.convertMarkdownTables(payload.text ?? "", tableMode),

1810-

);

1811-

const chunks =

1812-

chunkMode === "newline"

1813-

? resolveTextChunksWithFallback(

1814-

text,

1815-

core.channel.text.chunkTextWithMode(text, textLimit, chunkMode),

1816-

)

1817-

: resolveTextChunksWithFallback(

1818-

text,

1819-

core.channel.text.chunkMarkdownText(text, textLimit),

1820-

);

1821-

if (!chunks.length) {

1822-

return;

1823-

}

1824-

for (const chunk of chunks) {

1825-

const pendingId = rememberPendingOutboundMessageId({

1826-

accountId: account.accountId,

1827-

sessionKey: route.sessionKey,

1828-

outboundTarget,

1829-

chatGuid: chatGuidForActions ?? chatGuid,

1830-

chatIdentifier,

1831-

chatId,

1832-

snippet: chunk,

1833-

});

1834-

let result: Awaited<ReturnType<typeof sendMessageBlueBubbles>>;

1835-

try {

1836-

result = await sendMessageBlueBubbles(outboundTarget, chunk, {

1837-

cfg: config,

1838-

accountId: account.accountId,

1839-

replyToMessageGuid: replyToMessageGuid || undefined,

1840-

});

1841-

} catch (err) {

1842-

forgetPendingOutboundMessageId(pendingId);

1843-

throw err;

1844-

}

1845-

if (maybeEnqueueOutboundMessageId(result.messageId, chunk)) {

1846-

forgetPendingOutboundMessageId(pendingId);

1847-

}

1848-

sentMessage = true;

1849-

statusSink?.({ lastOutboundAt: Date.now() });

1850-

if (info.kind === "block") {

1851-

restartTypingSoon();

1852-

}

1853-

}

1854-

},

1855-

onError: (err, info) => {

1856-

// Flag the outer dedupe wrapper so it releases the claim instead

1857-

// of committing. Without this, a transient BlueBubbles send failure

1858-

// would permanently block replay-retry for 7 days and the user

1859-

// would never receive a reply to that message.

1860-

//

1861-

// Only the terminal `final` delivery represents the user-visible

1862-

// answer. The dispatcher continues past `tool` / `block` failures

1863-

// and may still deliver `final` successfully — releasing the

1864-

// dedupe claim for those would invite a replay that re-runs tool

1865-

// side effects and resends partially-delivered content.

1866-

if (info.kind === "final") {

1867-

dedupeSignal.deliveryFailed = true;

1868-

}

1869-

runtime.error?.(`BlueBubbles ${info.kind} reply failed: ${sanitizeForLog(err)}`);

1870-

},

1871-

},

1872-

dispatcherOptions: {

1873-

...replyPipeline,

1874-

onReplyStart: typingCallbacks?.onReplyStart,

1875-

onIdle: typingCallbacks?.onIdle,

1876-

},

1877-

replyOptions: {

1878-

onModelSelected,

1879-

disableBlockStreaming:

1880-

typeof account.config.blockStreaming === "boolean"

1881-

? !account.config.blockStreaming

1882-

: undefined,

1883-

},

1884-

record: {

1885-

onRecordError: (err) => {

1886-

runtime.error?.(`[bluebubbles] failed updating session meta: ${sanitizeForLog(err)}`);

1887-

},

1866+

}

1867+

},

1868+

onError: (err, info) => {

1869+

// Flag the outer dedupe wrapper so it releases the claim instead

1870+

// of committing. Without this, a transient BlueBubbles send failure

1871+

// would permanently block replay-retry for 7 days and the user

1872+

// would never receive a reply to that message.

1873+

//

1874+

// Only the terminal `final` delivery represents the user-visible

1875+

// answer. The dispatcher continues past `tool` / `block` failures

1876+

// and may still deliver `final` successfully — releasing the

1877+

// dedupe claim for those would invite a replay that re-runs tool

1878+

// side effects and resends partially-delivered content.

1879+

if (info.kind === "final") {

1880+

dedupeSignal.deliveryFailed = true;

1881+

}

1882+

runtime.error?.(`BlueBubbles ${info.kind} reply failed: ${sanitizeForLog(err)}`);

1883+

},

1884+

},

1885+

dispatcherOptions: {

1886+

...replyPipeline,

1887+

onReplyStart: typingCallbacks?.onReplyStart,

1888+

onIdle: typingCallbacks?.onIdle,

1889+

},

1890+

replyOptions: {

1891+

onModelSelected,

1892+

disableBlockStreaming:

1893+

typeof account.config.blockStreaming === "boolean"

1894+

? !account.config.blockStreaming

1895+

: undefined,

1896+

},

1897+

record: {

1898+

onRecordError: (err) => {

1899+

runtime.error?.(`[bluebubbles] failed updating session meta: ${sanitizeForLog(err)}`);

1900+

},

1901+

},

1902+

}),

18881903

},

18891904

});

18901905

} finally {