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

推荐订阅源

Google DeepMind News
Google DeepMind News
F
Fortinet All Blogs
阮一峰的网络日志
阮一峰的网络日志
Apple Machine Learning Research
Apple Machine Learning Research
爱范儿
爱范儿
WordPress大学
WordPress大学
让小产品的独立变现更简单 - ezindie.com
让小产品的独立变现更简单 - ezindie.com
J
Java Code Geeks
罗磊的独立博客
S
SegmentFault 最新的问题
V
V2EX
V
Visual Studio Blog
钛媒体:引领未来商业与生活新知
钛媒体:引领未来商业与生活新知
美团技术团队
博客园 - 三生石上(FineUI控件)
Stack Overflow Blog
Stack Overflow Blog
Y
Y Combinator Blog
MyScale Blog
MyScale Blog
D
Docker
Google DeepMind News
Google DeepMind News
Blog — PlanetScale
Blog — PlanetScale
M
Microsoft Research Blog - Microsoft Research
Martin Fowler
Martin Fowler
S
Secure Thoughts
B
Blog
cs.CL updates on arXiv.org
cs.CL updates on arXiv.org
www.infosecurity-magazine.com
www.infosecurity-magazine.com
Recent Announcements
Recent Announcements
MongoDB | Blog
MongoDB | Blog
C
Cisco Blogs
C
CERT Recently Published Vulnerability Notes
T
True Tiger Recordings
GbyAI
GbyAI
P
Proofpoint News Feed
P
Privacy International News Feed
Jina AI
Jina AI
The Cloudflare Blog
I
Intezer
AWS News Blog
AWS News Blog
Hacker News - Newest:
Hacker News - Newest: "LLM"
S
Security Archives - TechRepublic
NISL@THU
NISL@THU
The Register - Security
The Register - Security
Recent Commits to openclaw:main
Recent Commits to openclaw:main
P
Palo Alto Networks Blog
S
Schneier on Security
L
LINUX DO - 热门话题
C
CXSECURITY Database RSS Feed - CXSecurity.com
Security Latest
Security Latest
C
Cybersecurity and Infrastructure Security Agency CISA

Recent Commits to openclaw:main

