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

推荐订阅源

N
News and Events Feed by Topic
Malwarebytes
Malwarebytes
Threat Intelligence Blog | Flashpoint
Threat Intelligence Blog | Flashpoint
C
Cybersecurity and Infrastructure Security Agency CISA
F
Future of Privacy Forum
C
Cisco Blogs
T
The Exploit Database - CXSecurity.com
A
Arctic Wolf
S
Securelist
K
Kaspersky official blog
S
Schneier on Security
T
ThreatConnect
T
Tenable Blog
Spread Privacy
Spread Privacy
T
True Tiger Recordings
AWS News Blog
AWS News Blog
F
Fox-IT International blog
量子位
T
Threatpost
V
Vulnerabilities – Threatpost
C
CERT Recently Published Vulnerability Notes
Cisco Talos Blog
Cisco Talos Blog
GbyAI
GbyAI
宝玉的分享
宝玉的分享
腾讯CDC
G
Google Developers Blog
aimingoo的专栏
aimingoo的专栏
Cyberwarzone
Cyberwarzone
有赞技术团队
有赞技术团队
S
SegmentFault 最新的问题
OSCHINA 社区最新新闻
OSCHINA 社区最新新闻
V
Visual Studio Blog
U
Unit 42
雷峰网
雷峰网
cs.CV updates on arXiv.org
cs.CV updates on arXiv.org
Simon Willison's Weblog
Simon Willison's Weblog
O
OpenAI News
freeCodeCamp Programming Tutorials: Python, JavaScript, Git & More
The GitHub Blog
The GitHub Blog
The Register - Security
The Register - Security
MyScale Blog
MyScale Blog
小众软件
小众软件
A
About on SuperTechFans
Last Week in AI
Last Week in AI
Y
Y Combinator Blog
博客园 - 三生石上(FineUI控件)
美团技术团队
Google Online Security Blog
Google Online Security Blog
P
Proofpoint News Feed
MongoDB | Blog
MongoDB | Blog

Recent Commits to openclaw:main

