UNPKG

@cloudflare/puppeteer

Version:

A high-level API to control headless Chrome over the DevTools Protocol

77 lines (68 loc) 2.23 kB
/** * @license * Copyright 2025 Google Inc. * SPDX-License-Identifier: Apache-2.0 */ import type {ConnectionTransport} from '../common/ConnectionTransport.js'; import {packageVersion} from '../generated/version.js'; import type {BrowserWorker} from './BrowserWorker.js'; import {messageToChunks, chunksToMessage} from './chunking.js'; const FAKE_HOST = 'https://fake.host'; export class WorkersWebSocketTransport implements ConnectionTransport { ws: WebSocket; pingInterval: NodeJS.Timer; chunks: Uint8Array[] = []; onmessage?: (message: string) => void; onclose?: () => void; sessionId: string; static async create( endpoint: BrowserWorker, sessionId: string ): Promise<WorkersWebSocketTransport> { const path = `${FAKE_HOST}/v1/connectDevtools?browser_session=${sessionId}`; const response = await endpoint.fetch(path, { headers: { Upgrade: 'websocket', 'cf-brapi-client': `@cloudflare/puppeteer@${packageVersion}`, }, }); response.webSocket!.accept(); return new WorkersWebSocketTransport(response.webSocket!, sessionId); } constructor(ws: WebSocket, sessionId: string) { this.pingInterval = setInterval(() => { return this.ws.send('ping'); }, 1000); // TODO more investigation this.ws = ws; this.sessionId = sessionId; this.ws.addEventListener('message', event => { this.chunks.push(new Uint8Array(event.data as ArrayBuffer)); const message = chunksToMessage(this.chunks, sessionId); if (message && this.onmessage) { this.onmessage!(message); } }); this.ws.addEventListener('close', () => { clearInterval(this.pingInterval as NodeJS.Timeout); if (this.onclose) { this.onclose(); } }); this.ws.addEventListener('error', e => { console.error(`Websocket error: SessionID: ${sessionId}`, e); clearInterval(this.pingInterval as NodeJS.Timeout); }); } send(message: string): void { for (const chunk of messageToChunks(message)) { this.ws.send(chunk); } } close(): void { clearInterval(this.pingInterval as NodeJS.Timeout); this.ws.close(); } toString(): string { return this.sessionId; } }