test(qa-lab): report scenario pack coverage · openclaw/openclaw@dcd98bf fix(plugins): drop stale tlon tool contract · openclaw/openclaw@d70dc4b fix(installer): prefer tar for portable Node extraction · openclaw/openclaw@a54a881 fix(codex): deliver native subagent completions feat: add context-engine host capability requirements (#84994) · openclaw/openclaw@cff5244 fix(release): keep shrinkwrap pinned to pnpm lock · openclaw/openclaw@9d24fde fix: surface plan updates as status notices · openclaw/openclaw@dc04503 test(google): narrow web search fake timers · openclaw/openclaw@fe7d13c fix(installer): extract portable Node with ZipFile · openclaw/openclaw@ffa6cd8 fix(gateway): defer provider auth prewarm after startup (#85369) · openclaw/openclaw@69255f8 fix(talk): stabilize realtime voice consults · openclaw/openclaw@683ad75 test(qa): tolerate slow gateway rpc startup · openclaw/openclaw@29118a0 fix(diagnostics): bound diagnostic buffers · openclaw/openclaw@bdcaac0 chore(diagnostics): refresh plugin sdk baseline · openclaw/openclaw@ab684f5 fix(diagnostics): surface async queue drops fix(installer): copy portable Node into place · openclaw/openclaw@c21ca88 fix(cli): recover replaced device approvals (#85342) · openclaw/openclaw@6ea907c test(release): align prerelease validation · openclaw/openclaw@0def3e2 fix(installer): install portable Node directory atomically · openclaw/openclaw@2890b1a fix(runtime-llm): avoid duplicate provider prefix in allowlist diagno… · openclaw/openclaw@937a756 fix(gateway): include openclaw bin in service PATH (#84475) · openclaw/openclaw@66d1d13 fix(gateway): handle concurrent launchd bootstrap restart race (#84722) · openclaw/openclaw@ba86716 feat: support pi and opencode autoreview engines · openclaw/openclaw@31a189d ci(package): gate acceptance on package integrity · openclaw/openclaw@5275929 ci(release): bypass pnpm for tsdown package build · openclaw/openclaw@fea89cd test(release): align prerelease validation baselines · openclaw/openclaw@04ebdc6 ci(release): harden docker package build · openclaw/openclaw@7b1fbe1 fix(codex): skip native web search transcript mirroring (#85346) · openclaw/openclaw@c3531fc fix(gateway): harden launchd reload handoff race recovery (#84641) · openclaw/openclaw@fc7a531 fix: honor per-model provider transport overrides (#80488) fix(skills): type watcher mock calls in dedupe regression tests · openclaw/openclaw@bb73f0a fix(skills): dedupe shared-directory watchers across agent workspaces… · openclaw/openclaw@3e94290 fix(skills): document watcher edge cases, add teardown/rebuild tests,… · openclaw/openclaw@19ff77e fix(infra): allow macos browser open over ssh env (#85340) · openclaw/openclaw@47d66fe fix(update): preserve package service state during cutover (#83026) · openclaw/openclaw@a15797a fix(gateway): broadcast agent-run error payloads (#85355) · openclaw/openclaw@07e61fc test(e2e): avoid synthetic channel config in plugin smoke fix(cli): suppress systemd hints for live gateway (#85336) · openclaw/openclaw@a00c583 fix(cli): honor agent for model auth logout (#85326) · openclaw/openclaw@fc47c1f fix(gateway): eager-load lifecycle runtime to survive in-place upgrad… · openclaw/openclaw@4a91385 fix(doctor): point codex asset warning at migrate plan (#85324) fix(update): harden managed handoff cwd (#83875) · openclaw/openclaw@1bafc23 docs(release): prepare 2026.5.21 notes ci(crabbox): harden docker hydration refactor(crabbox): parse provider list from binary help instead of ha… test(plugins): add kitchen sink rpc docker lane · openclaw/openclaw@6f6da5f test(plugins): run kitchen sink rpc lane without tsx test(plugins): keep rpc source walk on source call gateway test(qa-lab): add bus tool trace scenario · openclaw/openclaw@2b39613 fix(cron): classify network retry errors (#85344) fix(installer): bootstrap portable Windows Node · openclaw/openclaw@3551e98 fix(ui): move chat session search into picker (#85303) · openclaw/openclaw@1fdc73a chore: add shrinkwrap to plugin npm packages · openclaw/openclaw@b6c8807 feat: bundle plugin npm dependencies · openclaw/openclaw@de022bb chore: harden npm shrinkwrap release path fix: cover plugin package locks in dependency review · openclaw/openclaw@bfa5b39 fix: make bundled plugin packages portable chore: refresh shrinkwrap for Testbox npm · openclaw/openclaw@b2dc449 test: refresh shrinkwrap after rebase · openclaw/openclaw@8b0537c fix: honor overrides in npm shrinkwrap generation · openclaw/openclaw@0d28040 fix: honor shrinkwrap when bundling plugin deps fix: include plugin shrinkwraps in dependency reports · openclaw/openclaw@82f69a2 fix: opt codex out of bundled runtime deps · openclaw/openclaw@fcecbd8 test: update shrinkwrap packaging expectations · openclaw/openclaw@a1b05aa fix: keep bundled plugin peers nested · openclaw/openclaw@86faf65 fix: opt acpx out of bundled runtime deps · openclaw/openclaw@9914e25 fix: publish explicit plugin bundled dependencies · openclaw/openclaw@976da39 chore: refresh shrinkwrap metadata fix: limit subagent bootstrap defaults · openclaw/openclaw@56308a7 feat: update autoreview engine coverage · openclaw/openclaw@ab1fedb fix(messages): strip unsupported citation markers (#85204) (thanks @n… · openclaw/openclaw@0a95e53 test(qa-lab): report live transport coverage lanes · openclaw/openclaw@fda0baf fix(gateway): close child ACP sessions on parent reset/delete · openclaw/openclaw@136c927 fix: preserve Google Gemini 3 cron thinking (#85300) docs(skills): exclude SDK boundary bug sweeps · openclaw/openclaw@85e468d feat(plugin-sdk): add generic channel poll sender (#85299) · openclaw/openclaw@c9a0f03 fix(agents): preserve OpenAI reasoning token usage · openclaw/openclaw@0ddf51c test(e2e): harden plugin smoke cleanup fix(plugins): resolve native plugin sdk aliases (#85298) · openclaw/openclaw@6b1c868 fix(update): keep service logs out of json output · openclaw/openclaw@03f61cd fix(agent): retry transient gateway handshake closes · openclaw/openclaw@ff79299 fix(codex): keep interrupted turns visible-answer eligible (#84494) · openclaw/openclaw@8523e09 test(agents): narrow bundle mcp e2e setup · openclaw/openclaw@6bd430e test: add mocked Control UI E2E tests and playwright for local verifi… fix: land code-mode structured worker errors (#83444) (thanks @Kaspre) · openclaw/openclaw@70dd315 fix(code-mode): return structured worker error codes · openclaw/openclaw@edab653 fix heartbeat event routing for main-scoped DMs fix: preserve route-bound direct thread events · openclaw/openclaw@0d8c9ca test: align exec event routing proof (#83743) (thanks @Kaspre) · openclaw/openclaw@7b48956 fix: route direct thread event wakes to main DMs · openclaw/openclaw@0acfb7b fix: break plugin metadata snapshot cycle · openclaw/openclaw@4ee8a2a test node exec event wake metadata · openclaw/openclaw@37207c6 test(plugins): retry bundled smoke health probes test(gateway): bind auth-free websocket harness to loopback · openclaw/openclaw@2b1c01f test(plugins): keep npm peer prune mock directory-safe · openclaw/openclaw@a12e302 chore(ui): refresh fa control ui locale fix(ci): allow release update restarts · openclaw/openclaw@b859654 chore(ui): refresh nl control ui locale · openclaw/openclaw@cc6d222 chore(ui): refresh vi control ui locale · openclaw/openclaw@b59ab5b chore(ui): refresh th control ui locale · openclaw/openclaw@f483f59
fix(telegram): distinguish partial reply snapshots · openclaw/openclaw@741eafe
obviyus · 2026-05-17 · via Recent Commits to openclaw:main

@@ -63,6 +63,13 @@ type PersistedMessageReadResult = TelegramMessageCacheBucket & {

6363

needsRewrite: boolean;

6464

};

656566+

type TelegramMessageObservationMode = "authoritative" | "partial";

67+68+

type TelegramCachedMessageObservation = {

69+

node: TelegramCachedMessageNode;

70+

mode: TelegramMessageObservationMode;

71+

};

72+6673

const DEFAULT_MAX_MESSAGES = 5000;

6774

const COMPACT_THRESHOLD_RATIO = 2;

6875

const persistedMessageCacheBuckets = new Map<string, TelegramMessageCacheBucket>();

@@ -164,14 +171,18 @@ function resolveMessageThreadId(msg: Message): number | undefined {

164171

function normalizeMessageNodes(

165172

msg: Message,

166173

params: { threadId?: number },

167-

): TelegramCachedMessageNode[] {

168-

const nodes: TelegramCachedMessageNode[] = [];

174+

): TelegramCachedMessageObservation[] {

175+

const observations: TelegramCachedMessageObservation[] = [];

169176

const visited = new Set<string>();

170177

const nodeThreadId = (node: TelegramCachedMessageNode) => {

171178

const threadId = Number(node.threadId);

172179

return Number.isFinite(threadId) ? threadId : undefined;

173180

};

174-

const visit = (message: Message, inheritedThreadId?: number) => {

181+

const visit = (

182+

message: Message,

183+

inheritedThreadId: number | undefined,

184+

mode: TelegramMessageObservationMode,

185+

) => {

175186

const node = normalizeMessageNode(message, {

176187

threadId: resolveMessageThreadId(message) ?? inheritedThreadId,

177188

});

@@ -181,12 +192,12 @@ function normalizeMessageNodes(

181192

visited.add(node.messageId);

182193

const replyMessage = resolveEmbeddedReplyMessage(message);

183194

if (replyMessage?.message_id != null) {

184-

visit(replyMessage, nodeThreadId(node) ?? inheritedThreadId);

195+

visit(replyMessage, nodeThreadId(node) ?? inheritedThreadId, "partial");

185196

}

186-

nodes.push(node);

197+

observations.push({ node, mode });

187198

};

188-

visit(msg, params.threadId);

189-

return nodes;

199+

visit(msg, params.threadId, "authoritative");

200+

return observations;

190201

}

191202192203

function isRecord(value: unknown): value is Record<string, unknown> {

@@ -215,6 +226,7 @@ function isTelegramSourceMessage(value: unknown): value is Message {

215226

function parsePersistedEntry(value: unknown): Array<{

216227

key: string;

217228

node: TelegramCachedMessageNode;

229+

mode: TelegramMessageObservationMode;

218230

}> {

219231

if (!isRecord(value) || !isString(value.key)) {

220232

return [];

@@ -229,10 +241,15 @@ function parsePersistedEntry(value: unknown): Array<{

229241

}

230242

const keyPrefix = value.key.slice(0, separatorIndex + 1);

231243

const threadId = Number(readOptionalString(value.node, "threadId"));

244+

const sourceMessageId = String(value.node.sourceMessage.message_id);

232245

return normalizeMessageNodes(

233246

value.node.sourceMessage,

234247

Number.isFinite(threadId) ? { threadId } : {},

235-

).map((node) => ({ key: `${keyPrefix}${node.messageId}`, node }));

248+

).map(({ node, mode }) => ({

249+

key: `${keyPrefix}${node.messageId}`,

250+

node,

251+

mode: node.messageId === sourceMessageId ? "authoritative" : mode,

252+

}));

236253

}

237254238255

function findJsonArrayEnd(text: string): number {

@@ -337,26 +354,41 @@ function mergeTelegramSourceMessage(existing: Message, incoming: Message): Messa

337354

return merged;

338355

}

339356357+

function mergeAuthoritativeTelegramSourceMessage(existing: Message, incoming: Message): Message {

358+

const existingReply = resolveEmbeddedReplyMessage(existing);

359+

const incomingReply = resolveEmbeddedReplyMessage(incoming);

360+

if (existingReply?.message_id != null && incomingReply?.message_id === existingReply.message_id) {

361+

return {

362+

...incoming,

363+

reply_to_message: mergeTelegramSourceMessage(existingReply, incomingReply),

364+

};

365+

}

366+

return incoming;

367+

}

368+340369

function mergeCachedMessageNode(

341370

existing: TelegramCachedMessageNode,

342371

incoming: TelegramCachedMessageNode,

372+

mode: TelegramMessageObservationMode,

343373

): TelegramCachedMessageNode {

344374

const threadId = Number(incoming.threadId ?? existing.threadId);

345-

return normalizeRequiredMessageNode(

346-

mergeTelegramSourceMessage(existing.sourceMessage, incoming.sourceMessage),

347-

{

348-

...(Number.isFinite(threadId) ? { threadId } : {}),

349-

},

350-

);

375+

const sourceMessage =

376+

mode === "authoritative"

377+

? mergeAuthoritativeTelegramSourceMessage(existing.sourceMessage, incoming.sourceMessage)

378+

: mergeTelegramSourceMessage(existing.sourceMessage, incoming.sourceMessage);

379+

return normalizeRequiredMessageNode(sourceMessage, {

380+

...(Number.isFinite(threadId) ? { threadId } : {}),

381+

});

351382

}

352383353384

function upsertCachedMessageNode(params: {

354385

messages: Map<string, TelegramCachedMessageNode>;

355386

key: string;

356387

node: TelegramCachedMessageNode;

388+

mode: TelegramMessageObservationMode;

357389

}): TelegramCachedMessageNode {

358390

const existing = params.messages.get(params.key);

359-

const node = existing ? mergeCachedMessageNode(existing, params.node) : params.node;

391+

const node = existing ? mergeCachedMessageNode(existing, params.node, params.mode) : params.node;

360392

params.messages.delete(params.key);

361393

params.messages.set(params.key, node);

362394

return node;

@@ -375,7 +407,12 @@ function readPersistedMessages(filePath: string, maxMessages: number): Persisted

375407

for (const value of persisted.values) {

376408

for (const entry of parsePersistedEntry(value)) {

377409

persistedEntryCount++;

378-

upsertCachedMessageNode({ messages, key: entry.key, node: entry.node });

410+

upsertCachedMessageNode({

411+

messages,

412+

key: entry.key,

413+

node: entry.node,

414+

mode: entry.mode,

415+

});

379416

trimMessages(messages, maxMessages);

380417

}

381418

}

@@ -515,16 +552,16 @@ export function createTelegramMessageCache(params?: {

515552516553

return {

517554

record: ({ accountId, chatId, msg, threadId }) => {

518-

const entries = normalizeMessageNodes(msg, { threadId });

519-

const entry = entries.at(-1);

520-

if (!entry) {

555+

const observations = normalizeMessageNodes(msg, { threadId });

556+

const currentObservation = observations.at(-1);

557+

if (!currentObservation) {

521558

return null;

522559

}

523560

let recordedEntry: TelegramCachedMessageNode | null = null;

524-

for (const node of entries) {

561+

for (const { node, mode } of observations) {

525562

const key = telegramMessageCacheKey({ accountId, chatId, messageId: node.messageId });

526-

const cachedNode = upsertCachedMessageNode({ messages, key, node });

527-

if (node.messageId === entry.messageId) {

563+

const cachedNode = upsertCachedMessageNode({ messages, key, node, mode });

564+

if (node.messageId === currentObservation.node.messageId) {

528565

recordedEntry = cachedNode;

529566

}

530567

trimMessages(messages, maxMessages);

@@ -544,7 +581,7 @@ export function createTelegramMessageCache(params?: {

544581

logVerbose(`telegram: failed to persist message cache: ${String(error)}`);

545582

}

546583

}

547-

return recordedEntry ?? entry;

584+

return recordedEntry ?? currentObservation.node;

548585

},

549586

get,

550587

recentBefore: ({ accountId, chatId, messageId, threadId, limit }) => {