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

推荐订阅源

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 chore(diagnostics): refresh plugin sdk baseline · openclaw/openclaw@ab684f5 fix(diagnostics): bound diagnostic buffers · openclaw/openclaw@bdcaac0 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 ci(release): harden docker package build · openclaw/openclaw@7b1fbe1 test(release): align prerelease validation baselines · openclaw/openclaw@04ebdc6 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): dedupe shared-directory watchers across agent workspaces… · openclaw/openclaw@3e94290 fix(skills): type watcher mock calls in dedupe regression tests · openclaw/openclaw@bb73f0a 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 fix: opt codex out of bundled runtime deps · openclaw/openclaw@fcecbd8 fix: keep bundled plugin peers nested · openclaw/openclaw@86faf65 test: refresh shrinkwrap after rebase · openclaw/openclaw@8b0537c feat: bundle plugin npm dependencies · openclaw/openclaw@de022bb fix: include plugin shrinkwraps in dependency reports · openclaw/openclaw@82f69a2 fix: honor shrinkwrap when bundling plugin deps fix: honor overrides in npm shrinkwrap generation · openclaw/openclaw@0d28040 chore: refresh shrinkwrap for Testbox npm · openclaw/openclaw@b2dc449 chore: refresh shrinkwrap metadata fix: publish explicit plugin bundled dependencies · openclaw/openclaw@976da39 fix: opt acpx out of bundled runtime deps · openclaw/openclaw@9914e25 test: update shrinkwrap packaging expectations · openclaw/openclaw@a1b05aa fix: cover plugin package locks in dependency review · openclaw/openclaw@bfa5b39 chore: harden npm shrinkwrap release path fix: make bundled plugin packages portable chore: add shrinkwrap to plugin npm packages · openclaw/openclaw@b6c8807 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 test node exec event wake metadata · openclaw/openclaw@37207c6 test: align exec event routing proof (#83743) (thanks @Kaspre) · openclaw/openclaw@7b48956 fix: break plugin metadata snapshot cycle · openclaw/openclaw@4ee8a2a fix: preserve route-bound direct thread events · openclaw/openclaw@0d8c9ca fix: route direct thread event wakes to main DMs · openclaw/openclaw@0acfb7b 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 chore(ui): refresh id control ui locale · openclaw/openclaw@c222ef0 chore(ui): refresh pl control ui locale · openclaw/openclaw@0050b8e chore(ui): refresh uk control ui locale · openclaw/openclaw@6b4aec9 chore(ui): refresh tr control ui locale · openclaw/openclaw@940a950 chore(ui): refresh ar control ui locale · openclaw/openclaw@d11c2e4 chore(ui): refresh it control ui locale · openclaw/openclaw@c99a29d chore(ui): refresh fr control ui locale · openclaw/openclaw@a7ba47c
fix(android): restart gateway session on reconnect · openclaw/openclaw@f1f92b8
obviyus · 2026-05-18 · via Recent Commits to openclaw:main

@@ -1,10 +1,129 @@

11

package ai.openclaw.app.gateway

223+

import kotlinx.coroutines.CompletableDeferred

4+

import kotlinx.coroutines.CoroutineScope

5+

import kotlinx.coroutines.Dispatchers

6+

import kotlinx.coroutines.Job

7+

import kotlinx.coroutines.SupervisorJob

8+

import kotlinx.coroutines.cancelAndJoin

9+

import kotlinx.coroutines.runBlocking

10+

import kotlinx.coroutines.withTimeout

11+

import kotlinx.serialization.json.Json

12+

import kotlinx.serialization.json.jsonObject

13+

import kotlinx.serialization.json.jsonPrimitive

14+

import okhttp3.Response

15+

import okhttp3.WebSocket

16+

import okhttp3.WebSocketListener

17+

import okhttp3.mockwebserver.Dispatcher

18+

import okhttp3.mockwebserver.MockResponse

19+

import okhttp3.mockwebserver.MockWebServer

20+

import okhttp3.mockwebserver.RecordedRequest

21+

import org.junit.Assert.assertEquals

322

import org.junit.Assert.assertFalse

423

import org.junit.Assert.assertTrue

524

import org.junit.Test

25+

import org.junit.runner.RunWith

26+

import org.robolectric.RobolectricTestRunner

27+

import org.robolectric.RuntimeEnvironment

28+

import org.robolectric.annotation.Config

29+

import java.util.concurrent.ConcurrentLinkedQueue

63031+

private const val LIFECYCLE_TEST_TIMEOUT_MS = 8_000L

32+

private const val LIFECYCLE_CONNECT_CHALLENGE_FRAME =

33+

"""{"type":"event","event":"connect.challenge","payload":{"nonce":"android-test-nonce"}}"""

34+35+

private class ReconnectDeviceAuthStore : DeviceAuthTokenStore {

36+

override fun loadEntry(

37+

deviceId: String,

38+

role: String,

39+

): DeviceAuthEntry? = null

40+41+

override fun saveToken(

42+

deviceId: String,

43+

role: String,

44+

token: String,

45+

scopes: List<String>,

46+

) = Unit

47+48+

override fun clearToken(

49+

deviceId: String,

50+

role: String,

51+

) = Unit

52+

}

53+54+

private data class ReconnectHarness(

55+

val session: GatewaySession,

56+

val sessionJob: Job,

57+

)

58+59+

private data class ReconnectServer(

60+

val server: MockWebServer,

61+

val sockets: ConcurrentLinkedQueue<WebSocket>,

62+

) {

63+

val port: Int

64+

get() = server.port

65+66+

val requestCount: Int

67+

get() = server.requestCount

68+69+

fun shutdown() {

70+

sockets.forEach { runCatching { it.cancel() } }

71+

runCatching { server.shutdown() }

72+

.onFailure { err ->

73+

if (err.message != "Gave up waiting for queue to shut down") throw err

74+

}

75+

}

76+

}

77+78+

@RunWith(RobolectricTestRunner::class)

79+

@Config(sdk = [34])

780

class GatewaySessionReconnectTest {

81+

@Test

82+

fun connectToNewGatewayClosesActiveConnectionAndStartsReplacement() =

83+

runBlocking {

84+

val json = Json { ignoreUnknownKeys = true }

85+

val firstConnect = CompletableDeferred<Unit>()

86+

val firstClosed = CompletableDeferred<Unit>()

87+

val secondConnect = CompletableDeferred<Unit>()

88+

val secondClosed = CompletableDeferred<Unit>()

89+

val firstServer =

90+

startGatewayServer(

91+

json = json,

92+

onClosed = { firstClosed.complete(Unit) },

93+

) { webSocket, id, method ->

94+

if (method == "connect") {

95+

firstConnect.complete(Unit)

96+

webSocket.send(connectResponseFrame(id))

97+

}

98+

}

99+

val secondServer =

100+

startGatewayServer(

101+

json = json,

102+

onClosed = { secondClosed.complete(Unit) },

103+

) { webSocket, id, method ->

104+

if (method == "connect") {

105+

secondConnect.complete(Unit)

106+

webSocket.send(connectResponseFrame(id))

107+

}

108+

}

109+

val harness = createReconnectHarness()

110+111+

try {

112+

connectNodeSession(harness.session, firstServer.port)

113+

withTimeout(LIFECYCLE_TEST_TIMEOUT_MS) { firstConnect.await() }

114+115+

connectNodeSession(harness.session, secondServer.port)

116+117+

withTimeout(LIFECYCLE_TEST_TIMEOUT_MS) { firstClosed.await() }

118+

withTimeout(LIFECYCLE_TEST_TIMEOUT_MS) { secondConnect.await() }

119+

assertEquals(1, secondServer.requestCount)

120+

harness.session.disconnect()

121+

withTimeout(LIFECYCLE_TEST_TIMEOUT_MS) { secondClosed.await() }

122+

} finally {

123+

shutdownReconnectHarness(harness, firstServer, secondServer)

124+

}

125+

}

126+8127

@Test

9128

fun bootstrapNodePairingRequiredKeepsReconnectActive() {

10129

val error =

@@ -113,4 +232,125 @@ class GatewaySessionReconnectTest {

113232

),

114233

)

115234

}

235+236+

private fun createReconnectHarness(): ReconnectHarness {

237+

val app = RuntimeEnvironment.getApplication()

238+

val sessionJob = SupervisorJob()

239+

val session =

240+

GatewaySession(

241+

scope = CoroutineScope(sessionJob + Dispatchers.Default),

242+

identityStore = DeviceIdentityStore(app),

243+

deviceAuthStore = ReconnectDeviceAuthStore(),

244+

onConnected = { _, _, _ -> },

245+

onDisconnected = { _ -> },

246+

onEvent = { _, _ -> },

247+

onInvoke = { GatewaySession.InvokeResult.ok("""{"handled":true}""") },

248+

)

249+

return ReconnectHarness(session = session, sessionJob = sessionJob)

250+

}

251+252+

private suspend fun connectNodeSession(

253+

session: GatewaySession,

254+

port: Int,

255+

) {

256+

session.connect(

257+

endpoint =

258+

GatewayEndpoint(

259+

stableId = "manual|127.0.0.1|$port",

260+

name = "test",

261+

host = "127.0.0.1",

262+

port = port,

263+

tlsEnabled = false,

264+

),

265+

token = "test-token",

266+

bootstrapToken = null,

267+

password = null,

268+

options =

269+

GatewayConnectOptions(

270+

role = "node",

271+

scopes = listOf("node:invoke"),

272+

caps = emptyList(),

273+

commands = emptyList(),

274+

permissions = emptyMap(),

275+

client =

276+

GatewayClientInfo(

277+

id = "openclaw-android-test",

278+

displayName = "Android Test",

279+

version = "1.0.0-test",

280+

platform = "android",

281+

mode = "node",

282+

instanceId = "android-test-instance",

283+

deviceFamily = "android",

284+

modelIdentifier = "test",

285+

),

286+

),

287+

tls = null,

288+

)

289+

}

290+291+

private suspend fun shutdownReconnectHarness(

292+

harness: ReconnectHarness,

293+

vararg servers: ReconnectServer,

294+

) {

295+

harness.session.disconnect()

296+

harness.sessionJob.cancelAndJoin()

297+

servers.forEach { it.shutdown() }

298+

}

299+300+

private fun connectResponseFrame(id: String): String = """{"type":"res","id":"$id","ok":true,"payload":{"snapshot":{"sessionDefaults":{"mainSessionKey":"main"}}}}"""

301+302+

private fun startGatewayServer(

303+

json: Json,

304+

onClosed: () -> Unit = {},

305+

onRequestFrame: (webSocket: WebSocket, id: String, method: String) -> Unit,

306+

): ReconnectServer {

307+

val sockets = ConcurrentLinkedQueue<WebSocket>()

308+

val server =

309+

MockWebServer().apply {

310+

dispatcher =

311+

object : Dispatcher() {

312+

override fun dispatch(request: RecordedRequest): MockResponse =

313+

MockResponse().withWebSocketUpgrade(

314+

object : WebSocketListener() {

315+

override fun onOpen(

316+

webSocket: WebSocket,

317+

response: Response,

318+

) {

319+

sockets += webSocket

320+

webSocket.send(LIFECYCLE_CONNECT_CHALLENGE_FRAME)

321+

}

322+323+

override fun onMessage(

324+

webSocket: WebSocket,

325+

text: String,

326+

) {

327+

val frame = json.parseToJsonElement(text).jsonObject

328+

if (frame["type"]?.jsonPrimitive?.content != "req") return

329+

val id = frame["id"]?.jsonPrimitive?.content ?: return

330+

val method = frame["method"]?.jsonPrimitive?.content ?: return

331+

onRequestFrame(webSocket, id, method)

332+

}

333+334+

override fun onClosing(

335+

webSocket: WebSocket,

336+

code: Int,

337+

reason: String,

338+

) {

339+

onClosed()

340+

}

341+342+

override fun onClosed(

343+

webSocket: WebSocket,

344+

code: Int,

345+

reason: String,

346+

) {

347+

onClosed()

348+

}

349+

},

350+

)

351+

}

352+

start()

353+

}

354+

return ReconnectServer(server = server, sockets = sockets)

355+

}

116356

}