refactor(browser): share relay token + options validation tests

This commit is contained in:
Peter Steinberger
2026-02-24 04:23:16 +00:00
parent 8c5cf2d5b2
commit f9de17106a
4 changed files with 179 additions and 51 deletions

View File

@@ -1,4 +1,4 @@
import { buildRelayWsUrl, deriveRelayToken, isRetryableReconnectError, reconnectDelayMs } from './background-utils.js'
import { buildRelayWsUrl, isRetryableReconnectError, reconnectDelayMs } from './background-utils.js'
const DEFAULT_PORT = 18792

View File

@@ -0,0 +1,57 @@
const PORT_GUIDANCE = 'Use gateway port + 3 (for gateway 18789, relay is 18792).'
function hasCdpVersionShape(data) {
return !!data && typeof data === 'object' && 'Browser' in data && 'Protocol-Version' in data
}
export function classifyRelayCheckResponse(res, port) {
if (!res) {
return { action: 'throw', error: 'No response from service worker' }
}
if (res.status === 401) {
return { action: 'status', kind: 'error', message: 'Gateway token rejected. Check token and save again.' }
}
if (res.error) {
return { action: 'throw', error: res.error }
}
if (!res.ok) {
return { action: 'throw', error: `HTTP ${res.status}` }
}
const contentType = String(res.contentType || '')
if (!contentType.includes('application/json')) {
return {
action: 'status',
kind: 'error',
message: `Wrong port: this is likely the gateway, not the relay. ${PORT_GUIDANCE}`,
}
}
if (!hasCdpVersionShape(res.json)) {
return {
action: 'status',
kind: 'error',
message: `Wrong port: expected relay /json/version response. ${PORT_GUIDANCE}`,
}
}
return { action: 'status', kind: 'ok', message: `Relay reachable and authenticated at http://127.0.0.1:${port}/` }
}
export function classifyRelayCheckException(err, port) {
const message = String(err || '').toLowerCase()
if (message.includes('json') || message.includes('syntax')) {
return {
kind: 'error',
message: `Wrong port: this is not a relay endpoint. ${PORT_GUIDANCE}`,
}
}
return {
kind: 'error',
message: `Relay not reachable/authenticated at http://127.0.0.1:${port}/. Start OpenClaw browser relay and verify token.`,
}
}

View File

@@ -1,3 +1,6 @@
import { deriveRelayToken } from './background-utils.js'
import { classifyRelayCheckException, classifyRelayCheckResponse } from './options-validation.js'
const DEFAULT_PORT = 18792
function clampPort(value) {
@@ -13,17 +16,6 @@ function updateRelayUrl(port) {
el.textContent = `http://127.0.0.1:${port}/`
}
async function deriveRelayToken(gatewayToken, port) {
const enc = new TextEncoder()
const key = await crypto.subtle.importKey(
'raw', enc.encode(gatewayToken), { name: 'HMAC', hash: 'SHA-256' }, false, ['sign'],
)
const sig = await crypto.subtle.sign(
'HMAC', key, enc.encode(`openclaw-extension-relay-v1:${port}`),
)
return [...new Uint8Array(sig)].map((b) => b.toString(16).padStart(2, '0')).join('')
}
function setStatus(kind, message) {
const status = document.getElementById('status')
if (!status) return
@@ -47,46 +39,12 @@ async function checkRelayReachable(port, token) {
url,
token: relayToken,
})
if (!res) throw new Error('No response from service worker')
if (res.status === 401) {
setStatus('error', 'Gateway token rejected. Check token and save again.')
return
}
if (res.error) throw new Error(res.error)
if (!res.ok) throw new Error(`HTTP ${res.status}`)
// Validate that this is a CDP relay /json/version payload, not gateway HTML.
const contentType = String(res.contentType || '')
const data = res.json
if (!contentType.includes('application/json')) {
setStatus(
'error',
'Wrong port: this is likely the gateway, not the relay. Use gateway port + 3 (for gateway 18789, relay is 18792).',
)
return
}
if (!data || typeof data !== 'object' || !('Browser' in data) || !('Protocol-Version' in data)) {
setStatus(
'error',
'Wrong port: expected relay /json/version response. Use gateway port + 3 (for gateway 18789, relay is 18792).',
)
return
}
setStatus('ok', `Relay reachable and authenticated at http://127.0.0.1:${port}/`)
const result = classifyRelayCheckResponse(res, port)
if (result.action === 'throw') throw new Error(result.error)
setStatus(result.kind, result.message)
} catch (err) {
const message = String(err || '').toLowerCase()
if (message.includes('json') || message.includes('syntax')) {
setStatus(
'error',
'Wrong port: this is not a relay endpoint. Use gateway port + 3 (for gateway 18789, relay is 18792).',
)
} else {
setStatus(
'error',
`Relay not reachable/authenticated at http://127.0.0.1:${port}/. Start OpenClaw browser relay and verify token.`,
)
}
const result = classifyRelayCheckException(err, port)
setStatus(result.kind, result.message)
}
}