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

推荐订阅源

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(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): surface async queue drops fix(diagnostics): bound diagnostic buffers · openclaw/openclaw@bdcaac0 chore(diagnostics): refresh plugin sdk baseline · openclaw/openclaw@ab684f5 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): document watcher edge cases, add teardown/rebuild tests,… · openclaw/openclaw@19ff77e 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(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): keep rpc source walk on source call gateway test(plugins): run kitchen sink rpc lane without tsx 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: harden npm shrinkwrap release path feat: bundle plugin npm dependencies · openclaw/openclaw@de022bb fix: opt codex out of bundled runtime deps · openclaw/openclaw@fcecbd8 test: update shrinkwrap packaging expectations · openclaw/openclaw@a1b05aa fix: include plugin shrinkwraps in dependency reports · openclaw/openclaw@82f69a2
修复(Android): 更改TLS指纹时的提示 · 开爪(openclaw)/开爪(openclaw)@848e048
sliekens · 2026-05-17 · via Recent Commits to openclaw:main

@@ -10,6 +10,7 @@ import ai.openclaw.app.node.InvokeDispatcher

1010

import ai.openclaw.app.protocol.OpenClawTalkCommand

1111

import ai.openclaw.app.voice.TalkModeManager

1212

import android.Manifest

13+

import kotlinx.coroutines.CompletableDeferred

1314

import kotlinx.coroutines.flow.MutableStateFlow

1415

import kotlinx.coroutines.runBlocking

1516

import org.junit.Assert.assertEquals

@@ -153,7 +154,7 @@ class GatewayBootstrapAuthTest {

153154

NodeRuntime(

154155

app,

155156

prefs,

156-

tlsFingerprintProbe = { _, _ -> GatewayTlsProbeResult(fingerprintSha256 = "fp-1") },

157+

tlsFingerprintProbe = { _, _ -> GatewayTlsProbeResult(fingerprintSha256 = "fp:1") },

157158

)

158159

val endpoint = GatewayEndpoint.manual(host = "gateway.example", port = 18789)

159160

val explicitAuth =

@@ -169,11 +170,93 @@ class GatewayBootstrapAuthTest {

169170170171

runtime.acceptGatewayTrustPrompt()

171172172-

assertEquals("fp-1", prefs.loadGatewayTlsFingerprint(endpoint.stableId))

173-

assertEquals("setup-bootstrap-token", desiredBootstrapToken(runtime, "nodeSession"))

173+

assertEquals("f1", prefs.loadGatewayTlsFingerprint(endpoint.stableId))

174+

assertEquals("setup-bootstrap-token", waitForDesiredBootstrapToken(runtime, "nodeSession"))

174175

assertNull(desiredBootstrapToken(runtime, "operatorSession"))

175176

}

176177178+

@Test

179+

fun connect_promptsBeforeReplacingChangedTlsFingerprint() =

180+

runBlocking {

181+

val app = RuntimeEnvironment.getApplication()

182+

val securePrefs =

183+

app.getSharedPreferences(

184+

"openclaw.node.secure.test.${UUID.randomUUID()}",

185+

android.content.Context.MODE_PRIVATE,

186+

)

187+

val prefs = SecurePrefs(app, securePrefsOverride = securePrefs)

188+

val endpoint = GatewayEndpoint.manual(host = "gateway.example", port = 18789)

189+

prefs.saveGatewayTlsFingerprint(endpoint.stableId, "sha256:aa:aa:aa:aa")

190+

val runtime =

191+

NodeRuntime(

192+

app,

193+

prefs,

194+

tlsFingerprintProbe = { _, _ -> GatewayTlsProbeResult(fingerprintSha256 = "sha256:bb:bb:bb:bb") },

195+

)

196+197+

runtime.connect(

198+

endpoint,

199+

NodeRuntime.GatewayConnectAuth(token = "shared-token", bootstrapToken = null, password = null),

200+

)

201+202+

val prompt = waitForGatewayTrustPrompt(runtime)

203+

assertEquals("aaaaaaaa", prompt.previousFingerprintSha256)

204+

assertEquals("bbbbbbbb", prompt.fingerprintSha256)

205+

assertEquals("sha256:aa:aa:aa:aa", prefs.loadGatewayTlsFingerprint(endpoint.stableId))

206+207+

runtime.declineGatewayTrustPrompt()

208+209+

assertEquals("sha256:aa:aa:aa:aa", prefs.loadGatewayTlsFingerprint(endpoint.stableId))

210+211+

runtime.connect(

212+

endpoint,

213+

NodeRuntime.GatewayConnectAuth(token = "shared-token", bootstrapToken = null, password = null),

214+

)

215+

waitForGatewayTrustPrompt(runtime)

216+

runtime.acceptGatewayTrustPrompt()

217+218+

assertEquals("bbbbbbbb", prefs.loadGatewayTlsFingerprint(endpoint.stableId))

219+

}

220+221+

@Test

222+

fun connect_ignoresStaleTlsProbeAfterDisconnect() =

223+

runBlocking {

224+

val app = RuntimeEnvironment.getApplication()

225+

val securePrefs =

226+

app.getSharedPreferences(

227+

"openclaw.node.secure.test.${UUID.randomUUID()}",

228+

android.content.Context.MODE_PRIVATE,

229+

)

230+

val prefs = SecurePrefs(app, securePrefsOverride = securePrefs)

231+

val endpoint = GatewayEndpoint.manual(host = "gateway.example", port = 18789)

232+

prefs.saveGatewayTlsFingerprint(endpoint.stableId, "aaaaaaaa")

233+

val probeStarted = CompletableDeferred<Unit>()

234+

val probeResult = CompletableDeferred<GatewayTlsProbeResult>()

235+

val runtime =

236+

NodeRuntime(

237+

app,

238+

prefs,

239+

tlsFingerprintProbe = { _, _ ->

240+

probeStarted.complete(Unit)

241+

probeResult.await()

242+

},

243+

)

244+245+

runtime.connect(

246+

endpoint,

247+

NodeRuntime.GatewayConnectAuth(token = "shared-token", bootstrapToken = null, password = null),

248+

)

249+

probeStarted.await()

250+251+

runtime.disconnect()

252+

probeResult.complete(GatewayTlsProbeResult(fingerprintSha256 = "aaaaaaaa"))

253+

Thread.sleep(100)

254+255+

assertNull(runtime.pendingGatewayTrust.value)

256+

assertNull(desiredBootstrapToken(runtime, "nodeSession"))

257+

assertEquals("aaaaaaaa", prefs.loadGatewayTlsFingerprint(endpoint.stableId))

258+

}

259+177260

@Test

178261

fun connect_showsSecureEndpointGuidanceWhenTlsProbeFails() {

179262

val app = RuntimeEnvironment.getApplication()

@@ -269,6 +352,21 @@ class GatewayBootstrapAuthTest {

269352

return readField(desired, "bootstrapToken")

270353

}

271354355+

private fun waitForDesiredBootstrapToken(

356+

runtime: NodeRuntime,

357+

sessionFieldName: String,

358+

): String {

359+

var lastObserved: String? = null

360+

repeat(50) {

361+

desiredBootstrapToken(runtime, sessionFieldName)?.let { token ->

362+

lastObserved = token

363+

return token

364+

}

365+

Thread.sleep(10)

366+

}

367+

error("Expected desired bootstrap token for $sessionFieldName; last observed=$lastObserved")

368+

}

369+272370

private fun <T> readField(

273371

target: Any,

274372

name: String,