@tutti-os/browser-node 0.0.4 → 0.0.6
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
|
@@ -480,42 +480,6 @@ function goGuestForward(contents) {
|
|
|
480
480
|
}
|
|
481
481
|
contents.goForward();
|
|
482
482
|
}
|
|
483
|
-
function getPopupLogMetadata(url) {
|
|
484
|
-
try {
|
|
485
|
-
const parsed = new URL(url);
|
|
486
|
-
return {
|
|
487
|
-
popupOrigin: parsed.origin,
|
|
488
|
-
popupPath: parsed.pathname,
|
|
489
|
-
popupProtocol: parsed.protocol
|
|
490
|
-
};
|
|
491
|
-
} catch {
|
|
492
|
-
return {
|
|
493
|
-
popupOrigin: null,
|
|
494
|
-
popupPath: null,
|
|
495
|
-
popupProtocol: null
|
|
496
|
-
};
|
|
497
|
-
}
|
|
498
|
-
}
|
|
499
|
-
function allowBrowserNodeGuestPopupWindow({
|
|
500
|
-
logger,
|
|
501
|
-
nodeId,
|
|
502
|
-
url,
|
|
503
|
-
webContentsId
|
|
504
|
-
}) {
|
|
505
|
-
logger?.info?.("Browser Node guest popup allowed", {
|
|
506
|
-
nodeId,
|
|
507
|
-
webContentsId,
|
|
508
|
-
...getPopupLogMetadata(url)
|
|
509
|
-
});
|
|
510
|
-
return {
|
|
511
|
-
action: "allow",
|
|
512
|
-
overrideBrowserWindowOptions: {
|
|
513
|
-
height: 720,
|
|
514
|
-
show: true,
|
|
515
|
-
width: 520
|
|
516
|
-
}
|
|
517
|
-
};
|
|
518
|
-
}
|
|
519
483
|
function resolveBrowserNodeUrlError(resolved) {
|
|
520
484
|
if (resolved.errorCode === "invalid-url") {
|
|
521
485
|
return { code: "invalid-url" };
|
|
@@ -565,6 +529,19 @@ function emitBrowserNavigationFailed(input) {
|
|
|
565
529
|
type: "error"
|
|
566
530
|
});
|
|
567
531
|
}
|
|
532
|
+
function isGoogleGisOAuthPopupUrl(url) {
|
|
533
|
+
try {
|
|
534
|
+
const parsed = new URL(url);
|
|
535
|
+
if (parsed.protocol !== "https:" || parsed.hostname !== "accounts.google.com" || parsed.pathname !== "/o/oauth2/v2/auth" && parsed.pathname !== "/o/oauth2/auth") {
|
|
536
|
+
return false;
|
|
537
|
+
}
|
|
538
|
+
const isGisSdkPopup = parsed.searchParams.get("gsiwebsdk") === "gis_attributes";
|
|
539
|
+
const isPopupResponse = parsed.searchParams.get("display") === "popup" && (parsed.searchParams.get("response_mode") === "form_post" || parsed.searchParams.get("redirect_uri") === "gis_transform");
|
|
540
|
+
return isGisSdkPopup || isPopupResponse;
|
|
541
|
+
} catch {
|
|
542
|
+
return false;
|
|
543
|
+
}
|
|
544
|
+
}
|
|
568
545
|
function resolveBrowserNavigationOrigin(url) {
|
|
569
546
|
const resolved = resolveBrowserNavigationUrl(url);
|
|
570
547
|
if (!resolved.url) {
|
|
@@ -857,7 +834,7 @@ function createBrowserGuestManager({
|
|
|
857
834
|
webContentsId: session.webContentsId
|
|
858
835
|
});
|
|
859
836
|
emit({
|
|
860
|
-
reuseIfOpen:
|
|
837
|
+
reuseIfOpen: true,
|
|
861
838
|
sourceNodeId: session.nodeId,
|
|
862
839
|
type: "open-url",
|
|
863
840
|
url: resolved.url
|
|
@@ -1050,14 +1027,16 @@ function createBrowserGuestManager({
|
|
|
1050
1027
|
sessionPartition: session.sessionPartition,
|
|
1051
1028
|
webContentsId: input.webContentsId
|
|
1052
1029
|
});
|
|
1053
|
-
contents.setWindowOpenHandler?.(
|
|
1054
|
-
(
|
|
1055
|
-
logger,
|
|
1056
|
-
|
|
1057
|
-
|
|
1058
|
-
|
|
1059
|
-
|
|
1060
|
-
|
|
1030
|
+
contents.setWindowOpenHandler?.(({ url }) => {
|
|
1031
|
+
if (isGoogleGisOAuthPopupUrl(url)) {
|
|
1032
|
+
logger?.info?.("Browser Node allowing Google GIS OAuth popup", {
|
|
1033
|
+
nodeId: session.nodeId,
|
|
1034
|
+
webContentsId: session.webContentsId
|
|
1035
|
+
});
|
|
1036
|
+
return { action: "allow" };
|
|
1037
|
+
}
|
|
1038
|
+
return emitOpenUrlFromGuest(session, url);
|
|
1039
|
+
});
|
|
1061
1040
|
attachGuestListeners(session);
|
|
1062
1041
|
await applyPreferredColorSchemeToGuest(
|
|
1063
1042
|
session,
|
|
@@ -1312,7 +1291,7 @@ function isBrowserNodeWebviewAttach(params, allowedSessionPartitions) {
|
|
|
1312
1291
|
function shouldAllowBrowserNodeNativePopups(params) {
|
|
1313
1292
|
return params["data-browser-node-webview"] === "true" || isBrowserSessionPartitionAllowed(params.partition);
|
|
1314
1293
|
}
|
|
1315
|
-
function
|
|
1294
|
+
function getPopupLogMetadata(url) {
|
|
1316
1295
|
try {
|
|
1317
1296
|
const parsed = new URL(url);
|
|
1318
1297
|
return {
|
|
@@ -1328,23 +1307,21 @@ function getPopupLogMetadata2(url) {
|
|
|
1328
1307
|
};
|
|
1329
1308
|
}
|
|
1330
1309
|
}
|
|
1331
|
-
function
|
|
1310
|
+
function externalizeBrowserNodePopupWindow({
|
|
1332
1311
|
guestWebContentsId,
|
|
1333
1312
|
logger,
|
|
1313
|
+
openExternal,
|
|
1334
1314
|
url
|
|
1335
1315
|
}) {
|
|
1336
|
-
logger?.info?.("Browser Node
|
|
1316
|
+
logger?.info?.("Browser Node webview popup externalized", {
|
|
1337
1317
|
guestWebContentsId,
|
|
1338
|
-
...
|
|
1318
|
+
...getPopupLogMetadata(url)
|
|
1339
1319
|
});
|
|
1340
|
-
|
|
1341
|
-
|
|
1342
|
-
|
|
1343
|
-
|
|
1344
|
-
|
|
1345
|
-
width: 520
|
|
1346
|
-
}
|
|
1347
|
-
};
|
|
1320
|
+
const resolved = resolveBrowserNavigationUrl(url);
|
|
1321
|
+
if (resolved.url) {
|
|
1322
|
+
void Promise.resolve(openExternal(resolved.url)).catch(() => void 0);
|
|
1323
|
+
}
|
|
1324
|
+
return { action: "deny" };
|
|
1348
1325
|
}
|
|
1349
1326
|
function enforceBrowserWebviewSecurity({
|
|
1350
1327
|
allowedSessionPartitions,
|
|
@@ -1446,26 +1423,22 @@ function installBrowserWebviewSecurity({
|
|
|
1446
1423
|
}
|
|
1447
1424
|
applyBrowserGuestUserAgent(guestContents, logger);
|
|
1448
1425
|
if (pendingAttach.allowNativePopups) {
|
|
1449
|
-
guestContents.setWindowOpenHandler(
|
|
1450
|
-
|
|
1426
|
+
guestContents.setWindowOpenHandler(({ url }) => {
|
|
1427
|
+
return externalizeBrowserNodePopupWindow({
|
|
1451
1428
|
guestWebContentsId: guestContents.id ?? null,
|
|
1452
1429
|
logger,
|
|
1430
|
+
openExternal,
|
|
1453
1431
|
url
|
|
1454
|
-
})
|
|
1455
|
-
);
|
|
1432
|
+
});
|
|
1433
|
+
});
|
|
1456
1434
|
} else {
|
|
1457
1435
|
guestContents.setWindowOpenHandler(({ url }) => {
|
|
1458
|
-
|
|
1436
|
+
return externalizeBrowserNodePopupWindow({
|
|
1459
1437
|
guestWebContentsId: guestContents.id ?? null,
|
|
1460
|
-
|
|
1438
|
+
logger,
|
|
1439
|
+
openExternal,
|
|
1440
|
+
url
|
|
1461
1441
|
});
|
|
1462
|
-
const resolved = resolveBrowserNavigationUrl(url);
|
|
1463
|
-
if (resolved.url) {
|
|
1464
|
-
void Promise.resolve(openExternal(resolved.url)).catch(
|
|
1465
|
-
() => void 0
|
|
1466
|
-
);
|
|
1467
|
-
}
|
|
1468
|
-
return { action: "deny" };
|
|
1469
1442
|
});
|
|
1470
1443
|
}
|
|
1471
1444
|
onGuestAttached?.(guestContents);
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../../src/electron-main/loopbackPreviewProxy.ts","../../src/electron-main/guestManager.ts","../../src/electron-main/registerElectronMain.ts","../../src/electron-main/userAgent.ts","../../src/electron-main/webviewSecurity.ts"],"sourcesContent":["import {\n createServer,\n request as httpRequest,\n type IncomingHttpHeaders,\n type IncomingMessage,\n type Server,\n type ServerResponse\n} from \"node:http\";\nimport { request as httpsRequest } from \"node:https\";\nimport type { Socket } from \"node:net\";\nimport { pipeline } from \"node:stream\";\nimport WebSocket, { WebSocketServer } from \"ws\";\nimport type { RawData } from \"ws\";\nimport type { BrowserNodeSessionMode } from \"../core/types.ts\";\nimport type {\n BrowserNodeLoopbackPreviewRoutingOptions,\n BrowserNodeLoopbackPreviewTarget\n} from \"./loopbackPreview.ts\";\nimport type { BrowserNodeElectronLogger } from \"./types.ts\";\n\nconst defaultLoopbackPreviewCacheTtlMs = 30_000;\nconst loopbackHostPattern = /^(localhost|127(?:\\.\\d{1,3}){0,3})$/i;\n// Order matters: bypass all, subtract loopback so local previews can proxy,\n// then keep HTTPS/WSS direct because this proxy does not implement CONNECT.\nconst loopbackPreviewProxyBypassRules = \"*;<-loopback>;https://*;wss://*\";\nconst hopByHopHeaders = new Set([\n \"connection\",\n \"keep-alive\",\n \"proxy-authenticate\",\n \"proxy-authorization\",\n \"proxy-connection\",\n \"te\",\n \"trailer\",\n \"transfer-encoding\",\n \"upgrade\"\n]);\n\ninterface CachedLoopbackPreviewTarget {\n expiresAt: number;\n target: BrowserNodeLoopbackPreviewTarget | null;\n}\n\nexport interface BrowserNodePreviewSession {\n setProxy(input: {\n mode: \"fixed_servers\";\n proxyBypassRules: string;\n proxyRules: string;\n }): Promise<void>;\n}\n\nexport interface BrowserNodeLoopbackPreviewProxy {\n configureSession(input: {\n profileId: string | null;\n sessionMode: BrowserNodeSessionMode;\n sessionPartition?: string | null;\n }): Promise<void>;\n dispose(): Promise<void>;\n addressForTesting(): string;\n serverForTesting(): Server;\n}\n\nexport interface BrowserNodeLoopbackPreviewProxyInput {\n logger?: BrowserNodeElectronLogger;\n resolveSession: (input: {\n profileId: string | null;\n sessionMode: BrowserNodeSessionMode;\n sessionPartition?: string | null;\n }) => BrowserNodePreviewSession | Promise<BrowserNodePreviewSession>;\n routing: BrowserNodeLoopbackPreviewRoutingOptions;\n}\n\nexport function createBrowserNodeLoopbackPreviewProxy({\n logger,\n resolveSession,\n routing\n}: BrowserNodeLoopbackPreviewProxyInput): BrowserNodeLoopbackPreviewProxy {\n const configuredSessions = new WeakSet<BrowserNodePreviewSession>();\n const targetCache = new Map<string, CachedLoopbackPreviewTarget>();\n const downstreamWebSocketServer = new WebSocketServer({ noServer: true });\n let server: Server | null = null;\n let serverStartPromise: Promise<string> | null = null;\n\n const cacheTtlMs = routing.cacheTtlMs ?? defaultLoopbackPreviewCacheTtlMs;\n const fallback = routing.fallback ?? \"direct\";\n\n const pruneExpiredTargets = (now: number): void => {\n for (const [cacheKey, cachedTarget] of targetCache.entries()) {\n if (cachedTarget.expiresAt <= now) {\n targetCache.delete(cacheKey);\n }\n }\n };\n\n const start = async (): Promise<string> => {\n if (server?.listening) {\n return addressForTesting();\n }\n if (serverStartPromise) {\n return serverStartPromise;\n }\n\n serverStartPromise = new Promise<string>((resolve, reject) => {\n const nextServer = createServer((request, response) => {\n void handleRequest(request, response);\n });\n nextServer.on(\"upgrade\", (request, socket, head) => {\n void handleUpgrade(request, socket as Socket, head);\n });\n\n const cleanup = (): void => {\n nextServer.removeListener(\"error\", onError);\n nextServer.removeListener(\"listening\", onListening);\n };\n const onError = (error: Error): void => {\n cleanup();\n reject(error);\n };\n const onListening = (): void => {\n cleanup();\n server = nextServer;\n resolve(addressForTesting());\n };\n\n nextServer.once(\"error\", onError);\n nextServer.once(\"listening\", onListening);\n nextServer.listen(0, \"127.0.0.1\");\n }).finally(() => {\n serverStartPromise = null;\n });\n\n return serverStartPromise;\n };\n\n const resolveLoopbackTarget = async (\n originalUrl: URL\n ): Promise<BrowserNodeLoopbackPreviewTarget | null> => {\n const port = Number(originalUrl.port);\n if (!Number.isInteger(port) || port < 1 || port > 65535) {\n return null;\n }\n\n const now = Date.now();\n pruneExpiredTargets(now);\n\n const cacheKey = originalUrl.toString();\n const cached = targetCache.get(cacheKey);\n if (cached && cached.expiresAt > now) {\n return cached.target;\n }\n\n const nextTarget = await Promise.resolve(\n routing.resolver.resolveTarget({\n port,\n url: originalUrl.toString()\n })\n );\n\n const target = normalizeLoopbackPreviewTarget(nextTarget);\n targetCache.set(cacheKey, {\n expiresAt: now + Math.max(0, cacheTtlMs),\n target\n });\n return target;\n };\n\n const configureSession = async (input: {\n profileId: string | null;\n sessionMode: BrowserNodeSessionMode;\n sessionPartition?: string | null;\n }): Promise<void> => {\n const nextSession = await resolveSession(input);\n if (configuredSessions.has(nextSession)) {\n return;\n }\n const address = await start();\n await nextSession.setProxy({\n mode: \"fixed_servers\",\n proxyBypassRules: loopbackPreviewProxyBypassRules,\n proxyRules: `http=${address}`\n });\n configuredSessions.add(nextSession);\n };\n\n const dispose = async (): Promise<void> => {\n targetCache.clear();\n downstreamWebSocketServer.close();\n const activeServer = server;\n server = null;\n if (!activeServer || !activeServer.listening) {\n return;\n }\n await new Promise<void>((resolve) => {\n activeServer.close(() => resolve());\n });\n };\n\n const handleRequest = async (\n request: IncomingMessage,\n response: ServerResponse\n ): Promise<void> => {\n const originalUrl = resolveProxyRequestUrl(request);\n if (!originalUrl) {\n response.writeHead(400);\n response.end(\"invalid proxy request\");\n return;\n }\n if (originalUrl.protocol !== \"http:\") {\n response.writeHead(400);\n response.end(\"unsupported proxy request\");\n return;\n }\n\n const targetContext = await resolveTargetRequestContext(originalUrl);\n if (!targetContext.targetUrl) {\n response.writeHead(502);\n response.end(\"unable to resolve loopback preview target\");\n return;\n }\n\n const headers = filterProxyHeaders(request.headers);\n headers.set(\"host\", targetContext.targetUrl.host);\n\n const forward =\n targetContext.targetUrl.protocol === \"https:\"\n ? httpsRequest\n : httpRequest;\n const upstream = forward(\n targetContext.targetUrl,\n {\n headers: Object.fromEntries(headers.entries()),\n method: request.method\n },\n (upstreamResponse) => {\n const responseHeaders = filterProxyHeaders(upstreamResponse.headers);\n const location = responseHeaders.get(\"location\");\n if (location && targetContext.loopbackTarget) {\n const rewritten = rewriteLoopbackLocation({\n location,\n originalUrl,\n targetUrl: targetContext.loopbackTarget.targetUrl\n });\n if (rewritten) {\n responseHeaders.set(\"location\", rewritten);\n }\n }\n\n response.writeHead(\n upstreamResponse.statusCode ?? 502,\n upstreamResponse.statusMessage,\n Object.fromEntries(responseHeaders.entries())\n );\n pipeline(upstreamResponse, response, () => undefined);\n }\n );\n\n upstream.on(\"error\", (error) => {\n logger?.warn?.(\"Browser Node loopback preview request failed\", {\n error: normalizeProxyError(error),\n originalUrl: originalUrl.toString(),\n targetUrl: targetContext.targetUrl?.toString() ?? null,\n workspaceId: targetContext.loopbackTarget?.workspaceId ?? null\n });\n if (!response.headersSent) {\n response.writeHead(502);\n }\n response.end(\n `loopback preview upstream request failed: ${normalizeProxyError(error)}`\n );\n });\n\n pipeline(request, upstream, () => undefined);\n };\n\n const handleUpgrade = async (\n request: IncomingMessage,\n socket: Socket,\n head: Buffer\n ): Promise<void> => {\n const originalUrl = resolveProxyRequestUrl(request, \"ws:\");\n if (!originalUrl) {\n socket.destroy();\n return;\n }\n if (originalUrl.protocol !== \"ws:\") {\n socket.destroy();\n return;\n }\n\n const targetContext = await resolveTargetRequestContext(originalUrl);\n const targetUrl = targetContext.targetUrl;\n if (!targetUrl) {\n socket.destroy();\n return;\n }\n\n downstreamWebSocketServer.handleUpgrade(\n request,\n socket,\n head,\n (downstream) => {\n const headers = Object.fromEntries(\n filterProxyHeaders(request.headers).entries()\n );\n headers.host = targetUrl.host;\n\n const upstream = new WebSocket(targetUrl, { headers });\n const pendingMessages: Array<{ data: RawData; isBinary: boolean }> = [];\n\n downstream.on(\"message\", (data, isBinary) => {\n if (upstream.readyState === WebSocket.OPEN) {\n upstream.send(data, { binary: isBinary });\n return;\n }\n pendingMessages.push({ data, isBinary });\n });\n\n upstream.once(\"open\", () => {\n for (const nextMessage of pendingMessages.splice(0)) {\n upstream.send(nextMessage.data, { binary: nextMessage.isBinary });\n }\n });\n\n upstream.on(\"message\", (data, isBinary) => {\n if (downstream.readyState === WebSocket.OPEN) {\n downstream.send(data, { binary: isBinary });\n }\n });\n\n upstream.once(\"close\", (code, reason) => {\n if (\n downstream.readyState === WebSocket.OPEN ||\n downstream.readyState === WebSocket.CONNECTING\n ) {\n downstream.close(normalizeWebSocketCloseCode(code), reason);\n }\n });\n\n downstream.once(\"close\", (code, reason) => {\n if (\n upstream.readyState === WebSocket.OPEN ||\n upstream.readyState === WebSocket.CONNECTING\n ) {\n upstream.close(normalizeWebSocketCloseCode(code), reason);\n }\n });\n\n upstream.once(\"error\", (error) => {\n logger?.warn?.(\"Browser Node loopback preview websocket failed\", {\n error: normalizeProxyError(error),\n originalUrl: originalUrl.toString(),\n targetUrl: targetContext.targetUrl?.toString() ?? null,\n workspaceId: targetContext.loopbackTarget?.workspaceId ?? null\n });\n downstream.close(1011, \"loopback preview websocket upstream failed\");\n });\n }\n );\n };\n\n const resolveTargetRequestContext = async (\n originalUrl: URL\n ): Promise<{\n loopbackTarget: BrowserNodeLoopbackPreviewTarget | null;\n targetUrl: URL | null;\n }> => {\n if (!isLoopbackUrl(originalUrl)) {\n return {\n loopbackTarget: null,\n targetUrl: null\n };\n }\n\n const loopbackTarget = await resolveLoopbackTarget(originalUrl);\n if (loopbackTarget) {\n return {\n loopbackTarget,\n targetUrl: buildTargetRequestUrl(loopbackTarget.targetUrl, originalUrl)\n };\n }\n\n if (fallback === \"direct\") {\n return {\n loopbackTarget: null,\n targetUrl: cloneUrl(originalUrl)\n };\n }\n\n return {\n loopbackTarget: null,\n targetUrl: null\n };\n };\n\n const addressForTesting = (): string => {\n const address = server?.address();\n if (!address || typeof address === \"string\") {\n throw new Error(\"Browser Node loopback preview proxy is not listening\");\n }\n return `127.0.0.1:${address.port}`;\n };\n\n const serverForTesting = (): Server => {\n if (!server) {\n throw new Error(\"Browser Node loopback preview proxy is not started\");\n }\n return server;\n };\n\n return {\n addressForTesting,\n configureSession,\n dispose,\n serverForTesting\n };\n}\n\nfunction normalizeLoopbackPreviewTarget(\n input: BrowserNodeLoopbackPreviewTarget | null | undefined\n): BrowserNodeLoopbackPreviewTarget | null {\n if (!input) {\n return null;\n }\n const normalizedTargetUrl = input.targetUrl.trim();\n if (normalizedTargetUrl.length === 0) {\n return null;\n }\n try {\n const parsed = new URL(normalizedTargetUrl);\n if (parsed.protocol !== \"http:\" && parsed.protocol !== \"https:\") {\n return null;\n }\n return {\n targetUrl: parsed.toString(),\n workspaceId: input.workspaceId\n };\n } catch {\n return null;\n }\n}\n\nfunction resolveProxyRequestUrl(\n request: IncomingMessage,\n defaultProtocol = \"http:\"\n): URL | null {\n const rawUrl = request.url ?? \"\";\n try {\n if (/^[a-z][a-z\\d+\\-.]*:\\/\\//i.test(rawUrl)) {\n return new URL(rawUrl);\n }\n const host = request.headers.host;\n if (typeof host !== \"string\" || host.trim().length === 0) {\n return null;\n }\n const path = rawUrl.startsWith(\"/\") ? rawUrl : `/${rawUrl}`;\n return new URL(`${defaultProtocol}//${host}${path}`);\n } catch {\n return null;\n }\n}\n\nfunction filterProxyHeaders(input: Headers | IncomingHttpHeaders): Headers {\n const output = new Headers();\n const appendValue = (key: string, value: string): void => {\n if (hopByHopHeaders.has(key.toLowerCase())) {\n return;\n }\n output.append(key, value);\n };\n\n if (input instanceof Headers) {\n input.forEach((value, key) => {\n appendValue(key, value);\n });\n return output;\n }\n\n for (const [key, value] of Object.entries(input)) {\n if (Array.isArray(value)) {\n for (const nextValue of value) {\n appendValue(key, nextValue);\n }\n continue;\n }\n if (typeof value === \"string\") {\n appendValue(key, value);\n }\n }\n return output;\n}\n\nfunction isLoopbackUrl(url: URL): boolean {\n return (\n (url.protocol === \"http:\" || url.protocol === \"ws:\") &&\n loopbackHostPattern.test(url.hostname) &&\n url.port.length > 0\n );\n}\n\nfunction buildTargetRequestUrl(\n targetUrl: string,\n originalUrl: URL\n): URL | null {\n try {\n const base = new URL(targetUrl);\n const normalizedBasePath = normalizeBasePath(base.pathname);\n const nextUrl = new URL(base.toString());\n nextUrl.pathname = joinBasePath(normalizedBasePath, originalUrl.pathname);\n nextUrl.search = originalUrl.search;\n nextUrl.hash = \"\";\n if (originalUrl.protocol === \"ws:\") {\n nextUrl.protocol = nextUrl.protocol === \"https:\" ? \"wss:\" : \"ws:\";\n }\n return nextUrl;\n } catch {\n return null;\n }\n}\n\nfunction rewriteLoopbackLocation({\n location,\n originalUrl,\n targetUrl\n}: {\n location: string;\n originalUrl: URL;\n targetUrl: string;\n}): string | null {\n try {\n const targetBase = new URL(targetUrl);\n const resolvedLocation = new URL(location, targetBase);\n if (resolvedLocation.origin !== targetBase.origin) {\n return location;\n }\n\n const basePath = normalizeBasePath(targetBase.pathname);\n const strippedPath =\n stripBasePath(basePath, resolvedLocation.pathname) ??\n resolvedLocation.pathname;\n\n const loopbackUrl = new URL(originalUrl.origin);\n resolvedLocation.protocol = loopbackUrl.protocol;\n resolvedLocation.host = loopbackUrl.host;\n resolvedLocation.pathname = strippedPath;\n resolvedLocation.username = \"\";\n resolvedLocation.password = \"\";\n return resolvedLocation.toString();\n } catch {\n return location;\n }\n}\n\nfunction normalizeBasePath(pathname: string): string {\n const normalized = pathname.startsWith(\"/\") ? pathname : `/${pathname}`;\n return normalized.endsWith(\"/\") ? normalized : `${normalized}/`;\n}\n\nfunction joinBasePath(basePath: string, requestPath: string): string {\n const normalizedRequestPath =\n requestPath === \"/\" ? \"\" : requestPath.replace(/^\\/+/, \"\");\n return normalizedRequestPath.length > 0\n ? `${basePath}${normalizedRequestPath}`.replace(/\\/{2,}/g, \"/\")\n : basePath;\n}\n\nfunction stripBasePath(basePath: string, pathname: string): string | null {\n const normalizedPath = pathname.startsWith(\"/\") ? pathname : `/${pathname}`;\n if (normalizedPath === basePath.slice(0, -1)) {\n return \"/\";\n }\n if (!normalizedPath.startsWith(basePath)) {\n return null;\n }\n const suffix = normalizedPath.slice(basePath.length);\n return suffix.length > 0 ? `/${suffix}` : \"/\";\n}\n\nfunction cloneUrl(url: URL): URL | null {\n try {\n return new URL(url.toString());\n } catch {\n return null;\n }\n}\n\nfunction normalizeProxyError(error: unknown): string {\n if (error instanceof Error) {\n const code =\n typeof (error as NodeJS.ErrnoException).code === \"string\"\n ? ` ${(error as NodeJS.ErrnoException).code}`\n : \"\";\n return `${error.message}${code}`;\n }\n return String(error);\n}\n\nfunction normalizeWebSocketCloseCode(code: number): number {\n if (code === 1000 || (code >= 3000 && code <= 4999)) {\n return code;\n }\n return 1000;\n}\n","import type {\n BrowserNodeLifecycle,\n BrowserNodeNavigationPolicy,\n BrowserNodeRuntimeError,\n BrowserNodeSessionMode\n} from \"../core/types.ts\";\nimport {\n normalizeHostBrowserComparableUrl,\n resolveBrowserNavigationUrl,\n resolveHostBrowserNavigationUrl,\n type BrowserNavigationUrlResolution\n} from \"../core/url.ts\";\nimport type {\n BrowserGuestManager,\n BrowserGuestManagerInput,\n BrowserGuestNativeImage,\n BrowserPreferredColorScheme,\n BrowserGuestWebContents\n} from \"./types.ts\";\n\nconst browserPreviewMaxWidth = 260;\nconst browserPreviewMaxHeight = 170;\nconst abortedNavigationErrorCode = -3;\n\ninterface BrowserGuestSession {\n appliedColorScheme: BrowserPreferredColorScheme | null;\n contents: BrowserGuestWebContents | null;\n desiredUrl: string;\n lifecycle: BrowserNodeLifecycle;\n listeners: Array<{\n event: string;\n listener: (...args: unknown[]) => void;\n }>;\n navigationFailureSequence: number;\n navigationPolicy: BrowserNodeNavigationPolicy | null;\n nodeId: string;\n profileId: string | null;\n sessionMode: BrowserNodeSessionMode;\n sessionPartition: string | null;\n webContentsId: number | null;\n}\n\nfunction canGuestGoBack(contents: BrowserGuestWebContents): boolean {\n return contents.navigationHistory?.canGoBack() ?? contents.canGoBack();\n}\n\nfunction canGuestGoForward(contents: BrowserGuestWebContents): boolean {\n return contents.navigationHistory?.canGoForward() ?? contents.canGoForward();\n}\n\nfunction goGuestBack(contents: BrowserGuestWebContents): void {\n if (contents.navigationHistory) {\n contents.navigationHistory.goBack();\n return;\n }\n contents.goBack();\n}\n\nfunction goGuestForward(contents: BrowserGuestWebContents): void {\n if (contents.navigationHistory) {\n contents.navigationHistory.goForward();\n return;\n }\n contents.goForward();\n}\n\nfunction getPopupLogMetadata(url: string): Record<string, unknown> {\n try {\n const parsed = new URL(url);\n return {\n popupOrigin: parsed.origin,\n popupPath: parsed.pathname,\n popupProtocol: parsed.protocol\n };\n } catch {\n return {\n popupOrigin: null,\n popupPath: null,\n popupProtocol: null\n };\n }\n}\n\nfunction allowBrowserNodeGuestPopupWindow({\n logger,\n nodeId,\n url,\n webContentsId\n}: {\n logger: BrowserGuestManagerInput[\"logger\"];\n nodeId: string;\n url: string;\n webContentsId: number | null;\n}) {\n logger?.info?.(\"Browser Node guest popup allowed\", {\n nodeId,\n webContentsId,\n ...getPopupLogMetadata(url)\n });\n return {\n action: \"allow\" as const,\n overrideBrowserWindowOptions: {\n height: 720,\n show: true,\n width: 520\n }\n };\n}\n\nfunction resolveBrowserNodeUrlError(\n resolved: BrowserNavigationUrlResolution\n): BrowserNodeRuntimeError {\n if (resolved.errorCode === \"invalid-url\") {\n return { code: \"invalid-url\" };\n }\n\n if (resolved.errorCode === \"unsupported-protocol\") {\n return {\n code: \"unsupported-protocol\",\n params: resolved.errorParams\n };\n }\n\n return { code: \"unsupported-url\" };\n}\n\nfunction resizeBrowserPreviewImage(\n image: BrowserGuestNativeImage\n): BrowserGuestNativeImage {\n if (image.isEmpty?.() === true || !image.resize || !image.getSize) {\n return image;\n }\n\n const size = image.getSize();\n if (size.width <= 0 || size.height <= 0) {\n return image;\n }\n\n const scale = Math.min(\n 1,\n browserPreviewMaxWidth / size.width,\n browserPreviewMaxHeight / size.height\n );\n if (scale >= 1) {\n return image;\n }\n\n return image.resize({\n height: Math.max(1, Math.round(size.height * scale)),\n quality: \"good\",\n width: Math.max(1, Math.round(size.width * scale))\n });\n}\n\nfunction isAbortedNavigationError(input: {\n errorCode?: number;\n errorDescription?: string;\n}): boolean {\n return (\n input.errorCode === abortedNavigationErrorCode ||\n input.errorDescription === \"ERR_ABORTED\"\n );\n}\n\nfunction isHttpErrorStatusCode(statusCode: number | undefined): boolean {\n return statusCode !== undefined && statusCode >= 400;\n}\n\nfunction emitBrowserNavigationFailed(input: {\n emit: BrowserGuestManagerInput[\"emit\"];\n errorCode?: number;\n errorDescription?: string;\n nodeId: string;\n}): void {\n input.emit({\n code: \"navigation-failed\",\n diagnosticMessage: input.errorDescription,\n nodeId: input.nodeId,\n params:\n input.errorCode === undefined\n ? undefined\n : { errorCode: input.errorCode },\n type: \"error\"\n });\n}\n\nfunction resolveBrowserNavigationOrigin(url: string): string | null {\n const resolved = resolveBrowserNavigationUrl(url);\n if (!resolved.url) {\n return null;\n }\n\n try {\n return new URL(resolved.url).origin;\n } catch {\n return null;\n }\n}\n\nfunction isBrowserNavigationAllowedByPolicy(input: {\n policy: BrowserNodeNavigationPolicy | null;\n url: string;\n}): boolean {\n if (!input.policy) {\n return true;\n }\n\n if (input.policy.mode === \"same-origin\") {\n const policyOrigin = resolveBrowserNavigationOrigin(input.policy.originUrl);\n const nextOrigin = resolveBrowserNavigationOrigin(input.url);\n return (\n policyOrigin !== null &&\n nextOrigin !== null &&\n policyOrigin === nextOrigin\n );\n }\n\n return true;\n}\n\nasync function applyPreferredColorSchemeToGuest(\n session: BrowserGuestSession,\n logger: BrowserGuestManagerInput[\"logger\"],\n syncPreferredColorScheme: BrowserGuestManagerInput[\"syncPreferredColorScheme\"],\n scheme: BrowserPreferredColorScheme | null\n): Promise<void> {\n const contents = session.contents;\n if (\n !contents ||\n contents.isDestroyed() ||\n !syncPreferredColorScheme ||\n scheme === null ||\n session.appliedColorScheme === scheme\n ) {\n return;\n }\n\n try {\n await syncPreferredColorScheme(contents, scheme);\n session.appliedColorScheme = scheme;\n } catch (error) {\n session.appliedColorScheme = null;\n logger?.warn?.(\"Browser Node failed to sync guest color scheme\", {\n error: error instanceof Error ? error.message : String(error),\n nodeId: session.nodeId,\n scheme,\n webContentsId: session.webContentsId\n });\n }\n}\n\nexport function createBrowserGuestManager({\n emit,\n getPreferredColorScheme,\n logger,\n openExternal,\n prepareSession,\n resolveWebContents,\n syncPreferredColorScheme,\n subscribePreferredColorScheme\n}: BrowserGuestManagerInput): BrowserGuestManager {\n const sessions = new Map<string, BrowserGuestSession>();\n const nodeIdByWebContentsId = new Map<number, string>();\n let preferredColorScheme = getPreferredColorScheme?.() ?? null;\n\n const getSession = (\n nodeId: string,\n input?: {\n navigationPolicy?: BrowserNodeNavigationPolicy | null;\n profileId?: string | null;\n sessionMode?: BrowserNodeSessionMode;\n sessionPartition?: string | null;\n url?: string;\n }\n ): BrowserGuestSession => {\n const existing = sessions.get(nodeId);\n if (existing) {\n if (input?.profileId !== undefined) {\n existing.profileId = input.profileId;\n }\n if (input?.sessionMode !== undefined) {\n existing.sessionMode = input.sessionMode;\n }\n if (input?.sessionPartition !== undefined) {\n existing.sessionPartition = input.sessionPartition;\n }\n if (input?.navigationPolicy !== undefined) {\n existing.navigationPolicy = input.navigationPolicy;\n }\n if (input?.url !== undefined) {\n existing.desiredUrl = input.url;\n }\n return existing;\n }\n\n const session: BrowserGuestSession = {\n appliedColorScheme: null,\n contents: null,\n desiredUrl: input?.url ?? \"about:blank\",\n lifecycle: \"cold\",\n listeners: [],\n navigationFailureSequence: 0,\n navigationPolicy: input?.navigationPolicy ?? null,\n nodeId,\n profileId: input?.profileId ?? null,\n sessionMode: input?.sessionMode ?? \"shared\",\n sessionPartition: input?.sessionPartition ?? null,\n webContentsId: null\n };\n sessions.set(nodeId, session);\n return session;\n };\n\n const publishState = (session: BrowserGuestSession): void => {\n const contents =\n session.contents && !session.contents.isDestroyed()\n ? session.contents\n : null;\n emit({\n canGoBack: contents ? canGuestGoBack(contents) : false,\n canGoForward: contents ? canGuestGoForward(contents) : false,\n isAttachedToWindow: Boolean(contents),\n isLoading: contents ? contents.isLoading() : false,\n isOccluded: session.lifecycle === \"cold\",\n lifecycle: session.lifecycle,\n nodeId: session.nodeId,\n title: contents ? contents.getTitle() || null : null,\n type: \"state\",\n url: contents\n ? contents.getURL() || session.desiredUrl\n : session.desiredUrl\n });\n };\n\n const detachGuest = (session: BrowserGuestSession): void => {\n const contents = session.contents;\n const webContentsId = session.webContentsId;\n if (contents) {\n for (const record of session.listeners) {\n contents.off(record.event, record.listener);\n }\n }\n session.listeners = [];\n session.contents = null;\n session.appliedColorScheme = null;\n session.webContentsId = null;\n if (\n webContentsId !== null &&\n nodeIdByWebContentsId.get(webContentsId) === session.nodeId\n ) {\n nodeIdByWebContentsId.delete(webContentsId);\n }\n publishState(session);\n };\n\n const handlePreferredColorSchemeChange = (\n scheme: BrowserPreferredColorScheme\n ) => {\n preferredColorScheme = scheme;\n for (const session of sessions.values()) {\n session.appliedColorScheme = null;\n void applyPreferredColorSchemeToGuest(\n session,\n logger,\n syncPreferredColorScheme,\n scheme\n ).catch(() => undefined);\n }\n };\n\n const unsubscribePreferredColorScheme =\n subscribePreferredColorScheme?.(handlePreferredColorSchemeChange) ?? null;\n\n const attachGuestListeners = (session: BrowserGuestSession): void => {\n const contents = session.contents;\n if (!contents) {\n return;\n }\n\n const onStateChange = () => publishState(session);\n const onDidNavigate = (...args: unknown[]) => {\n const url = typeof args[1] === \"string\" ? args[1] : undefined;\n const statusCode = typeof args[2] === \"number\" ? args[2] : undefined;\n const statusText = typeof args[3] === \"string\" ? args[3] : undefined;\n publishState(session);\n if (!isHttpErrorStatusCode(statusCode)) {\n return;\n }\n\n logger?.warn?.(\"Browser Node guest navigation returned HTTP error\", {\n currentUrl: contents.getURL(),\n desiredUrl: session.desiredUrl,\n nodeId: session.nodeId,\n statusCode,\n statusText,\n url,\n webContentsId: session.webContentsId\n });\n emit({\n code: \"navigation-failed\",\n diagnosticMessage: statusText,\n nodeId: session.nodeId,\n params: {\n statusCode,\n ...(statusText ? { statusText } : {})\n },\n type: \"error\"\n });\n };\n const onFailLoad = (...args: unknown[]) => {\n const errorCode = typeof args[1] === \"number\" ? args[1] : undefined;\n const errorDescription =\n typeof args[2] === \"string\" ? args[2] : undefined;\n const validatedUrl = typeof args[3] === \"string\" ? args[3] : undefined;\n const isMainFrame = typeof args[4] === \"boolean\" ? args[4] : undefined;\n if (isAbortedNavigationError({ errorCode, errorDescription })) {\n publishState(session);\n return;\n }\n logger?.warn?.(\"Browser Node guest navigation failed\", {\n currentUrl: contents.getURL(),\n desiredUrl: session.desiredUrl,\n errorCode,\n errorDescription,\n isMainFrame,\n nodeId: session.nodeId,\n validatedUrl,\n webContentsId: session.webContentsId\n });\n session.navigationFailureSequence += 1;\n publishState(session);\n emitBrowserNavigationFailed({\n emit,\n errorCode,\n errorDescription,\n nodeId: session.nodeId\n });\n };\n const onDestroyed = () => detachGuest(session);\n const onWillNavigate = (...args: unknown[]) => {\n const event =\n args[0] && typeof args[0] === \"object\" && \"preventDefault\" in args[0]\n ? (args[0] as { preventDefault?: () => void })\n : null;\n const url = typeof args[1] === \"string\" ? args[1] : \"\";\n if (\n url.length === 0 ||\n isBrowserNavigationAllowedByPolicy({\n policy: session.navigationPolicy,\n url\n })\n ) {\n return;\n }\n\n event?.preventDefault?.();\n emitOpenUrlFromGuest(session, url);\n publishState(session);\n };\n\n const records: BrowserGuestSession[\"listeners\"] = [\n { event: \"did-start-loading\", listener: onStateChange },\n { event: \"did-stop-loading\", listener: onStateChange },\n { event: \"did-navigate\", listener: onDidNavigate },\n { event: \"did-navigate-in-page\", listener: onStateChange },\n { event: \"page-title-updated\", listener: onStateChange },\n { event: \"will-navigate\", listener: onWillNavigate },\n { event: \"did-fail-load\", listener: onFailLoad },\n { event: \"destroyed\", listener: onDestroyed }\n ];\n\n for (const record of records) {\n contents.on(record.event, record.listener);\n }\n session.listeners = records;\n };\n\n const loadDesiredUrl = async (\n session: BrowserGuestSession\n ): Promise<void> => {\n const contents = session.contents;\n if (!contents || contents.isDestroyed()) {\n publishState(session);\n return;\n }\n\n const resolved = resolveHostBrowserNavigationUrl(session.desiredUrl);\n if (!resolved.url) {\n emit({\n ...resolveBrowserNodeUrlError(resolved),\n nodeId: session.nodeId,\n type: \"error\"\n });\n publishState(session);\n return;\n }\n if (\n !isBrowserNavigationAllowedByPolicy({\n policy: session.navigationPolicy,\n url: resolved.url\n })\n ) {\n emitOpenUrlFromGuest(session, resolved.url);\n publishState(session);\n return;\n }\n\n const currentComparable = normalizeHostBrowserComparableUrl(\n contents.getURL()\n );\n const nextComparable = normalizeHostBrowserComparableUrl(resolved.url);\n if (currentComparable && currentComparable === nextComparable) {\n publishState(session);\n return;\n }\n\n const failureSequenceBeforeLoad = session.navigationFailureSequence;\n try {\n await contents.loadURL(resolved.url);\n publishState(session);\n } catch (error) {\n const message = error instanceof Error ? error.message : String(error);\n logger?.warn?.(\"Browser Node guest loadURL failed\", {\n currentUrl: contents.getURL(),\n desiredUrl: session.desiredUrl,\n error: message,\n nodeId: session.nodeId,\n resolvedUrl: resolved.url,\n webContentsId: session.webContentsId\n });\n if (session.navigationFailureSequence !== failureSequenceBeforeLoad) {\n return;\n }\n publishState(session);\n emitBrowserNavigationFailed({\n emit,\n errorDescription: message,\n nodeId: session.nodeId\n });\n }\n };\n\n const emitOpenUrlFromGuest = (\n session: BrowserGuestSession,\n url: string\n ): { action: \"deny\" } => {\n const resolved = resolveBrowserNavigationUrl(url);\n if (resolved.url) {\n logger?.info?.(\"Browser Node guest emitted open-url\", {\n nodeId: session.nodeId,\n url: resolved.url,\n webContentsId: session.webContentsId\n });\n emit({\n reuseIfOpen: false,\n sourceNodeId: session.nodeId,\n type: \"open-url\",\n url: resolved.url\n });\n return { action: \"deny\" };\n }\n void Promise.resolve(openExternal(url)).catch((error: unknown) => {\n logger?.warn?.(\"Browser Node openExternal failed\", {\n error: error instanceof Error ? error.message : String(error)\n });\n });\n return { action: \"deny\" };\n };\n\n return {\n async activate(input) {\n const resolved = resolveHostBrowserNavigationUrl(input.url);\n if (!resolved.url) {\n throw new Error(\"Browser Node rejected navigation URL\");\n }\n const session = getSession(input.nodeId, {\n navigationPolicy: input.navigationPolicy,\n profileId: input.profileId,\n sessionMode: input.sessionMode,\n sessionPartition: input.sessionPartition,\n url: resolved.url\n });\n session.lifecycle = \"active\";\n await loadDesiredUrl(session);\n },\n async capturePreview(input) {\n const session = sessions.get(input.nodeId);\n const contents =\n session?.contents && !session.contents.isDestroyed()\n ? session.contents\n : null;\n if (!contents?.capturePage) {\n return null;\n }\n\n const image = await contents.capturePage();\n if (image.isEmpty?.() === true) {\n return null;\n }\n\n return resizeBrowserPreviewImage(image).toDataURL();\n },\n close(input) {\n const session = sessions.get(input.nodeId);\n if (session) {\n detachGuest(session);\n sessions.delete(input.nodeId);\n }\n emit({ nodeId: input.nodeId, type: \"closed\" });\n return Promise.resolve();\n },\n debugDump(input) {\n const session = sessions.get(input.nodeId);\n if (!session) {\n return null;\n }\n const contents =\n session.contents && !session.contents.isDestroyed()\n ? session.contents\n : null;\n return {\n canGoBack: contents ? canGuestGoBack(contents) : false,\n canGoForward: contents ? canGuestGoForward(contents) : false,\n currentUrl: contents ? contents.getURL() : null,\n desiredUrl: session.desiredUrl,\n isAttachedToWindow: Boolean(contents),\n isLoading: contents ? contents.isLoading() : false,\n lifecycle: session.lifecycle,\n nodeId: session.nodeId,\n profileId: session.profileId,\n sessionMode: session.sessionMode,\n sessionPartition: session.sessionPartition,\n title: contents ? contents.getTitle() : null,\n userAgent: contents?.getUserAgent?.() ?? null,\n webContentsDestroyed: session.contents\n ? session.contents.isDestroyed()\n : null,\n webContentsId: session.webContentsId\n };\n },\n goBack(input) {\n const contents = sessions.get(input.nodeId)?.contents;\n if (contents && !contents.isDestroyed() && canGuestGoBack(contents)) {\n goGuestBack(contents);\n }\n return Promise.resolve();\n },\n goForward(input) {\n const contents = sessions.get(input.nodeId)?.contents;\n if (contents && !contents.isDestroyed() && canGuestGoForward(contents)) {\n goGuestForward(contents);\n }\n return Promise.resolve();\n },\n handleGuestOpenUrl(webContentsId, input) {\n const nodeId = nodeIdByWebContentsId.get(webContentsId);\n const session = nodeId ? sessions.get(nodeId) : null;\n if (!session) {\n logger?.warn?.(\"Browser Node ignored guest open-url request\", {\n url: input.url,\n webContentsId\n });\n return;\n }\n logger?.info?.(\"Browser Node handling guest open-url request\", {\n nodeId: session.nodeId,\n url: input.url,\n webContentsId\n });\n emitOpenUrlFromGuest(session, input.url);\n },\n async navigate(input) {\n const resolved = resolveBrowserNavigationUrl(input.url);\n if (!resolved.url) {\n throw new Error(\"Browser Node rejected navigation URL\");\n }\n const session = getSession(input.nodeId, {\n navigationPolicy: input.navigationPolicy,\n url: resolved.url\n });\n session.lifecycle = \"active\";\n await loadDesiredUrl(session);\n },\n async openExternal(input) {\n const resolved = resolveBrowserNavigationUrl(input.url);\n if (!resolved.url) {\n throw new Error(\"Browser Node rejected external URL\");\n }\n await Promise.resolve(openExternal(resolved.url));\n },\n openDevTools(input) {\n const session = sessions.get(input.nodeId);\n const contents = session?.contents ?? null;\n if (contents && !contents.isDestroyed()) {\n try {\n contents.openDevTools?.({ activate: true, mode: \"detach\" });\n } catch (error) {\n logger?.warn?.(\"Browser Node open devtools failed\", {\n error: error instanceof Error ? error.message : String(error),\n nodeId: input.nodeId,\n webContentsId: session?.webContentsId ?? null\n });\n throw error;\n }\n }\n return Promise.resolve();\n },\n async prepareSession(input) {\n await prepareSession?.(input);\n getSession(input.nodeId, {\n navigationPolicy: input.navigationPolicy,\n profileId: input.profileId,\n sessionMode: input.sessionMode,\n sessionPartition: input.sessionPartition\n });\n },\n async registerGuest(input) {\n await prepareSession?.({\n nodeId: input.nodeId,\n profileId: input.profileId,\n sessionMode: input.sessionMode,\n sessionPartition: input.sessionPartition,\n navigationPolicy: input.navigationPolicy\n });\n const contents = resolveWebContents(input.webContentsId);\n if (!contents || contents.isDestroyed()) {\n throw new Error(\n `Browser Node guest ${input.webContentsId} is not available`\n );\n }\n\n const ownerNodeId = nodeIdByWebContentsId.get(input.webContentsId);\n if (ownerNodeId && ownerNodeId !== input.nodeId) {\n throw new Error(\n `Browser Node guest ${input.webContentsId} is already registered`\n );\n }\n\n const session = getSession(input.nodeId, {\n navigationPolicy: input.navigationPolicy,\n profileId: input.profileId,\n sessionMode: input.sessionMode,\n sessionPartition: input.sessionPartition\n });\n if (\n session.webContentsId === input.webContentsId &&\n session.contents === contents\n ) {\n publishState(session);\n return;\n }\n if (session.contents && session.contents !== contents) {\n detachGuest(session);\n }\n session.contents = contents;\n session.webContentsId = input.webContentsId;\n nodeIdByWebContentsId.set(input.webContentsId, input.nodeId);\n session.lifecycle = \"active\";\n logger?.info?.(\"Browser Node registered guest owner\", {\n nodeId: input.nodeId,\n sessionPartition: session.sessionPartition,\n webContentsId: input.webContentsId\n });\n contents.setWindowOpenHandler?.(({ url }) =>\n allowBrowserNodeGuestPopupWindow({\n logger,\n nodeId: input.nodeId,\n url,\n webContentsId: input.webContentsId\n })\n );\n attachGuestListeners(session);\n await applyPreferredColorSchemeToGuest(\n session,\n logger,\n syncPreferredColorScheme,\n preferredColorScheme\n );\n await loadDesiredUrl(session);\n },\n reload(input) {\n const contents = sessions.get(input.nodeId)?.contents;\n if (contents && !contents.isDestroyed()) {\n contents.reload();\n }\n return Promise.resolve();\n },\n unregisterGuest(input) {\n const session = sessions.get(input.nodeId);\n if (!session || session.webContentsId !== input.webContentsId) {\n return Promise.resolve();\n }\n session.lifecycle = \"cold\";\n detachGuest(session);\n return Promise.resolve();\n },\n dispose() {\n unsubscribePreferredColorScheme?.();\n }\n };\n}\n","import type { BrowserWindow } from \"electron\";\nimport { resolveBrowserSessionPartition } from \"../core/session.ts\";\nimport type { BrowserNodeLoopbackPreviewRoutingOptions } from \"./loopbackPreview.ts\";\nimport { createBrowserNodeLoopbackPreviewProxy } from \"./loopbackPreviewProxy.ts\";\nimport type {\n BrowserNodeActivationInput,\n BrowserNodeGuestOpenUrlInput,\n BrowserNodeNavigateInput,\n BrowserNodeNodeIdInput,\n BrowserNodeOpenExternalInput,\n BrowserNodePrepareSessionInput,\n BrowserNodeRegisterGuestInput,\n BrowserNodeShowDevToolsContextMenuInput,\n BrowserNodeUnregisterGuestInput\n} from \"../core/types.ts\";\nimport { createBrowserGuestManager } from \"./guestManager.ts\";\nimport type {\n BrowserGuestManager,\n BrowserPreferredColorScheme,\n BrowserGuestWebContents,\n BrowserNodeElectronLogger\n} from \"./types.ts\";\n\nexport interface BrowserNodeElectronMainChannels {\n readonly activate: string;\n readonly capturePreview?: string;\n readonly close: string;\n readonly debugDump?: string;\n readonly event: string;\n readonly goBack: string;\n readonly goForward: string;\n readonly guestOpenUrl?: string;\n readonly navigate: string;\n readonly openDevTools?: string;\n readonly openExternal?: string;\n readonly prepareSession: string;\n readonly registerGuest: string;\n readonly reload: string;\n readonly showDevToolsContextMenu?: string;\n readonly unregisterGuest: string;\n}\n\nexport interface BrowserNodeElectronDevToolsContextMenuInput {\n readonly label: string;\n readonly openDevTools: () => Promise<void> | void;\n readonly ownerWindow: BrowserWindow;\n readonly point: { x: number; y: number };\n}\n\nexport interface RegisterBrowserNodeElectronMainInput {\n readonly channels: BrowserNodeElectronMainChannels;\n readonly getOwnerWindow: (event: unknown) => BrowserWindow | null;\n readonly getPreferredColorScheme?: () => BrowserPreferredColorScheme;\n readonly logger?: BrowserNodeElectronLogger;\n readonly loopbackPreviewRouting?: BrowserNodeLoopbackPreviewRoutingOptions;\n readonly openExternal: (url: string) => Promise<void> | void;\n readonly registerHandler: <TPayload, TResult>(\n channel: string,\n handler: (event: unknown, payload: TPayload) => Promise<TResult> | TResult\n ) => void;\n readonly registerListener?: <TPayload>(\n channel: string,\n handler: (event: unknown, payload: TPayload) => void\n ) => void;\n readonly resolveWebContents: (input: {\n event: unknown;\n ownerWindow: BrowserWindow;\n webContentsId: number;\n }) => BrowserGuestWebContents | null;\n readonly showDevToolsContextMenu?: (\n input: BrowserNodeElectronDevToolsContextMenuInput\n ) => Promise<void> | void;\n readonly syncPreferredColorScheme?: (\n contents: BrowserGuestWebContents,\n scheme: BrowserPreferredColorScheme\n ) => Promise<void> | void;\n readonly subscribePreferredColorScheme?: (\n listener: (scheme: BrowserPreferredColorScheme) => void\n ) => () => void;\n}\n\nexport function registerBrowserNodeElectronMain(\n input: RegisterBrowserNodeElectronMainInput\n): void {\n const managersByWindow = new WeakMap<BrowserWindow, BrowserGuestManager>();\n const loopbackPreviewProxy =\n input.loopbackPreviewRouting !== undefined\n ? createBrowserNodeLoopbackPreviewProxy({\n logger: input.logger,\n resolveSession: async ({\n profileId,\n sessionMode,\n sessionPartition\n }) => {\n const { session } = await import(\"electron\");\n return session.fromPartition(\n resolveBrowserSessionPartition({\n profileId,\n sessionMode,\n sessionPartition\n })\n );\n },\n routing: input.loopbackPreviewRouting\n })\n : null;\n\n const resolveManagerForWindow = (\n event: unknown,\n ownerWindow: BrowserWindow\n ): BrowserGuestManager => {\n const existing = managersByWindow.get(ownerWindow);\n if (existing) {\n return existing;\n }\n\n const manager = createBrowserGuestManager({\n emit(browserEvent) {\n if (!ownerWindow.isDestroyed()) {\n ownerWindow.webContents.send(input.channels.event, browserEvent);\n }\n },\n getPreferredColorScheme: input.getPreferredColorScheme,\n logger: input.logger,\n openExternal: input.openExternal,\n prepareSession:\n loopbackPreviewProxy !== null\n ? (payload) => loopbackPreviewProxy.configureSession(payload)\n : undefined,\n resolveWebContents: (webContentsId) =>\n input.resolveWebContents({\n event,\n ownerWindow,\n webContentsId\n }),\n syncPreferredColorScheme: input.syncPreferredColorScheme,\n subscribePreferredColorScheme: input.subscribePreferredColorScheme\n });\n ownerWindow.once(\"closed\", () => {\n manager.dispose();\n managersByWindow.delete(ownerWindow);\n });\n managersByWindow.set(ownerWindow, manager);\n return manager;\n };\n\n const resolveOwnedManager = (event: unknown) => {\n const ownerWindow = input.getOwnerWindow(event);\n if (!ownerWindow) {\n throw new Error(\"Browser Node IPC requires an owner window\");\n }\n return {\n manager: resolveManagerForWindow(event, ownerWindow),\n ownerWindow\n };\n };\n\n const showDevToolsContextMenu =\n input.showDevToolsContextMenu ?? showElectronDevToolsContextMenu;\n\n input.registerHandler(input.channels.prepareSession, (event, payload) =>\n resolveOwnedManager(event).manager.prepareSession(\n payload as BrowserNodePrepareSessionInput\n )\n );\n input.registerHandler(input.channels.activate, (event, payload) =>\n resolveOwnedManager(event).manager.activate(\n payload as BrowserNodeActivationInput\n )\n );\n if (input.channels.capturePreview) {\n input.registerHandler(input.channels.capturePreview, (event, payload) =>\n resolveOwnedManager(event).manager.capturePreview(\n payload as BrowserNodeNodeIdInput\n )\n );\n }\n input.registerHandler(input.channels.registerGuest, (event, payload) =>\n resolveOwnedManager(event).manager.registerGuest(\n payload as BrowserNodeRegisterGuestInput\n )\n );\n input.registerHandler(input.channels.unregisterGuest, (event, payload) =>\n resolveOwnedManager(event).manager.unregisterGuest(\n payload as BrowserNodeUnregisterGuestInput\n )\n );\n input.registerHandler(input.channels.navigate, (event, payload) =>\n resolveOwnedManager(event).manager.navigate(\n payload as BrowserNodeNavigateInput\n )\n );\n if (input.channels.openExternal) {\n input.registerHandler(input.channels.openExternal, (event, payload) =>\n resolveOwnedManager(event).manager.openExternal(\n payload as BrowserNodeOpenExternalInput\n )\n );\n }\n if (input.channels.openDevTools) {\n input.registerHandler(input.channels.openDevTools, (event, payload) => {\n const nodePayload = payload as BrowserNodeNodeIdInput;\n return resolveOwnedManager(event).manager.openDevTools(nodePayload);\n });\n }\n if (input.channels.showDevToolsContextMenu) {\n input.registerHandler(\n input.channels.showDevToolsContextMenu,\n (event, payload) => {\n const contextMenuPayload =\n payload as BrowserNodeShowDevToolsContextMenuInput;\n const { manager, ownerWindow } = resolveOwnedManager(event);\n return showDevToolsContextMenu({\n label: contextMenuPayload.label,\n openDevTools: () => {\n return manager.openDevTools({ nodeId: contextMenuPayload.nodeId });\n },\n ownerWindow,\n point: contextMenuPayload.point\n });\n }\n );\n }\n input.registerHandler(input.channels.goBack, (event, payload) =>\n resolveOwnedManager(event).manager.goBack(payload as BrowserNodeNodeIdInput)\n );\n input.registerHandler(input.channels.goForward, (event, payload) =>\n resolveOwnedManager(event).manager.goForward(\n payload as BrowserNodeNodeIdInput\n )\n );\n input.registerHandler(input.channels.reload, (event, payload) =>\n resolveOwnedManager(event).manager.reload(payload as BrowserNodeNodeIdInput)\n );\n input.registerHandler(input.channels.close, (event, payload) =>\n resolveOwnedManager(event).manager.close(payload as BrowserNodeNodeIdInput)\n );\n if (input.channels.debugDump) {\n input.registerHandler(input.channels.debugDump, (event, payload) =>\n resolveOwnedManager(event).manager.debugDump(\n payload as BrowserNodeNodeIdInput\n )\n );\n }\n if (input.channels.guestOpenUrl && input.registerListener) {\n input.registerListener(input.channels.guestOpenUrl, (event, payload) => {\n const senderId = readBrowserNodeIpcSenderId(event);\n const openUrlPayload = payload as BrowserNodeGuestOpenUrlInput | null;\n if (\n typeof senderId !== \"number\" ||\n !Number.isFinite(senderId) ||\n !openUrlPayload ||\n typeof openUrlPayload.url !== \"string\"\n ) {\n return;\n }\n resolveOwnedManager(event).manager.handleGuestOpenUrl(\n senderId,\n openUrlPayload\n );\n });\n }\n}\n\nasync function showElectronDevToolsContextMenu(\n input: BrowserNodeElectronDevToolsContextMenuInput\n): Promise<void> {\n const { Menu } = await import(\"electron\");\n const menu = Menu.buildFromTemplate([\n {\n click: () => {\n void Promise.resolve(input.openDevTools()).catch(() => undefined);\n },\n label: input.label\n }\n ]);\n menu.popup({\n window: input.ownerWindow,\n x: Math.round(input.point.x),\n y: Math.round(input.point.y)\n });\n}\n\nfunction readBrowserNodeIpcSenderId(event: unknown): number | null {\n const sender = (event as { sender?: { id?: unknown } } | null)?.sender;\n return typeof sender?.id === \"number\" ? sender.id : null;\n}\n","import type { WebContents } from \"electron\";\nimport type { BrowserNodeElectronLogger } from \"./types.ts\";\n\nconst electronUserAgentTokenPattern = /\\sElectron\\/[^\\s]+/g;\n\nexport function sanitizeBrowserGuestUserAgent(userAgent: string): string {\n return userAgent\n .trim()\n .replace(electronUserAgentTokenPattern, \"\")\n .replace(/\\s{2,}/g, \" \");\n}\n\nexport function applyBrowserGuestUserAgent(\n contents: WebContents,\n logger?: BrowserNodeElectronLogger\n): void {\n const guestContents = contents as WebContents & {\n getUserAgent?: () => string;\n setUserAgent?: (userAgent: string) => void;\n };\n if (\n typeof guestContents.getUserAgent !== \"function\" ||\n typeof guestContents.setUserAgent !== \"function\"\n ) {\n return;\n }\n\n const currentUserAgent = guestContents.getUserAgent().trim();\n const nextUserAgent = sanitizeBrowserGuestUserAgent(currentUserAgent);\n if (!nextUserAgent || nextUserAgent === currentUserAgent) {\n return;\n }\n\n guestContents.setUserAgent(nextUserAgent);\n logger?.debug?.(\"Browser Node sanitized guest user agent\", {\n webContentsId: contents.id ?? null\n });\n}\n","import type { Event, WebContents, WebPreferences } from \"electron\";\nimport {\n isBrowserSessionPartitionAllowed,\n type BrowserSessionPartitionAllowedOptions\n} from \"../core/session.ts\";\nimport { resolveBrowserNavigationUrl } from \"../core/url.ts\";\nimport { applyBrowserGuestUserAgent } from \"./userAgent.ts\";\nimport type { BrowserNodeElectronLogger } from \"./types.ts\";\n\nexport interface BrowserWebviewSecurityInput {\n allowedSessionPartitions?: BrowserSessionPartitionAllowedOptions;\n params: Record<string, string>;\n resolvePreload?: BrowserWebviewPreloadResolver;\n webPreferences: WebPreferences;\n}\n\nexport interface BrowserWebviewSecurityResult {\n allowed: boolean;\n reason: string | null;\n}\n\nexport type BrowserNodeWebviewMatcher = (\n params: Record<string, string>\n) => boolean;\n\ninterface PendingBrowserWebviewAttach {\n allowNativePopups: boolean;\n}\n\nexport interface BrowserWebviewPreloadResolverInput {\n params: Readonly<Record<string, string>>;\n}\n\nexport type BrowserWebviewPreloadResolver = (\n input: BrowserWebviewPreloadResolverInput\n) => string | null | undefined;\n\nfunction isBrowserNodeInitialWebviewUrl(url: string | undefined): boolean {\n return (url ?? \"\").trim() === \"about:blank\";\n}\n\nexport function isBrowserNodeWebviewAttach(\n params: Record<string, string>,\n allowedSessionPartitions?: BrowserSessionPartitionAllowedOptions\n): boolean {\n return (\n params[\"data-browser-node-webview\"] === \"true\" ||\n isBrowserSessionPartitionAllowed(params.partition, allowedSessionPartitions)\n );\n}\n\nfunction shouldAllowBrowserNodeNativePopups(\n params: Record<string, string>\n): boolean {\n return (\n params[\"data-browser-node-webview\"] === \"true\" ||\n isBrowserSessionPartitionAllowed(params.partition)\n );\n}\n\nfunction getPopupLogMetadata(url: string): Record<string, unknown> {\n try {\n const parsed = new URL(url);\n return {\n popupOrigin: parsed.origin,\n popupPath: parsed.pathname,\n popupProtocol: parsed.protocol\n };\n } catch {\n return {\n popupOrigin: null,\n popupPath: null,\n popupProtocol: null\n };\n }\n}\n\nfunction allowBrowserNodeNativePopupWindow({\n guestWebContentsId,\n logger,\n url\n}: {\n guestWebContentsId: number | null;\n logger?: BrowserNodeElectronLogger;\n url: string;\n}) {\n logger?.info?.(\"Browser Node native popup allowed\", {\n guestWebContentsId,\n ...getPopupLogMetadata(url)\n });\n return {\n action: \"allow\" as const,\n overrideBrowserWindowOptions: {\n height: 720,\n show: true,\n width: 520\n }\n };\n}\n\nexport function enforceBrowserWebviewSecurity({\n allowedSessionPartitions,\n params,\n resolvePreload,\n webPreferences\n}: BrowserWebviewSecurityInput): BrowserWebviewSecurityResult {\n webPreferences.allowRunningInsecureContent = false;\n webPreferences.contextIsolation = true;\n webPreferences.javascript = true;\n webPreferences.nodeIntegration = false;\n webPreferences.plugins = false;\n webPreferences.sandbox = true;\n webPreferences.webSecurity = true;\n delete webPreferences.preload;\n\n const partition = params.partition;\n if (\n !partition ||\n !isBrowserSessionPartitionAllowed(partition, allowedSessionPartitions)\n ) {\n return {\n allowed: false,\n reason: \"Unsupported Browser Node session partition\"\n };\n }\n\n if (isBrowserNodeInitialWebviewUrl(params.src)) {\n params.src = \"about:blank\";\n } else {\n const resolved = resolveBrowserNavigationUrl(params.src ?? \"about:blank\");\n if (!resolved.url) {\n return {\n allowed: false,\n reason: \"Unsupported browser URL\"\n };\n }\n params.src = resolved.url;\n }\n\n const preload = resolvePreload?.({ params: { ...params } });\n const resolvedPreload = typeof preload === \"string\" ? preload.trim() : \"\";\n if (resolvedPreload.length > 0) {\n webPreferences.preload = resolvedPreload;\n }\n\n return { allowed: true, reason: null };\n}\n\nexport interface InstallBrowserWebviewSecurityInput {\n allowedSessionPartitions?: BrowserSessionPartitionAllowedOptions;\n contents: WebContents;\n logger?: BrowserNodeElectronLogger;\n onGuestAttached?: (guestContents: WebContents) => void;\n openExternal: (url: string) => Promise<void> | void;\n resolvePreload?: BrowserWebviewPreloadResolver;\n shouldHandleWebview?: BrowserNodeWebviewMatcher;\n}\n\nexport function installBrowserWebviewSecurity({\n allowedSessionPartitions,\n contents,\n logger,\n onGuestAttached,\n openExternal,\n resolvePreload,\n shouldHandleWebview\n}: InstallBrowserWebviewSecurityInput): () => void {\n const pendingBrowserAttaches: PendingBrowserWebviewAttach[] = [];\n\n const handleWillAttachWebview = (\n event: Event,\n webPreferences: WebPreferences,\n params: Record<string, string>\n ) => {\n const shouldHandle =\n shouldHandleWebview?.(params) ??\n isBrowserNodeWebviewAttach(params, allowedSessionPartitions);\n logger?.debug?.(\"Browser Node webview will attach\", {\n partition: params.partition ?? null,\n shouldHandle,\n src: params.src ?? null\n });\n if (!shouldHandle) {\n return;\n }\n\n const allowNativePopups = shouldAllowBrowserNodeNativePopups(params);\n if (allowNativePopups) {\n params.allowpopups = \"true\";\n }\n logger?.info?.(\"Browser Node webview popup policy applied\", {\n allowNativePopups,\n allowpopups: params.allowpopups ?? null,\n partition: params.partition ?? null,\n src: params.src ?? null\n });\n\n const result = enforceBrowserWebviewSecurity({\n allowedSessionPartitions,\n params,\n resolvePreload,\n webPreferences\n });\n if (!result.allowed) {\n logger?.warn?.(\"Browser Node webview blocked\", { reason: result.reason });\n event.preventDefault();\n return;\n }\n pendingBrowserAttaches.push({\n allowNativePopups\n });\n logger?.debug?.(\"Browser Node webview attach allowed\", {\n partition: params.partition ?? null,\n src: params.src ?? null\n });\n };\n\n const handleDidAttachWebview = (\n _event: Event,\n guestContents: WebContents\n ) => {\n const pendingAttach = pendingBrowserAttaches.shift();\n if (!pendingAttach) {\n logger?.debug?.(\"Browser Node webview did attach ignored\", {\n guestWebContentsId: guestContents.id ?? null,\n pendingBrowserAttachCount: pendingBrowserAttaches.length\n });\n return;\n }\n\n applyBrowserGuestUserAgent(guestContents, logger);\n if (pendingAttach.allowNativePopups) {\n guestContents.setWindowOpenHandler(({ url }) =>\n allowBrowserNodeNativePopupWindow({\n guestWebContentsId: guestContents.id ?? null,\n logger,\n url\n })\n );\n } else {\n guestContents.setWindowOpenHandler(({ url }) => {\n logger?.info?.(\"Browser Node webview popup externalized\", {\n guestWebContentsId: guestContents.id ?? null,\n ...getPopupLogMetadata(url)\n });\n const resolved = resolveBrowserNavigationUrl(url);\n if (resolved.url) {\n void Promise.resolve(openExternal(resolved.url)).catch(\n () => undefined\n );\n }\n return { action: \"deny\" };\n });\n }\n onGuestAttached?.(guestContents);\n logger?.debug?.(\"Browser Node webview guest attached\", {\n guestWebContentsId: guestContents.id ?? null,\n pendingBrowserAttachCount: pendingBrowserAttaches.length\n });\n };\n\n contents.on(\"will-attach-webview\", handleWillAttachWebview);\n contents.on(\"did-attach-webview\", handleDidAttachWebview);\n\n return () => {\n contents.off(\"will-attach-webview\", handleWillAttachWebview);\n contents.off(\"did-attach-webview\", handleDidAttachWebview);\n };\n}\n"],"mappings":"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","names":["getPopupLogMetadata"]}
|
|
1
|
+
{"version":3,"sources":["../../src/electron-main/loopbackPreviewProxy.ts","../../src/electron-main/guestManager.ts","../../src/electron-main/registerElectronMain.ts","../../src/electron-main/userAgent.ts","../../src/electron-main/webviewSecurity.ts"],"sourcesContent":["import {\n createServer,\n request as httpRequest,\n type IncomingHttpHeaders,\n type IncomingMessage,\n type Server,\n type ServerResponse\n} from \"node:http\";\nimport { request as httpsRequest } from \"node:https\";\nimport type { Socket } from \"node:net\";\nimport { pipeline } from \"node:stream\";\nimport WebSocket, { WebSocketServer } from \"ws\";\nimport type { RawData } from \"ws\";\nimport type { BrowserNodeSessionMode } from \"../core/types.ts\";\nimport type {\n BrowserNodeLoopbackPreviewRoutingOptions,\n BrowserNodeLoopbackPreviewTarget\n} from \"./loopbackPreview.ts\";\nimport type { BrowserNodeElectronLogger } from \"./types.ts\";\n\nconst defaultLoopbackPreviewCacheTtlMs = 30_000;\nconst loopbackHostPattern = /^(localhost|127(?:\\.\\d{1,3}){0,3})$/i;\n// Order matters: bypass all, subtract loopback so local previews can proxy,\n// then keep HTTPS/WSS direct because this proxy does not implement CONNECT.\nconst loopbackPreviewProxyBypassRules = \"*;<-loopback>;https://*;wss://*\";\nconst hopByHopHeaders = new Set([\n \"connection\",\n \"keep-alive\",\n \"proxy-authenticate\",\n \"proxy-authorization\",\n \"proxy-connection\",\n \"te\",\n \"trailer\",\n \"transfer-encoding\",\n \"upgrade\"\n]);\n\ninterface CachedLoopbackPreviewTarget {\n expiresAt: number;\n target: BrowserNodeLoopbackPreviewTarget | null;\n}\n\nexport interface BrowserNodePreviewSession {\n setProxy(input: {\n mode: \"fixed_servers\";\n proxyBypassRules: string;\n proxyRules: string;\n }): Promise<void>;\n}\n\nexport interface BrowserNodeLoopbackPreviewProxy {\n configureSession(input: {\n profileId: string | null;\n sessionMode: BrowserNodeSessionMode;\n sessionPartition?: string | null;\n }): Promise<void>;\n dispose(): Promise<void>;\n addressForTesting(): string;\n serverForTesting(): Server;\n}\n\nexport interface BrowserNodeLoopbackPreviewProxyInput {\n logger?: BrowserNodeElectronLogger;\n resolveSession: (input: {\n profileId: string | null;\n sessionMode: BrowserNodeSessionMode;\n sessionPartition?: string | null;\n }) => BrowserNodePreviewSession | Promise<BrowserNodePreviewSession>;\n routing: BrowserNodeLoopbackPreviewRoutingOptions;\n}\n\nexport function createBrowserNodeLoopbackPreviewProxy({\n logger,\n resolveSession,\n routing\n}: BrowserNodeLoopbackPreviewProxyInput): BrowserNodeLoopbackPreviewProxy {\n const configuredSessions = new WeakSet<BrowserNodePreviewSession>();\n const targetCache = new Map<string, CachedLoopbackPreviewTarget>();\n const downstreamWebSocketServer = new WebSocketServer({ noServer: true });\n let server: Server | null = null;\n let serverStartPromise: Promise<string> | null = null;\n\n const cacheTtlMs = routing.cacheTtlMs ?? defaultLoopbackPreviewCacheTtlMs;\n const fallback = routing.fallback ?? \"direct\";\n\n const pruneExpiredTargets = (now: number): void => {\n for (const [cacheKey, cachedTarget] of targetCache.entries()) {\n if (cachedTarget.expiresAt <= now) {\n targetCache.delete(cacheKey);\n }\n }\n };\n\n const start = async (): Promise<string> => {\n if (server?.listening) {\n return addressForTesting();\n }\n if (serverStartPromise) {\n return serverStartPromise;\n }\n\n serverStartPromise = new Promise<string>((resolve, reject) => {\n const nextServer = createServer((request, response) => {\n void handleRequest(request, response);\n });\n nextServer.on(\"upgrade\", (request, socket, head) => {\n void handleUpgrade(request, socket as Socket, head);\n });\n\n const cleanup = (): void => {\n nextServer.removeListener(\"error\", onError);\n nextServer.removeListener(\"listening\", onListening);\n };\n const onError = (error: Error): void => {\n cleanup();\n reject(error);\n };\n const onListening = (): void => {\n cleanup();\n server = nextServer;\n resolve(addressForTesting());\n };\n\n nextServer.once(\"error\", onError);\n nextServer.once(\"listening\", onListening);\n nextServer.listen(0, \"127.0.0.1\");\n }).finally(() => {\n serverStartPromise = null;\n });\n\n return serverStartPromise;\n };\n\n const resolveLoopbackTarget = async (\n originalUrl: URL\n ): Promise<BrowserNodeLoopbackPreviewTarget | null> => {\n const port = Number(originalUrl.port);\n if (!Number.isInteger(port) || port < 1 || port > 65535) {\n return null;\n }\n\n const now = Date.now();\n pruneExpiredTargets(now);\n\n const cacheKey = originalUrl.toString();\n const cached = targetCache.get(cacheKey);\n if (cached && cached.expiresAt > now) {\n return cached.target;\n }\n\n const nextTarget = await Promise.resolve(\n routing.resolver.resolveTarget({\n port,\n url: originalUrl.toString()\n })\n );\n\n const target = normalizeLoopbackPreviewTarget(nextTarget);\n targetCache.set(cacheKey, {\n expiresAt: now + Math.max(0, cacheTtlMs),\n target\n });\n return target;\n };\n\n const configureSession = async (input: {\n profileId: string | null;\n sessionMode: BrowserNodeSessionMode;\n sessionPartition?: string | null;\n }): Promise<void> => {\n const nextSession = await resolveSession(input);\n if (configuredSessions.has(nextSession)) {\n return;\n }\n const address = await start();\n await nextSession.setProxy({\n mode: \"fixed_servers\",\n proxyBypassRules: loopbackPreviewProxyBypassRules,\n proxyRules: `http=${address}`\n });\n configuredSessions.add(nextSession);\n };\n\n const dispose = async (): Promise<void> => {\n targetCache.clear();\n downstreamWebSocketServer.close();\n const activeServer = server;\n server = null;\n if (!activeServer || !activeServer.listening) {\n return;\n }\n await new Promise<void>((resolve) => {\n activeServer.close(() => resolve());\n });\n };\n\n const handleRequest = async (\n request: IncomingMessage,\n response: ServerResponse\n ): Promise<void> => {\n const originalUrl = resolveProxyRequestUrl(request);\n if (!originalUrl) {\n response.writeHead(400);\n response.end(\"invalid proxy request\");\n return;\n }\n if (originalUrl.protocol !== \"http:\") {\n response.writeHead(400);\n response.end(\"unsupported proxy request\");\n return;\n }\n\n const targetContext = await resolveTargetRequestContext(originalUrl);\n if (!targetContext.targetUrl) {\n response.writeHead(502);\n response.end(\"unable to resolve loopback preview target\");\n return;\n }\n\n const headers = filterProxyHeaders(request.headers);\n headers.set(\"host\", targetContext.targetUrl.host);\n\n const forward =\n targetContext.targetUrl.protocol === \"https:\"\n ? httpsRequest\n : httpRequest;\n const upstream = forward(\n targetContext.targetUrl,\n {\n headers: Object.fromEntries(headers.entries()),\n method: request.method\n },\n (upstreamResponse) => {\n const responseHeaders = filterProxyHeaders(upstreamResponse.headers);\n const location = responseHeaders.get(\"location\");\n if (location && targetContext.loopbackTarget) {\n const rewritten = rewriteLoopbackLocation({\n location,\n originalUrl,\n targetUrl: targetContext.loopbackTarget.targetUrl\n });\n if (rewritten) {\n responseHeaders.set(\"location\", rewritten);\n }\n }\n\n response.writeHead(\n upstreamResponse.statusCode ?? 502,\n upstreamResponse.statusMessage,\n Object.fromEntries(responseHeaders.entries())\n );\n pipeline(upstreamResponse, response, () => undefined);\n }\n );\n\n upstream.on(\"error\", (error) => {\n logger?.warn?.(\"Browser Node loopback preview request failed\", {\n error: normalizeProxyError(error),\n originalUrl: originalUrl.toString(),\n targetUrl: targetContext.targetUrl?.toString() ?? null,\n workspaceId: targetContext.loopbackTarget?.workspaceId ?? null\n });\n if (!response.headersSent) {\n response.writeHead(502);\n }\n response.end(\n `loopback preview upstream request failed: ${normalizeProxyError(error)}`\n );\n });\n\n pipeline(request, upstream, () => undefined);\n };\n\n const handleUpgrade = async (\n request: IncomingMessage,\n socket: Socket,\n head: Buffer\n ): Promise<void> => {\n const originalUrl = resolveProxyRequestUrl(request, \"ws:\");\n if (!originalUrl) {\n socket.destroy();\n return;\n }\n if (originalUrl.protocol !== \"ws:\") {\n socket.destroy();\n return;\n }\n\n const targetContext = await resolveTargetRequestContext(originalUrl);\n const targetUrl = targetContext.targetUrl;\n if (!targetUrl) {\n socket.destroy();\n return;\n }\n\n downstreamWebSocketServer.handleUpgrade(\n request,\n socket,\n head,\n (downstream) => {\n const headers = Object.fromEntries(\n filterProxyHeaders(request.headers).entries()\n );\n headers.host = targetUrl.host;\n\n const upstream = new WebSocket(targetUrl, { headers });\n const pendingMessages: Array<{ data: RawData; isBinary: boolean }> = [];\n\n downstream.on(\"message\", (data, isBinary) => {\n if (upstream.readyState === WebSocket.OPEN) {\n upstream.send(data, { binary: isBinary });\n return;\n }\n pendingMessages.push({ data, isBinary });\n });\n\n upstream.once(\"open\", () => {\n for (const nextMessage of pendingMessages.splice(0)) {\n upstream.send(nextMessage.data, { binary: nextMessage.isBinary });\n }\n });\n\n upstream.on(\"message\", (data, isBinary) => {\n if (downstream.readyState === WebSocket.OPEN) {\n downstream.send(data, { binary: isBinary });\n }\n });\n\n upstream.once(\"close\", (code, reason) => {\n if (\n downstream.readyState === WebSocket.OPEN ||\n downstream.readyState === WebSocket.CONNECTING\n ) {\n downstream.close(normalizeWebSocketCloseCode(code), reason);\n }\n });\n\n downstream.once(\"close\", (code, reason) => {\n if (\n upstream.readyState === WebSocket.OPEN ||\n upstream.readyState === WebSocket.CONNECTING\n ) {\n upstream.close(normalizeWebSocketCloseCode(code), reason);\n }\n });\n\n upstream.once(\"error\", (error) => {\n logger?.warn?.(\"Browser Node loopback preview websocket failed\", {\n error: normalizeProxyError(error),\n originalUrl: originalUrl.toString(),\n targetUrl: targetContext.targetUrl?.toString() ?? null,\n workspaceId: targetContext.loopbackTarget?.workspaceId ?? null\n });\n downstream.close(1011, \"loopback preview websocket upstream failed\");\n });\n }\n );\n };\n\n const resolveTargetRequestContext = async (\n originalUrl: URL\n ): Promise<{\n loopbackTarget: BrowserNodeLoopbackPreviewTarget | null;\n targetUrl: URL | null;\n }> => {\n if (!isLoopbackUrl(originalUrl)) {\n return {\n loopbackTarget: null,\n targetUrl: null\n };\n }\n\n const loopbackTarget = await resolveLoopbackTarget(originalUrl);\n if (loopbackTarget) {\n return {\n loopbackTarget,\n targetUrl: buildTargetRequestUrl(loopbackTarget.targetUrl, originalUrl)\n };\n }\n\n if (fallback === \"direct\") {\n return {\n loopbackTarget: null,\n targetUrl: cloneUrl(originalUrl)\n };\n }\n\n return {\n loopbackTarget: null,\n targetUrl: null\n };\n };\n\n const addressForTesting = (): string => {\n const address = server?.address();\n if (!address || typeof address === \"string\") {\n throw new Error(\"Browser Node loopback preview proxy is not listening\");\n }\n return `127.0.0.1:${address.port}`;\n };\n\n const serverForTesting = (): Server => {\n if (!server) {\n throw new Error(\"Browser Node loopback preview proxy is not started\");\n }\n return server;\n };\n\n return {\n addressForTesting,\n configureSession,\n dispose,\n serverForTesting\n };\n}\n\nfunction normalizeLoopbackPreviewTarget(\n input: BrowserNodeLoopbackPreviewTarget | null | undefined\n): BrowserNodeLoopbackPreviewTarget | null {\n if (!input) {\n return null;\n }\n const normalizedTargetUrl = input.targetUrl.trim();\n if (normalizedTargetUrl.length === 0) {\n return null;\n }\n try {\n const parsed = new URL(normalizedTargetUrl);\n if (parsed.protocol !== \"http:\" && parsed.protocol !== \"https:\") {\n return null;\n }\n return {\n targetUrl: parsed.toString(),\n workspaceId: input.workspaceId\n };\n } catch {\n return null;\n }\n}\n\nfunction resolveProxyRequestUrl(\n request: IncomingMessage,\n defaultProtocol = \"http:\"\n): URL | null {\n const rawUrl = request.url ?? \"\";\n try {\n if (/^[a-z][a-z\\d+\\-.]*:\\/\\//i.test(rawUrl)) {\n return new URL(rawUrl);\n }\n const host = request.headers.host;\n if (typeof host !== \"string\" || host.trim().length === 0) {\n return null;\n }\n const path = rawUrl.startsWith(\"/\") ? rawUrl : `/${rawUrl}`;\n return new URL(`${defaultProtocol}//${host}${path}`);\n } catch {\n return null;\n }\n}\n\nfunction filterProxyHeaders(input: Headers | IncomingHttpHeaders): Headers {\n const output = new Headers();\n const appendValue = (key: string, value: string): void => {\n if (hopByHopHeaders.has(key.toLowerCase())) {\n return;\n }\n output.append(key, value);\n };\n\n if (input instanceof Headers) {\n input.forEach((value, key) => {\n appendValue(key, value);\n });\n return output;\n }\n\n for (const [key, value] of Object.entries(input)) {\n if (Array.isArray(value)) {\n for (const nextValue of value) {\n appendValue(key, nextValue);\n }\n continue;\n }\n if (typeof value === \"string\") {\n appendValue(key, value);\n }\n }\n return output;\n}\n\nfunction isLoopbackUrl(url: URL): boolean {\n return (\n (url.protocol === \"http:\" || url.protocol === \"ws:\") &&\n loopbackHostPattern.test(url.hostname) &&\n url.port.length > 0\n );\n}\n\nfunction buildTargetRequestUrl(\n targetUrl: string,\n originalUrl: URL\n): URL | null {\n try {\n const base = new URL(targetUrl);\n const normalizedBasePath = normalizeBasePath(base.pathname);\n const nextUrl = new URL(base.toString());\n nextUrl.pathname = joinBasePath(normalizedBasePath, originalUrl.pathname);\n nextUrl.search = originalUrl.search;\n nextUrl.hash = \"\";\n if (originalUrl.protocol === \"ws:\") {\n nextUrl.protocol = nextUrl.protocol === \"https:\" ? \"wss:\" : \"ws:\";\n }\n return nextUrl;\n } catch {\n return null;\n }\n}\n\nfunction rewriteLoopbackLocation({\n location,\n originalUrl,\n targetUrl\n}: {\n location: string;\n originalUrl: URL;\n targetUrl: string;\n}): string | null {\n try {\n const targetBase = new URL(targetUrl);\n const resolvedLocation = new URL(location, targetBase);\n if (resolvedLocation.origin !== targetBase.origin) {\n return location;\n }\n\n const basePath = normalizeBasePath(targetBase.pathname);\n const strippedPath =\n stripBasePath(basePath, resolvedLocation.pathname) ??\n resolvedLocation.pathname;\n\n const loopbackUrl = new URL(originalUrl.origin);\n resolvedLocation.protocol = loopbackUrl.protocol;\n resolvedLocation.host = loopbackUrl.host;\n resolvedLocation.pathname = strippedPath;\n resolvedLocation.username = \"\";\n resolvedLocation.password = \"\";\n return resolvedLocation.toString();\n } catch {\n return location;\n }\n}\n\nfunction normalizeBasePath(pathname: string): string {\n const normalized = pathname.startsWith(\"/\") ? pathname : `/${pathname}`;\n return normalized.endsWith(\"/\") ? normalized : `${normalized}/`;\n}\n\nfunction joinBasePath(basePath: string, requestPath: string): string {\n const normalizedRequestPath =\n requestPath === \"/\" ? \"\" : requestPath.replace(/^\\/+/, \"\");\n return normalizedRequestPath.length > 0\n ? `${basePath}${normalizedRequestPath}`.replace(/\\/{2,}/g, \"/\")\n : basePath;\n}\n\nfunction stripBasePath(basePath: string, pathname: string): string | null {\n const normalizedPath = pathname.startsWith(\"/\") ? pathname : `/${pathname}`;\n if (normalizedPath === basePath.slice(0, -1)) {\n return \"/\";\n }\n if (!normalizedPath.startsWith(basePath)) {\n return null;\n }\n const suffix = normalizedPath.slice(basePath.length);\n return suffix.length > 0 ? `/${suffix}` : \"/\";\n}\n\nfunction cloneUrl(url: URL): URL | null {\n try {\n return new URL(url.toString());\n } catch {\n return null;\n }\n}\n\nfunction normalizeProxyError(error: unknown): string {\n if (error instanceof Error) {\n const code =\n typeof (error as NodeJS.ErrnoException).code === \"string\"\n ? ` ${(error as NodeJS.ErrnoException).code}`\n : \"\";\n return `${error.message}${code}`;\n }\n return String(error);\n}\n\nfunction normalizeWebSocketCloseCode(code: number): number {\n if (code === 1000 || (code >= 3000 && code <= 4999)) {\n return code;\n }\n return 1000;\n}\n","import type {\n BrowserNodeLifecycle,\n BrowserNodeNavigationPolicy,\n BrowserNodeRuntimeError,\n BrowserNodeSessionMode\n} from \"../core/types.ts\";\nimport {\n normalizeHostBrowserComparableUrl,\n resolveBrowserNavigationUrl,\n resolveHostBrowserNavigationUrl,\n type BrowserNavigationUrlResolution\n} from \"../core/url.ts\";\nimport type {\n BrowserGuestManager,\n BrowserGuestManagerInput,\n BrowserGuestNativeImage,\n BrowserPreferredColorScheme,\n BrowserGuestWebContents\n} from \"./types.ts\";\n\nconst browserPreviewMaxWidth = 260;\nconst browserPreviewMaxHeight = 170;\nconst abortedNavigationErrorCode = -3;\n\ninterface BrowserGuestSession {\n appliedColorScheme: BrowserPreferredColorScheme | null;\n contents: BrowserGuestWebContents | null;\n desiredUrl: string;\n lifecycle: BrowserNodeLifecycle;\n listeners: Array<{\n event: string;\n listener: (...args: unknown[]) => void;\n }>;\n navigationFailureSequence: number;\n navigationPolicy: BrowserNodeNavigationPolicy | null;\n nodeId: string;\n profileId: string | null;\n sessionMode: BrowserNodeSessionMode;\n sessionPartition: string | null;\n webContentsId: number | null;\n}\n\nfunction canGuestGoBack(contents: BrowserGuestWebContents): boolean {\n return contents.navigationHistory?.canGoBack() ?? contents.canGoBack();\n}\n\nfunction canGuestGoForward(contents: BrowserGuestWebContents): boolean {\n return contents.navigationHistory?.canGoForward() ?? contents.canGoForward();\n}\n\nfunction goGuestBack(contents: BrowserGuestWebContents): void {\n if (contents.navigationHistory) {\n contents.navigationHistory.goBack();\n return;\n }\n contents.goBack();\n}\n\nfunction goGuestForward(contents: BrowserGuestWebContents): void {\n if (contents.navigationHistory) {\n contents.navigationHistory.goForward();\n return;\n }\n contents.goForward();\n}\n\nfunction resolveBrowserNodeUrlError(\n resolved: BrowserNavigationUrlResolution\n): BrowserNodeRuntimeError {\n if (resolved.errorCode === \"invalid-url\") {\n return { code: \"invalid-url\" };\n }\n\n if (resolved.errorCode === \"unsupported-protocol\") {\n return {\n code: \"unsupported-protocol\",\n params: resolved.errorParams\n };\n }\n\n return { code: \"unsupported-url\" };\n}\n\nfunction resizeBrowserPreviewImage(\n image: BrowserGuestNativeImage\n): BrowserGuestNativeImage {\n if (image.isEmpty?.() === true || !image.resize || !image.getSize) {\n return image;\n }\n\n const size = image.getSize();\n if (size.width <= 0 || size.height <= 0) {\n return image;\n }\n\n const scale = Math.min(\n 1,\n browserPreviewMaxWidth / size.width,\n browserPreviewMaxHeight / size.height\n );\n if (scale >= 1) {\n return image;\n }\n\n return image.resize({\n height: Math.max(1, Math.round(size.height * scale)),\n quality: \"good\",\n width: Math.max(1, Math.round(size.width * scale))\n });\n}\n\nfunction isAbortedNavigationError(input: {\n errorCode?: number;\n errorDescription?: string;\n}): boolean {\n return (\n input.errorCode === abortedNavigationErrorCode ||\n input.errorDescription === \"ERR_ABORTED\"\n );\n}\n\nfunction isHttpErrorStatusCode(statusCode: number | undefined): boolean {\n return statusCode !== undefined && statusCode >= 400;\n}\n\nfunction emitBrowserNavigationFailed(input: {\n emit: BrowserGuestManagerInput[\"emit\"];\n errorCode?: number;\n errorDescription?: string;\n nodeId: string;\n}): void {\n input.emit({\n code: \"navigation-failed\",\n diagnosticMessage: input.errorDescription,\n nodeId: input.nodeId,\n params:\n input.errorCode === undefined\n ? undefined\n : { errorCode: input.errorCode },\n type: \"error\"\n });\n}\n\nfunction isGoogleGisOAuthPopupUrl(url: string): boolean {\n try {\n const parsed = new URL(url);\n if (\n parsed.protocol !== \"https:\" ||\n parsed.hostname !== \"accounts.google.com\" ||\n (parsed.pathname !== \"/o/oauth2/v2/auth\" &&\n parsed.pathname !== \"/o/oauth2/auth\")\n ) {\n return false;\n }\n\n const isGisSdkPopup =\n parsed.searchParams.get(\"gsiwebsdk\") === \"gis_attributes\";\n const isPopupResponse =\n parsed.searchParams.get(\"display\") === \"popup\" &&\n (parsed.searchParams.get(\"response_mode\") === \"form_post\" ||\n parsed.searchParams.get(\"redirect_uri\") === \"gis_transform\");\n\n return isGisSdkPopup || isPopupResponse;\n } catch {\n return false;\n }\n}\n\nfunction resolveBrowserNavigationOrigin(url: string): string | null {\n const resolved = resolveBrowserNavigationUrl(url);\n if (!resolved.url) {\n return null;\n }\n\n try {\n return new URL(resolved.url).origin;\n } catch {\n return null;\n }\n}\n\nfunction isBrowserNavigationAllowedByPolicy(input: {\n policy: BrowserNodeNavigationPolicy | null;\n url: string;\n}): boolean {\n if (!input.policy) {\n return true;\n }\n\n if (input.policy.mode === \"same-origin\") {\n const policyOrigin = resolveBrowserNavigationOrigin(input.policy.originUrl);\n const nextOrigin = resolveBrowserNavigationOrigin(input.url);\n return (\n policyOrigin !== null &&\n nextOrigin !== null &&\n policyOrigin === nextOrigin\n );\n }\n\n return true;\n}\n\nasync function applyPreferredColorSchemeToGuest(\n session: BrowserGuestSession,\n logger: BrowserGuestManagerInput[\"logger\"],\n syncPreferredColorScheme: BrowserGuestManagerInput[\"syncPreferredColorScheme\"],\n scheme: BrowserPreferredColorScheme | null\n): Promise<void> {\n const contents = session.contents;\n if (\n !contents ||\n contents.isDestroyed() ||\n !syncPreferredColorScheme ||\n scheme === null ||\n session.appliedColorScheme === scheme\n ) {\n return;\n }\n\n try {\n await syncPreferredColorScheme(contents, scheme);\n session.appliedColorScheme = scheme;\n } catch (error) {\n session.appliedColorScheme = null;\n logger?.warn?.(\"Browser Node failed to sync guest color scheme\", {\n error: error instanceof Error ? error.message : String(error),\n nodeId: session.nodeId,\n scheme,\n webContentsId: session.webContentsId\n });\n }\n}\n\nexport function createBrowserGuestManager({\n emit,\n getPreferredColorScheme,\n logger,\n openExternal,\n prepareSession,\n resolveWebContents,\n syncPreferredColorScheme,\n subscribePreferredColorScheme\n}: BrowserGuestManagerInput): BrowserGuestManager {\n const sessions = new Map<string, BrowserGuestSession>();\n const nodeIdByWebContentsId = new Map<number, string>();\n let preferredColorScheme = getPreferredColorScheme?.() ?? null;\n\n const getSession = (\n nodeId: string,\n input?: {\n navigationPolicy?: BrowserNodeNavigationPolicy | null;\n profileId?: string | null;\n sessionMode?: BrowserNodeSessionMode;\n sessionPartition?: string | null;\n url?: string;\n }\n ): BrowserGuestSession => {\n const existing = sessions.get(nodeId);\n if (existing) {\n if (input?.profileId !== undefined) {\n existing.profileId = input.profileId;\n }\n if (input?.sessionMode !== undefined) {\n existing.sessionMode = input.sessionMode;\n }\n if (input?.sessionPartition !== undefined) {\n existing.sessionPartition = input.sessionPartition;\n }\n if (input?.navigationPolicy !== undefined) {\n existing.navigationPolicy = input.navigationPolicy;\n }\n if (input?.url !== undefined) {\n existing.desiredUrl = input.url;\n }\n return existing;\n }\n\n const session: BrowserGuestSession = {\n appliedColorScheme: null,\n contents: null,\n desiredUrl: input?.url ?? \"about:blank\",\n lifecycle: \"cold\",\n listeners: [],\n navigationFailureSequence: 0,\n navigationPolicy: input?.navigationPolicy ?? null,\n nodeId,\n profileId: input?.profileId ?? null,\n sessionMode: input?.sessionMode ?? \"shared\",\n sessionPartition: input?.sessionPartition ?? null,\n webContentsId: null\n };\n sessions.set(nodeId, session);\n return session;\n };\n\n const publishState = (session: BrowserGuestSession): void => {\n const contents =\n session.contents && !session.contents.isDestroyed()\n ? session.contents\n : null;\n emit({\n canGoBack: contents ? canGuestGoBack(contents) : false,\n canGoForward: contents ? canGuestGoForward(contents) : false,\n isAttachedToWindow: Boolean(contents),\n isLoading: contents ? contents.isLoading() : false,\n isOccluded: session.lifecycle === \"cold\",\n lifecycle: session.lifecycle,\n nodeId: session.nodeId,\n title: contents ? contents.getTitle() || null : null,\n type: \"state\",\n url: contents\n ? contents.getURL() || session.desiredUrl\n : session.desiredUrl\n });\n };\n\n const detachGuest = (session: BrowserGuestSession): void => {\n const contents = session.contents;\n const webContentsId = session.webContentsId;\n if (contents) {\n for (const record of session.listeners) {\n contents.off(record.event, record.listener);\n }\n }\n session.listeners = [];\n session.contents = null;\n session.appliedColorScheme = null;\n session.webContentsId = null;\n if (\n webContentsId !== null &&\n nodeIdByWebContentsId.get(webContentsId) === session.nodeId\n ) {\n nodeIdByWebContentsId.delete(webContentsId);\n }\n publishState(session);\n };\n\n const handlePreferredColorSchemeChange = (\n scheme: BrowserPreferredColorScheme\n ) => {\n preferredColorScheme = scheme;\n for (const session of sessions.values()) {\n session.appliedColorScheme = null;\n void applyPreferredColorSchemeToGuest(\n session,\n logger,\n syncPreferredColorScheme,\n scheme\n ).catch(() => undefined);\n }\n };\n\n const unsubscribePreferredColorScheme =\n subscribePreferredColorScheme?.(handlePreferredColorSchemeChange) ?? null;\n\n const attachGuestListeners = (session: BrowserGuestSession): void => {\n const contents = session.contents;\n if (!contents) {\n return;\n }\n\n const onStateChange = () => publishState(session);\n const onDidNavigate = (...args: unknown[]) => {\n const url = typeof args[1] === \"string\" ? args[1] : undefined;\n const statusCode = typeof args[2] === \"number\" ? args[2] : undefined;\n const statusText = typeof args[3] === \"string\" ? args[3] : undefined;\n publishState(session);\n if (!isHttpErrorStatusCode(statusCode)) {\n return;\n }\n\n logger?.warn?.(\"Browser Node guest navigation returned HTTP error\", {\n currentUrl: contents.getURL(),\n desiredUrl: session.desiredUrl,\n nodeId: session.nodeId,\n statusCode,\n statusText,\n url,\n webContentsId: session.webContentsId\n });\n emit({\n code: \"navigation-failed\",\n diagnosticMessage: statusText,\n nodeId: session.nodeId,\n params: {\n statusCode,\n ...(statusText ? { statusText } : {})\n },\n type: \"error\"\n });\n };\n const onFailLoad = (...args: unknown[]) => {\n const errorCode = typeof args[1] === \"number\" ? args[1] : undefined;\n const errorDescription =\n typeof args[2] === \"string\" ? args[2] : undefined;\n const validatedUrl = typeof args[3] === \"string\" ? args[3] : undefined;\n const isMainFrame = typeof args[4] === \"boolean\" ? args[4] : undefined;\n if (isAbortedNavigationError({ errorCode, errorDescription })) {\n publishState(session);\n return;\n }\n logger?.warn?.(\"Browser Node guest navigation failed\", {\n currentUrl: contents.getURL(),\n desiredUrl: session.desiredUrl,\n errorCode,\n errorDescription,\n isMainFrame,\n nodeId: session.nodeId,\n validatedUrl,\n webContentsId: session.webContentsId\n });\n session.navigationFailureSequence += 1;\n publishState(session);\n emitBrowserNavigationFailed({\n emit,\n errorCode,\n errorDescription,\n nodeId: session.nodeId\n });\n };\n const onDestroyed = () => detachGuest(session);\n const onWillNavigate = (...args: unknown[]) => {\n const event =\n args[0] && typeof args[0] === \"object\" && \"preventDefault\" in args[0]\n ? (args[0] as { preventDefault?: () => void })\n : null;\n const url = typeof args[1] === \"string\" ? args[1] : \"\";\n if (\n url.length === 0 ||\n isBrowserNavigationAllowedByPolicy({\n policy: session.navigationPolicy,\n url\n })\n ) {\n return;\n }\n\n event?.preventDefault?.();\n emitOpenUrlFromGuest(session, url);\n publishState(session);\n };\n\n const records: BrowserGuestSession[\"listeners\"] = [\n { event: \"did-start-loading\", listener: onStateChange },\n { event: \"did-stop-loading\", listener: onStateChange },\n { event: \"did-navigate\", listener: onDidNavigate },\n { event: \"did-navigate-in-page\", listener: onStateChange },\n { event: \"page-title-updated\", listener: onStateChange },\n { event: \"will-navigate\", listener: onWillNavigate },\n { event: \"did-fail-load\", listener: onFailLoad },\n { event: \"destroyed\", listener: onDestroyed }\n ];\n\n for (const record of records) {\n contents.on(record.event, record.listener);\n }\n session.listeners = records;\n };\n\n const loadDesiredUrl = async (\n session: BrowserGuestSession\n ): Promise<void> => {\n const contents = session.contents;\n if (!contents || contents.isDestroyed()) {\n publishState(session);\n return;\n }\n\n const resolved = resolveHostBrowserNavigationUrl(session.desiredUrl);\n if (!resolved.url) {\n emit({\n ...resolveBrowserNodeUrlError(resolved),\n nodeId: session.nodeId,\n type: \"error\"\n });\n publishState(session);\n return;\n }\n if (\n !isBrowserNavigationAllowedByPolicy({\n policy: session.navigationPolicy,\n url: resolved.url\n })\n ) {\n emitOpenUrlFromGuest(session, resolved.url);\n publishState(session);\n return;\n }\n\n const currentComparable = normalizeHostBrowserComparableUrl(\n contents.getURL()\n );\n const nextComparable = normalizeHostBrowserComparableUrl(resolved.url);\n if (currentComparable && currentComparable === nextComparable) {\n publishState(session);\n return;\n }\n\n const failureSequenceBeforeLoad = session.navigationFailureSequence;\n try {\n await contents.loadURL(resolved.url);\n publishState(session);\n } catch (error) {\n const message = error instanceof Error ? error.message : String(error);\n logger?.warn?.(\"Browser Node guest loadURL failed\", {\n currentUrl: contents.getURL(),\n desiredUrl: session.desiredUrl,\n error: message,\n nodeId: session.nodeId,\n resolvedUrl: resolved.url,\n webContentsId: session.webContentsId\n });\n if (session.navigationFailureSequence !== failureSequenceBeforeLoad) {\n return;\n }\n publishState(session);\n emitBrowserNavigationFailed({\n emit,\n errorDescription: message,\n nodeId: session.nodeId\n });\n }\n };\n\n const emitOpenUrlFromGuest = (\n session: BrowserGuestSession,\n url: string\n ): { action: \"deny\" } => {\n const resolved = resolveBrowserNavigationUrl(url);\n if (resolved.url) {\n logger?.info?.(\"Browser Node guest emitted open-url\", {\n nodeId: session.nodeId,\n url: resolved.url,\n webContentsId: session.webContentsId\n });\n emit({\n reuseIfOpen: true,\n sourceNodeId: session.nodeId,\n type: \"open-url\",\n url: resolved.url\n });\n return { action: \"deny\" };\n }\n void Promise.resolve(openExternal(url)).catch((error: unknown) => {\n logger?.warn?.(\"Browser Node openExternal failed\", {\n error: error instanceof Error ? error.message : String(error)\n });\n });\n return { action: \"deny\" };\n };\n\n return {\n async activate(input) {\n const resolved = resolveHostBrowserNavigationUrl(input.url);\n if (!resolved.url) {\n throw new Error(\"Browser Node rejected navigation URL\");\n }\n const session = getSession(input.nodeId, {\n navigationPolicy: input.navigationPolicy,\n profileId: input.profileId,\n sessionMode: input.sessionMode,\n sessionPartition: input.sessionPartition,\n url: resolved.url\n });\n session.lifecycle = \"active\";\n await loadDesiredUrl(session);\n },\n async capturePreview(input) {\n const session = sessions.get(input.nodeId);\n const contents =\n session?.contents && !session.contents.isDestroyed()\n ? session.contents\n : null;\n if (!contents?.capturePage) {\n return null;\n }\n\n const image = await contents.capturePage();\n if (image.isEmpty?.() === true) {\n return null;\n }\n\n return resizeBrowserPreviewImage(image).toDataURL();\n },\n close(input) {\n const session = sessions.get(input.nodeId);\n if (session) {\n detachGuest(session);\n sessions.delete(input.nodeId);\n }\n emit({ nodeId: input.nodeId, type: \"closed\" });\n return Promise.resolve();\n },\n debugDump(input) {\n const session = sessions.get(input.nodeId);\n if (!session) {\n return null;\n }\n const contents =\n session.contents && !session.contents.isDestroyed()\n ? session.contents\n : null;\n return {\n canGoBack: contents ? canGuestGoBack(contents) : false,\n canGoForward: contents ? canGuestGoForward(contents) : false,\n currentUrl: contents ? contents.getURL() : null,\n desiredUrl: session.desiredUrl,\n isAttachedToWindow: Boolean(contents),\n isLoading: contents ? contents.isLoading() : false,\n lifecycle: session.lifecycle,\n nodeId: session.nodeId,\n profileId: session.profileId,\n sessionMode: session.sessionMode,\n sessionPartition: session.sessionPartition,\n title: contents ? contents.getTitle() : null,\n userAgent: contents?.getUserAgent?.() ?? null,\n webContentsDestroyed: session.contents\n ? session.contents.isDestroyed()\n : null,\n webContentsId: session.webContentsId\n };\n },\n goBack(input) {\n const contents = sessions.get(input.nodeId)?.contents;\n if (contents && !contents.isDestroyed() && canGuestGoBack(contents)) {\n goGuestBack(contents);\n }\n return Promise.resolve();\n },\n goForward(input) {\n const contents = sessions.get(input.nodeId)?.contents;\n if (contents && !contents.isDestroyed() && canGuestGoForward(contents)) {\n goGuestForward(contents);\n }\n return Promise.resolve();\n },\n handleGuestOpenUrl(webContentsId, input) {\n const nodeId = nodeIdByWebContentsId.get(webContentsId);\n const session = nodeId ? sessions.get(nodeId) : null;\n if (!session) {\n logger?.warn?.(\"Browser Node ignored guest open-url request\", {\n url: input.url,\n webContentsId\n });\n return;\n }\n logger?.info?.(\"Browser Node handling guest open-url request\", {\n nodeId: session.nodeId,\n url: input.url,\n webContentsId\n });\n emitOpenUrlFromGuest(session, input.url);\n },\n async navigate(input) {\n const resolved = resolveBrowserNavigationUrl(input.url);\n if (!resolved.url) {\n throw new Error(\"Browser Node rejected navigation URL\");\n }\n const session = getSession(input.nodeId, {\n navigationPolicy: input.navigationPolicy,\n url: resolved.url\n });\n session.lifecycle = \"active\";\n await loadDesiredUrl(session);\n },\n async openExternal(input) {\n const resolved = resolveBrowserNavigationUrl(input.url);\n if (!resolved.url) {\n throw new Error(\"Browser Node rejected external URL\");\n }\n await Promise.resolve(openExternal(resolved.url));\n },\n openDevTools(input) {\n const session = sessions.get(input.nodeId);\n const contents = session?.contents ?? null;\n if (contents && !contents.isDestroyed()) {\n try {\n contents.openDevTools?.({ activate: true, mode: \"detach\" });\n } catch (error) {\n logger?.warn?.(\"Browser Node open devtools failed\", {\n error: error instanceof Error ? error.message : String(error),\n nodeId: input.nodeId,\n webContentsId: session?.webContentsId ?? null\n });\n throw error;\n }\n }\n return Promise.resolve();\n },\n async prepareSession(input) {\n await prepareSession?.(input);\n getSession(input.nodeId, {\n navigationPolicy: input.navigationPolicy,\n profileId: input.profileId,\n sessionMode: input.sessionMode,\n sessionPartition: input.sessionPartition\n });\n },\n async registerGuest(input) {\n await prepareSession?.({\n nodeId: input.nodeId,\n profileId: input.profileId,\n sessionMode: input.sessionMode,\n sessionPartition: input.sessionPartition,\n navigationPolicy: input.navigationPolicy\n });\n const contents = resolveWebContents(input.webContentsId);\n if (!contents || contents.isDestroyed()) {\n throw new Error(\n `Browser Node guest ${input.webContentsId} is not available`\n );\n }\n\n const ownerNodeId = nodeIdByWebContentsId.get(input.webContentsId);\n if (ownerNodeId && ownerNodeId !== input.nodeId) {\n throw new Error(\n `Browser Node guest ${input.webContentsId} is already registered`\n );\n }\n\n const session = getSession(input.nodeId, {\n navigationPolicy: input.navigationPolicy,\n profileId: input.profileId,\n sessionMode: input.sessionMode,\n sessionPartition: input.sessionPartition\n });\n if (\n session.webContentsId === input.webContentsId &&\n session.contents === contents\n ) {\n publishState(session);\n return;\n }\n if (session.contents && session.contents !== contents) {\n detachGuest(session);\n }\n session.contents = contents;\n session.webContentsId = input.webContentsId;\n nodeIdByWebContentsId.set(input.webContentsId, input.nodeId);\n session.lifecycle = \"active\";\n logger?.info?.(\"Browser Node registered guest owner\", {\n nodeId: input.nodeId,\n sessionPartition: session.sessionPartition,\n webContentsId: input.webContentsId\n });\n contents.setWindowOpenHandler?.(({ url }) => {\n if (isGoogleGisOAuthPopupUrl(url)) {\n logger?.info?.(\"Browser Node allowing Google GIS OAuth popup\", {\n nodeId: session.nodeId,\n webContentsId: session.webContentsId\n });\n return { action: \"allow\" };\n }\n\n return emitOpenUrlFromGuest(session, url);\n });\n attachGuestListeners(session);\n await applyPreferredColorSchemeToGuest(\n session,\n logger,\n syncPreferredColorScheme,\n preferredColorScheme\n );\n await loadDesiredUrl(session);\n },\n reload(input) {\n const contents = sessions.get(input.nodeId)?.contents;\n if (contents && !contents.isDestroyed()) {\n contents.reload();\n }\n return Promise.resolve();\n },\n unregisterGuest(input) {\n const session = sessions.get(input.nodeId);\n if (!session || session.webContentsId !== input.webContentsId) {\n return Promise.resolve();\n }\n session.lifecycle = \"cold\";\n detachGuest(session);\n return Promise.resolve();\n },\n dispose() {\n unsubscribePreferredColorScheme?.();\n }\n };\n}\n","import type { BrowserWindow } from \"electron\";\nimport { resolveBrowserSessionPartition } from \"../core/session.ts\";\nimport type { BrowserNodeLoopbackPreviewRoutingOptions } from \"./loopbackPreview.ts\";\nimport { createBrowserNodeLoopbackPreviewProxy } from \"./loopbackPreviewProxy.ts\";\nimport type {\n BrowserNodeActivationInput,\n BrowserNodeGuestOpenUrlInput,\n BrowserNodeNavigateInput,\n BrowserNodeNodeIdInput,\n BrowserNodeOpenExternalInput,\n BrowserNodePrepareSessionInput,\n BrowserNodeRegisterGuestInput,\n BrowserNodeShowDevToolsContextMenuInput,\n BrowserNodeUnregisterGuestInput\n} from \"../core/types.ts\";\nimport { createBrowserGuestManager } from \"./guestManager.ts\";\nimport type {\n BrowserGuestManager,\n BrowserPreferredColorScheme,\n BrowserGuestWebContents,\n BrowserNodeElectronLogger\n} from \"./types.ts\";\n\nexport interface BrowserNodeElectronMainChannels {\n readonly activate: string;\n readonly capturePreview?: string;\n readonly close: string;\n readonly debugDump?: string;\n readonly event: string;\n readonly goBack: string;\n readonly goForward: string;\n readonly guestOpenUrl?: string;\n readonly navigate: string;\n readonly openDevTools?: string;\n readonly openExternal?: string;\n readonly prepareSession: string;\n readonly registerGuest: string;\n readonly reload: string;\n readonly showDevToolsContextMenu?: string;\n readonly unregisterGuest: string;\n}\n\nexport interface BrowserNodeElectronDevToolsContextMenuInput {\n readonly label: string;\n readonly openDevTools: () => Promise<void> | void;\n readonly ownerWindow: BrowserWindow;\n readonly point: { x: number; y: number };\n}\n\nexport interface RegisterBrowserNodeElectronMainInput {\n readonly channels: BrowserNodeElectronMainChannels;\n readonly getOwnerWindow: (event: unknown) => BrowserWindow | null;\n readonly getPreferredColorScheme?: () => BrowserPreferredColorScheme;\n readonly logger?: BrowserNodeElectronLogger;\n readonly loopbackPreviewRouting?: BrowserNodeLoopbackPreviewRoutingOptions;\n readonly openExternal: (url: string) => Promise<void> | void;\n readonly registerHandler: <TPayload, TResult>(\n channel: string,\n handler: (event: unknown, payload: TPayload) => Promise<TResult> | TResult\n ) => void;\n readonly registerListener?: <TPayload>(\n channel: string,\n handler: (event: unknown, payload: TPayload) => void\n ) => void;\n readonly resolveWebContents: (input: {\n event: unknown;\n ownerWindow: BrowserWindow;\n webContentsId: number;\n }) => BrowserGuestWebContents | null;\n readonly showDevToolsContextMenu?: (\n input: BrowserNodeElectronDevToolsContextMenuInput\n ) => Promise<void> | void;\n readonly syncPreferredColorScheme?: (\n contents: BrowserGuestWebContents,\n scheme: BrowserPreferredColorScheme\n ) => Promise<void> | void;\n readonly subscribePreferredColorScheme?: (\n listener: (scheme: BrowserPreferredColorScheme) => void\n ) => () => void;\n}\n\nexport function registerBrowserNodeElectronMain(\n input: RegisterBrowserNodeElectronMainInput\n): void {\n const managersByWindow = new WeakMap<BrowserWindow, BrowserGuestManager>();\n const loopbackPreviewProxy =\n input.loopbackPreviewRouting !== undefined\n ? createBrowserNodeLoopbackPreviewProxy({\n logger: input.logger,\n resolveSession: async ({\n profileId,\n sessionMode,\n sessionPartition\n }) => {\n const { session } = await import(\"electron\");\n return session.fromPartition(\n resolveBrowserSessionPartition({\n profileId,\n sessionMode,\n sessionPartition\n })\n );\n },\n routing: input.loopbackPreviewRouting\n })\n : null;\n\n const resolveManagerForWindow = (\n event: unknown,\n ownerWindow: BrowserWindow\n ): BrowserGuestManager => {\n const existing = managersByWindow.get(ownerWindow);\n if (existing) {\n return existing;\n }\n\n const manager = createBrowserGuestManager({\n emit(browserEvent) {\n if (!ownerWindow.isDestroyed()) {\n ownerWindow.webContents.send(input.channels.event, browserEvent);\n }\n },\n getPreferredColorScheme: input.getPreferredColorScheme,\n logger: input.logger,\n openExternal: input.openExternal,\n prepareSession:\n loopbackPreviewProxy !== null\n ? (payload) => loopbackPreviewProxy.configureSession(payload)\n : undefined,\n resolveWebContents: (webContentsId) =>\n input.resolveWebContents({\n event,\n ownerWindow,\n webContentsId\n }),\n syncPreferredColorScheme: input.syncPreferredColorScheme,\n subscribePreferredColorScheme: input.subscribePreferredColorScheme\n });\n ownerWindow.once(\"closed\", () => {\n manager.dispose();\n managersByWindow.delete(ownerWindow);\n });\n managersByWindow.set(ownerWindow, manager);\n return manager;\n };\n\n const resolveOwnedManager = (event: unknown) => {\n const ownerWindow = input.getOwnerWindow(event);\n if (!ownerWindow) {\n throw new Error(\"Browser Node IPC requires an owner window\");\n }\n return {\n manager: resolveManagerForWindow(event, ownerWindow),\n ownerWindow\n };\n };\n\n const showDevToolsContextMenu =\n input.showDevToolsContextMenu ?? showElectronDevToolsContextMenu;\n\n input.registerHandler(input.channels.prepareSession, (event, payload) =>\n resolveOwnedManager(event).manager.prepareSession(\n payload as BrowserNodePrepareSessionInput\n )\n );\n input.registerHandler(input.channels.activate, (event, payload) =>\n resolveOwnedManager(event).manager.activate(\n payload as BrowserNodeActivationInput\n )\n );\n if (input.channels.capturePreview) {\n input.registerHandler(input.channels.capturePreview, (event, payload) =>\n resolveOwnedManager(event).manager.capturePreview(\n payload as BrowserNodeNodeIdInput\n )\n );\n }\n input.registerHandler(input.channels.registerGuest, (event, payload) =>\n resolveOwnedManager(event).manager.registerGuest(\n payload as BrowserNodeRegisterGuestInput\n )\n );\n input.registerHandler(input.channels.unregisterGuest, (event, payload) =>\n resolveOwnedManager(event).manager.unregisterGuest(\n payload as BrowserNodeUnregisterGuestInput\n )\n );\n input.registerHandler(input.channels.navigate, (event, payload) =>\n resolveOwnedManager(event).manager.navigate(\n payload as BrowserNodeNavigateInput\n )\n );\n if (input.channels.openExternal) {\n input.registerHandler(input.channels.openExternal, (event, payload) =>\n resolveOwnedManager(event).manager.openExternal(\n payload as BrowserNodeOpenExternalInput\n )\n );\n }\n if (input.channels.openDevTools) {\n input.registerHandler(input.channels.openDevTools, (event, payload) => {\n const nodePayload = payload as BrowserNodeNodeIdInput;\n return resolveOwnedManager(event).manager.openDevTools(nodePayload);\n });\n }\n if (input.channels.showDevToolsContextMenu) {\n input.registerHandler(\n input.channels.showDevToolsContextMenu,\n (event, payload) => {\n const contextMenuPayload =\n payload as BrowserNodeShowDevToolsContextMenuInput;\n const { manager, ownerWindow } = resolveOwnedManager(event);\n return showDevToolsContextMenu({\n label: contextMenuPayload.label,\n openDevTools: () => {\n return manager.openDevTools({ nodeId: contextMenuPayload.nodeId });\n },\n ownerWindow,\n point: contextMenuPayload.point\n });\n }\n );\n }\n input.registerHandler(input.channels.goBack, (event, payload) =>\n resolveOwnedManager(event).manager.goBack(payload as BrowserNodeNodeIdInput)\n );\n input.registerHandler(input.channels.goForward, (event, payload) =>\n resolveOwnedManager(event).manager.goForward(\n payload as BrowserNodeNodeIdInput\n )\n );\n input.registerHandler(input.channels.reload, (event, payload) =>\n resolveOwnedManager(event).manager.reload(payload as BrowserNodeNodeIdInput)\n );\n input.registerHandler(input.channels.close, (event, payload) =>\n resolveOwnedManager(event).manager.close(payload as BrowserNodeNodeIdInput)\n );\n if (input.channels.debugDump) {\n input.registerHandler(input.channels.debugDump, (event, payload) =>\n resolveOwnedManager(event).manager.debugDump(\n payload as BrowserNodeNodeIdInput\n )\n );\n }\n if (input.channels.guestOpenUrl && input.registerListener) {\n input.registerListener(input.channels.guestOpenUrl, (event, payload) => {\n const senderId = readBrowserNodeIpcSenderId(event);\n const openUrlPayload = payload as BrowserNodeGuestOpenUrlInput | null;\n if (\n typeof senderId !== \"number\" ||\n !Number.isFinite(senderId) ||\n !openUrlPayload ||\n typeof openUrlPayload.url !== \"string\"\n ) {\n return;\n }\n resolveOwnedManager(event).manager.handleGuestOpenUrl(\n senderId,\n openUrlPayload\n );\n });\n }\n}\n\nasync function showElectronDevToolsContextMenu(\n input: BrowserNodeElectronDevToolsContextMenuInput\n): Promise<void> {\n const { Menu } = await import(\"electron\");\n const menu = Menu.buildFromTemplate([\n {\n click: () => {\n void Promise.resolve(input.openDevTools()).catch(() => undefined);\n },\n label: input.label\n }\n ]);\n menu.popup({\n window: input.ownerWindow,\n x: Math.round(input.point.x),\n y: Math.round(input.point.y)\n });\n}\n\nfunction readBrowserNodeIpcSenderId(event: unknown): number | null {\n const sender = (event as { sender?: { id?: unknown } } | null)?.sender;\n return typeof sender?.id === \"number\" ? sender.id : null;\n}\n","import type { WebContents } from \"electron\";\nimport type { BrowserNodeElectronLogger } from \"./types.ts\";\n\nconst electronUserAgentTokenPattern = /\\sElectron\\/[^\\s]+/g;\n\nexport function sanitizeBrowserGuestUserAgent(userAgent: string): string {\n return userAgent\n .trim()\n .replace(electronUserAgentTokenPattern, \"\")\n .replace(/\\s{2,}/g, \" \");\n}\n\nexport function applyBrowserGuestUserAgent(\n contents: WebContents,\n logger?: BrowserNodeElectronLogger\n): void {\n const guestContents = contents as WebContents & {\n getUserAgent?: () => string;\n setUserAgent?: (userAgent: string) => void;\n };\n if (\n typeof guestContents.getUserAgent !== \"function\" ||\n typeof guestContents.setUserAgent !== \"function\"\n ) {\n return;\n }\n\n const currentUserAgent = guestContents.getUserAgent().trim();\n const nextUserAgent = sanitizeBrowserGuestUserAgent(currentUserAgent);\n if (!nextUserAgent || nextUserAgent === currentUserAgent) {\n return;\n }\n\n guestContents.setUserAgent(nextUserAgent);\n logger?.debug?.(\"Browser Node sanitized guest user agent\", {\n webContentsId: contents.id ?? null\n });\n}\n","import type { Event, WebContents, WebPreferences } from \"electron\";\nimport {\n isBrowserSessionPartitionAllowed,\n type BrowserSessionPartitionAllowedOptions\n} from \"../core/session.ts\";\nimport { resolveBrowserNavigationUrl } from \"../core/url.ts\";\nimport { applyBrowserGuestUserAgent } from \"./userAgent.ts\";\nimport type { BrowserNodeElectronLogger } from \"./types.ts\";\n\nexport interface BrowserWebviewSecurityInput {\n allowedSessionPartitions?: BrowserSessionPartitionAllowedOptions;\n params: Record<string, string>;\n resolvePreload?: BrowserWebviewPreloadResolver;\n webPreferences: WebPreferences;\n}\n\nexport interface BrowserWebviewSecurityResult {\n allowed: boolean;\n reason: string | null;\n}\n\nexport type BrowserNodeWebviewMatcher = (\n params: Record<string, string>\n) => boolean;\n\ninterface PendingBrowserWebviewAttach {\n allowNativePopups: boolean;\n}\n\nexport interface BrowserWebviewPreloadResolverInput {\n params: Readonly<Record<string, string>>;\n}\n\nexport type BrowserWebviewPreloadResolver = (\n input: BrowserWebviewPreloadResolverInput\n) => string | null | undefined;\n\nfunction isBrowserNodeInitialWebviewUrl(url: string | undefined): boolean {\n return (url ?? \"\").trim() === \"about:blank\";\n}\n\nexport function isBrowserNodeWebviewAttach(\n params: Record<string, string>,\n allowedSessionPartitions?: BrowserSessionPartitionAllowedOptions\n): boolean {\n return (\n params[\"data-browser-node-webview\"] === \"true\" ||\n isBrowserSessionPartitionAllowed(params.partition, allowedSessionPartitions)\n );\n}\n\nfunction shouldAllowBrowserNodeNativePopups(\n params: Record<string, string>\n): boolean {\n return (\n params[\"data-browser-node-webview\"] === \"true\" ||\n isBrowserSessionPartitionAllowed(params.partition)\n );\n}\n\nfunction getPopupLogMetadata(url: string): Record<string, unknown> {\n try {\n const parsed = new URL(url);\n return {\n popupOrigin: parsed.origin,\n popupPath: parsed.pathname,\n popupProtocol: parsed.protocol\n };\n } catch {\n return {\n popupOrigin: null,\n popupPath: null,\n popupProtocol: null\n };\n }\n}\n\nfunction externalizeBrowserNodePopupWindow({\n guestWebContentsId,\n logger,\n openExternal,\n url\n}: {\n guestWebContentsId: number | null;\n logger?: BrowserNodeElectronLogger;\n openExternal: (url: string) => Promise<void> | void;\n url: string;\n}) {\n logger?.info?.(\"Browser Node webview popup externalized\", {\n guestWebContentsId,\n ...getPopupLogMetadata(url)\n });\n const resolved = resolveBrowserNavigationUrl(url);\n if (resolved.url) {\n void Promise.resolve(openExternal(resolved.url)).catch(() => undefined);\n }\n return { action: \"deny\" as const };\n}\n\nexport function enforceBrowserWebviewSecurity({\n allowedSessionPartitions,\n params,\n resolvePreload,\n webPreferences\n}: BrowserWebviewSecurityInput): BrowserWebviewSecurityResult {\n webPreferences.allowRunningInsecureContent = false;\n webPreferences.contextIsolation = true;\n webPreferences.javascript = true;\n webPreferences.nodeIntegration = false;\n webPreferences.plugins = false;\n webPreferences.sandbox = true;\n webPreferences.webSecurity = true;\n delete webPreferences.preload;\n\n const partition = params.partition;\n if (\n !partition ||\n !isBrowserSessionPartitionAllowed(partition, allowedSessionPartitions)\n ) {\n return {\n allowed: false,\n reason: \"Unsupported Browser Node session partition\"\n };\n }\n\n if (isBrowserNodeInitialWebviewUrl(params.src)) {\n params.src = \"about:blank\";\n } else {\n const resolved = resolveBrowserNavigationUrl(params.src ?? \"about:blank\");\n if (!resolved.url) {\n return {\n allowed: false,\n reason: \"Unsupported browser URL\"\n };\n }\n params.src = resolved.url;\n }\n\n const preload = resolvePreload?.({ params: { ...params } });\n const resolvedPreload = typeof preload === \"string\" ? preload.trim() : \"\";\n if (resolvedPreload.length > 0) {\n webPreferences.preload = resolvedPreload;\n }\n\n return { allowed: true, reason: null };\n}\n\nexport interface InstallBrowserWebviewSecurityInput {\n allowedSessionPartitions?: BrowserSessionPartitionAllowedOptions;\n contents: WebContents;\n logger?: BrowserNodeElectronLogger;\n onGuestAttached?: (guestContents: WebContents) => void;\n openExternal: (url: string) => Promise<void> | void;\n resolvePreload?: BrowserWebviewPreloadResolver;\n shouldHandleWebview?: BrowserNodeWebviewMatcher;\n}\n\nexport function installBrowserWebviewSecurity({\n allowedSessionPartitions,\n contents,\n logger,\n onGuestAttached,\n openExternal,\n resolvePreload,\n shouldHandleWebview\n}: InstallBrowserWebviewSecurityInput): () => void {\n const pendingBrowserAttaches: PendingBrowserWebviewAttach[] = [];\n\n const handleWillAttachWebview = (\n event: Event,\n webPreferences: WebPreferences,\n params: Record<string, string>\n ) => {\n const shouldHandle =\n shouldHandleWebview?.(params) ??\n isBrowserNodeWebviewAttach(params, allowedSessionPartitions);\n logger?.debug?.(\"Browser Node webview will attach\", {\n partition: params.partition ?? null,\n shouldHandle,\n src: params.src ?? null\n });\n if (!shouldHandle) {\n return;\n }\n\n const allowNativePopups = shouldAllowBrowserNodeNativePopups(params);\n if (allowNativePopups) {\n params.allowpopups = \"true\";\n }\n logger?.info?.(\"Browser Node webview popup policy applied\", {\n allowNativePopups,\n allowpopups: params.allowpopups ?? null,\n partition: params.partition ?? null,\n src: params.src ?? null\n });\n\n const result = enforceBrowserWebviewSecurity({\n allowedSessionPartitions,\n params,\n resolvePreload,\n webPreferences\n });\n if (!result.allowed) {\n logger?.warn?.(\"Browser Node webview blocked\", { reason: result.reason });\n event.preventDefault();\n return;\n }\n pendingBrowserAttaches.push({\n allowNativePopups\n });\n logger?.debug?.(\"Browser Node webview attach allowed\", {\n partition: params.partition ?? null,\n src: params.src ?? null\n });\n };\n\n const handleDidAttachWebview = (\n _event: Event,\n guestContents: WebContents\n ) => {\n const pendingAttach = pendingBrowserAttaches.shift();\n if (!pendingAttach) {\n logger?.debug?.(\"Browser Node webview did attach ignored\", {\n guestWebContentsId: guestContents.id ?? null,\n pendingBrowserAttachCount: pendingBrowserAttaches.length\n });\n return;\n }\n\n applyBrowserGuestUserAgent(guestContents, logger);\n if (pendingAttach.allowNativePopups) {\n guestContents.setWindowOpenHandler(({ url }) => {\n return externalizeBrowserNodePopupWindow({\n guestWebContentsId: guestContents.id ?? null,\n logger,\n openExternal,\n url\n });\n });\n } else {\n guestContents.setWindowOpenHandler(({ url }) => {\n return externalizeBrowserNodePopupWindow({\n guestWebContentsId: guestContents.id ?? null,\n logger,\n openExternal,\n url\n });\n });\n }\n onGuestAttached?.(guestContents);\n logger?.debug?.(\"Browser Node webview guest attached\", {\n guestWebContentsId: guestContents.id ?? null,\n pendingBrowserAttachCount: pendingBrowserAttaches.length\n });\n };\n\n contents.on(\"will-attach-webview\", handleWillAttachWebview);\n contents.on(\"did-attach-webview\", handleDidAttachWebview);\n\n return () => {\n contents.off(\"will-attach-webview\", handleWillAttachWebview);\n contents.off(\"did-attach-webview\", handleDidAttachWebview);\n };\n}\n"],"mappings":"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","names":[]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@tutti-os/browser-node",
|
|
3
|
-
"version": "0.0.
|
|
3
|
+
"version": "0.0.6",
|
|
4
4
|
"private": false,
|
|
5
5
|
"type": "module",
|
|
6
6
|
"types": "./dist/index.d.ts",
|
|
@@ -48,10 +48,10 @@
|
|
|
48
48
|
"directory": "packages/browser/workbench-node"
|
|
49
49
|
},
|
|
50
50
|
"dependencies": {
|
|
51
|
-
"@tutti-os/ui-i18n-runtime": "0.0.
|
|
52
|
-
"@tutti-os/ui-react-hooks": "0.0.
|
|
53
|
-
"@tutti-os/ui-system": "0.0.
|
|
54
|
-
"@tutti-os/workbench-surface": "0.0.
|
|
51
|
+
"@tutti-os/ui-i18n-runtime": "0.0.6",
|
|
52
|
+
"@tutti-os/ui-react-hooks": "0.0.6",
|
|
53
|
+
"@tutti-os/ui-system": "0.0.6",
|
|
54
|
+
"@tutti-os/workbench-surface": "0.0.6",
|
|
55
55
|
"ws": "^8.21.0"
|
|
56
56
|
},
|
|
57
57
|
"devDependencies": {
|