fix(openai): scope external codex auth to realtime fix(openai): prefer codex auth for GPT realtime · openclaw/openclaw@48c4f57 fix(openai): discover codex cli auth for provider checks · openclaw/openclaw@4656275 fix(android): keep talk mode on realtime relay · openclaw/openclaw@70614f8 test(android): add gateway connect adb probe · openclaw/openclaw@d7aa1f3 fix(android): stabilize realtime talk connection state · openclaw/openclaw@ffb02a5 fix(ci): stabilize deadcode and catalog checks · openclaw/openclaw@3db1508 fix(scripts): prebuild gateway cpu bench · openclaw/openclaw@ca70015 fix(e2e): harden bundled lifecycle probe on Windows · openclaw/openclaw@4798264 test(e2e): sample kitchen sink rpc peak rss · openclaw/openclaw@60c0f24 fix(scripts): remove stale deadcode allowlist entries · openclaw/openclaw@ea3bb92 fix(telegram): route polling diagnostics away from errors · openclaw/openclaw@b5c1199 fix(plugins): support linked source checkouts on Windows · openclaw/openclaw@793e300 fix(gateway): back off session tool mirrors under pressure (#84846) · openclaw/openclaw@42bdc94 fix(config): skip shell env fallback on Windows (#85739) · openclaw/openclaw@06bf302 fix(gateway): avoid duplicate session message broadcasts · openclaw/openclaw@1459044 fix: repair anchorless iMessage watch payloads · openclaw/openclaw@f37fbc9 fix(cli): route node status hints to stdout (#85780) · openclaw/openclaw@749692e fix(oc-path): support deep config edits (#86060) · openclaw/openclaw@3a72a30 fix(config): quiet benign metadata anomaly output · openclaw/openclaw@f3f4f29 fix(test): fail multi-node update regressions · openclaw/openclaw@732cf54 fix(google-vertex): support production ADC modes (#83971) · openclaw/openclaw@f09b4eb test(e2e): expose corrupt plugin deps smoke · openclaw/openclaw@fa3ff4d fix(codex): log app-server approval promotion trigger · openclaw/openclaw@d9af23f test(e2e): harden multi-node update smoke Clean up browser MCP subprocess tree (#85832) · openclaw/openclaw@8dc6b4d fix(agents): log warnings instead of swallowing subagent errors (#82943) · openclaw/openclaw@907bc03 fix(compaction): preserve partial summary on mid-chain chunk failure … · openclaw/openclaw@f0061dd fix(config): do not suppress recovery retry after failed backup resto… · openclaw/openclaw@5d174a5 chore: release 2026.5.25 fix(installer): support alpine cli installs · openclaw/openclaw@f68ed72 test(agents): keep runtime-plan provider mock current fix(scripts): launch env package scripts on Windows · openclaw/openclaw@4d4ce9e fix(agents): cache fallback provider resolution · openclaw/openclaw@3c8d101 fix(test): make import timing scripts Windows-safe · openclaw/openclaw@8ae9977 fix(telegram): transient Telegram pairing prompts (#85555) · openclaw/openclaw@8209426 fix(test): make max Vitest scripts Windows-safe · openclaw/openclaw@b681d5d fix(doctor): migrate Feishu account bot names (#86081) · openclaw/openclaw@9e8cc7e fix(scripts): prefilter conflict marker scans docs: add ClawSweeper review policy to AGENTS (#86197) · openclaw/openclaw@242e876 fix(installer): avoid before with npm release-age configs (#85491) · openclaw/openclaw@4742db6 fix(e2e): retry Windows kitchen sink probes · openclaw/openclaw@3e275a5 fix(installer): install node with apk on alpine fix(installer): detect musl linux shells · openclaw/openclaw@acfed37 perf(plugins,gateway): thread metadata snapshot + discovery through h… · openclaw/openclaw@8ccb11c fix(ui): split control ui runtime chunks · openclaw/openclaw@8bf4f7d refactor(config): extract GoogleChat schema into zod-schema.providers… · openclaw/openclaw@fe34141 fix(update): suppress internal handoff version warnings · openclaw/openclaw@6cc8244 test(e2e): select installable bundled plugins · openclaw/openclaw@0acc3e3 fix(scripts): harden Windows native opus install · openclaw/openclaw@43252c8 fix(agents): match runtime policy entries when session provider is em… fix(scripts): harden Windows generated formatting · openclaw/openclaw@0a98559 fix(mcp): bound tools/list during catalog discovery (#85063) · openclaw/openclaw@07f500a fix(test): focus plugin binding Docker smoke · openclaw/openclaw@dfa1a51 test(e2e): fail release memory indexing errors test(daemon): fail launchd integration bootstrap errors · openclaw/openclaw@af07769 feat(imessage): support thumb approval reactions (#85952) · openclaw/openclaw@5c7980f fix(crabbox): default macos aws runs on demand fix(scripts): preserve test passthrough args · openclaw/openclaw@e4332f7 fix(e2e): harden Windows plugin assertions fix(test): mount upgrade survivor helper · openclaw/openclaw@5f03154 fix(android): prevent stale chat during session switches fix(android): keep permission setup action visible · openclaw/openclaw@94bc18a style(android): sharpen voice mode surfaces · openclaw/openclaw@c452510 fix(android): hide internal chat content blocks · openclaw/openclaw@d86ed21 style(android): refine list surface spacing · openclaw/openclaw@955909c feat(android): add pair new gateway action · openclaw/openclaw@cc5eb97 Advance iMessage catchup cursor after live handling (#85475) · openclaw/openclaw@102555c fix(scripts): ignore forwarded arg separator · openclaw/openclaw@79ee70c fix(test): fail empty gateway startup samples · openclaw/openclaw@5a8ce6a fix(e2e): harden Windows kitchen sink assertions · openclaw/openclaw@87a2eba fix(e2e): harden Telegram credential paths on Windows · openclaw/openclaw@c643370 fix(android): align setup pairing scopes fix(android): complete qr setup operator handoff · openclaw/openclaw@be9bb77 fix(test): copy cleanup smoke prepare hook · openclaw/openclaw@dbc08f6 fix(secrets): allow hash in exec SecretRef ids (#86072) · openclaw/openclaw@675158c fix(media): use static image compression metadata · openclaw/openclaw@694d45e fix(release): verify large plugin npm packs fix(test): require kitchen sink diagnostic canaries · openclaw/openclaw@7e51f83 fix(scripts): harden Windows upgrade survivor recipe · openclaw/openclaw@483d7be fix(installer): count verify progress stage fix: Refine PR template for review state (#86054) fix(test): repair split agent shard runs · openclaw/openclaw@125d82c fix(codex): harden Windows protocol formatting · openclaw/openclaw@ce48e4c fix(openrouter): use endpoint context limits (#86041) · openclaw/openclaw@dd01a2e test(qa): remove brittle capability flip setup turn fix(telegram): migrate legacy cache sidecars · openclaw/openclaw@eb9b882 fix(telegram): migrate account topic cache sidecars · openclaw/openclaw@5cfb12f fix(scripts): harden Windows ZAI fallback repro · openclaw/openclaw@5be62e7 style(android): sharpen v2 screen rhythm · openclaw/openclaw@400d90a test(qa): extend capability flip setup budget · openclaw/openclaw@c91c3c6 fix(android): simplify gateway status copy fix(android): route offline voice to gateway setup fix(scripts): harden Windows control UI i18n commands · openclaw/openclaw@581c8a6 fix(android): stop operator chat subscription · openclaw/openclaw@5c15859 fix(test): fail missing kitchen sink rss samples test(qa): widen capability flip restart budget · openclaw/openclaw@c7d4e9e fix(android): smooth gateway pairing recovery · openclaw/openclaw@60e6ccd fix(test): suppress rolldown timing noise · openclaw/openclaw@6d9b388 style(android): fix talk mode ktlint formatting · openclaw/openclaw@01b284c
test(android): add voice mode adb e2e harness · openclaw/openclaw@e52a3b3
obviyus · 2026-05-25 · via Recent Commits to openclaw:main

@@ -0,0 +1,188 @@

1+

package ai.openclaw.app

2+3+

import android.app.Service

4+

import android.content.BroadcastReceiver

5+

import android.content.Context

6+

import android.content.Intent

7+

import android.os.IBinder

8+

import android.util.Base64

9+

import android.util.Log

10+

import kotlinx.coroutines.CoroutineScope

11+

import kotlinx.coroutines.Dispatchers

12+

import kotlinx.coroutines.SupervisorJob

13+

import kotlinx.coroutines.cancel

14+

import kotlinx.coroutines.delay

15+

import kotlinx.coroutines.launch

16+

import kotlinx.coroutines.withTimeout

17+

import kotlinx.serialization.json.JsonNull

18+

import kotlinx.serialization.json.JsonPrimitive

19+

import kotlinx.serialization.json.buildJsonObject

20+

import java.io.File

21+22+

private const val tag = "VoiceE2E"

23+

private const val resultFileName = "voice_e2e_result.json"

24+25+

class VoiceE2eReceiver : BroadcastReceiver() {

26+

override fun onReceive(

27+

context: Context,

28+

intent: Intent,

29+

) {

30+

context.startService(

31+

Intent(context, VoiceE2eService::class.java)

32+

.putExtras(intent),

33+

)

34+

}

35+

}

36+37+

class VoiceE2eService : Service() {

38+

private val serviceScope = CoroutineScope(SupervisorJob() + Dispatchers.IO)

39+40+

override fun onBind(intent: Intent?): IBinder? = null

41+42+

override fun onStartCommand(

43+

intent: Intent?,

44+

flags: Int,

45+

startId: Int,

46+

): Int {

47+

val command = intent ?: return START_NOT_STICKY

48+

serviceScope.launch {

49+

try {

50+

runCommand(command)

51+

} finally {

52+

stopSelf(startId)

53+

}

54+

}

55+

return START_NOT_STICKY

56+

}

57+58+

override fun onDestroy() {

59+

serviceScope.cancel()

60+

super.onDestroy()

61+

}

62+63+

private suspend fun runCommand(intent: Intent) {

64+

try {

65+

val app = applicationContext as NodeApp

66+

val runtime = app.ensureRuntime()

67+

val mode =

68+

intent

69+

.getDecodedStringExtra("mode")

70+

?.trim()

71+

.orEmpty()

72+

.ifEmpty { "both" }

73+

if (mode == "stop") {

74+

runtime.cancelMicCapture()

75+

runtime.setTalkModeEnabled(false)

76+

writeResult("""{"ok":true,"mode":"stop"}""")

77+

return

78+

}

79+80+

val connect = !intent.getBooleanExtra("noConnect", false)

81+

val connectTimeoutMs = intent.getLongExtra("connectTimeoutMs", 20_000L)

82+

if (connect) {

83+

configureGateway(runtime = runtime, intent = intent)

84+

}

85+

if (connect || !runtime.isConnected.value) {

86+

awaitGateway(runtime = runtime, timeoutMs = connectTimeoutMs)

87+

}

88+89+

startActivity(

90+

Intent(actionOpenVoiceE2e)

91+

.setClass(this, MainActivity::class.java)

92+

.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK or Intent.FLAG_ACTIVITY_SINGLE_TOP or Intent.FLAG_ACTIVITY_CLEAR_TOP),

93+

)

94+95+

val transcript =

96+

intent

97+

.getDecodedStringExtra("transcript")

98+

?.trim()

99+

.orEmpty()

100+

.ifEmpty { "Reply exactly: Android voice e2e normal path ok." }

101+

val realtimeReply =

102+

intent

103+

.getDecodedStringExtra("realtimeAssistant")

104+

?.trim()

105+

.orEmpty()

106+

.ifEmpty { "Android realtime voice e2e relay path ok." }

107+

val timeoutMs = intent.getLongExtra("timeoutMs", 60_000L)

108+

val result =

109+

runtime.runVoiceE2e(

110+

mode = mode,

111+

transcript = transcript,

112+

realtimeAssistantText = realtimeReply,

113+

timeoutMs = timeoutMs,

114+

)

115+

val resultJson = encodeResult(result)

116+

writeResult(resultJson)

117+

Log.i(tag, "PASS $resultJson")

118+

} catch (err: Throwable) {

119+

val resultJson =

120+

buildJsonObject {

121+

put("ok", JsonPrimitive(false))

122+

put("error", JsonPrimitive(err.message ?: err::class.java.simpleName))

123+

}.toString()

124+

writeResult(resultJson)

125+

Log.e(tag, "FAIL $resultJson", err)

126+

}

127+

}

128+129+

private fun configureGateway(

130+

runtime: NodeRuntime,

131+

intent: Intent,

132+

) {

133+

val host =

134+

intent

135+

.getDecodedStringExtra("host")

136+

?.trim()

137+

.orEmpty()

138+

.ifEmpty { "127.0.0.1" }

139+

val port = intent.getIntExtra("port", 18789)

140+

runtime.setManualEnabled(true)

141+

runtime.setManualHost(host)

142+

runtime.setManualPort(port)

143+

runtime.setManualTls(intent.getBooleanExtra("tls", false))

144+

runtime.setGatewayToken(intent.getDecodedStringExtra("token").orEmpty())

145+

runtime.setGatewayBootstrapToken(intent.getDecodedStringExtra("bootstrapToken").orEmpty())

146+

runtime.setGatewayPassword(intent.getDecodedStringExtra("password").orEmpty())

147+

runtime.setOnboardingCompleted(true)

148+

runtime.connectManual()

149+

}

150+151+

private suspend fun awaitGateway(

152+

runtime: NodeRuntime,

153+

timeoutMs: Long,

154+

) {

155+

withTimeout(timeoutMs) {

156+

while (!runtime.isConnected.value) {

157+

delay(100L)

158+

}

159+

}

160+

}

161+162+

private fun encodeResult(result: NodeRuntime.VoiceE2eResult): String =

163+

buildJsonObject {

164+

put("ok", JsonPrimitive(true))

165+

put("normal", result.normal?.let(::encodeSlice) ?: JsonNull)

166+

put("realtime", result.realtime?.let(::encodeSlice) ?: JsonNull)

167+

}.toString()

168+169+

private fun encodeSlice(slice: NodeRuntime.VoiceE2eSliceResult) =

170+

buildJsonObject {

171+

put("mode", JsonPrimitive(slice.mode))

172+

put("status", JsonPrimitive(slice.status))

173+

put("userText", slice.userText?.let(::JsonPrimitive) ?: JsonNull)

174+

put("assistantText", slice.assistantText?.let(::JsonPrimitive) ?: JsonNull)

175+

}

176+177+

private fun writeResult(json: String) {

178+

File(cacheDir, resultFileName).writeText(json)

179+

}

180+

}

181+182+

private fun Intent.getDecodedStringExtra(name: String): String? {

183+

val encoded = getStringExtra("${name}Base64")

184+

if (!encoded.isNullOrBlank()) {

185+

return String(Base64.decode(encoded, Base64.NO_WRAP), Charsets.UTF_8)

186+

}

187+

return getStringExtra(name)

188+

}