export class MockWebSocketHelper extends EventTarget { static #mocks = new Map(); static async init(page) { this.#mocks = new Map(); await page.exposeFunction("onMockWebSocketConstructor", (url) => { const webSocket = new MockWebSocketHelper(page, url); this.#mocks.set(url, webSocket); }); await page.exposeFunction("onMockWebSocketSpyMessage", (url, data) => { if (!this.#mocks.has(url)) { throw new Error(`WebSocket with URL ${url} not found`); } this.#mocks.get(url).dispatchEvent(new MessageEvent("message", { data })); }); await page.exposeFunction("onMockWebSocketSpyClose", (url, code, reason) => { if (!this.#mocks.has(url)) { throw new Error(`WebSocket with URL ${url} not found`); } this.#mocks.get(url).dispatchEvent(new CloseEvent("close", { code, reason })); }); await page.addInitScript({ path: "playwright/scripts/MockWebSocket.js" }); } static waitForURL(url) { return new Promise((resolve) => { const intervalID = setInterval(() => { for (const [wsURL, ws] of this.#mocks) { if (wsURL.includes(url)) { clearInterval(intervalID); return resolve(ws); } } }, 30); }); } #page = null; #url; constructor(page, url, protocols) { super(); this.#page = page; this.#url = url; } mockOpen(options) { return this.#page.evaluate( ({ url, options }) => { if (typeof WebSocket.getByURL !== "function") { throw new Error( "WebSocket.getByURL is not a function. Did you forget to call MockWebSocket.init(page)?", ); } WebSocket.getByURL(url).mockOpen(options); }, { url: this.#url, options }, ); } mockMessage(data) { return this.#page.evaluate( ({ url, data }) => { if (typeof WebSocket.getByURL !== "function") { throw new Error( "WebSocket.getByURL is not a function. Did you forget to call MockWebSocket.init(page)?", ); } WebSocket.getByURL(url).mockMessage(data); }, { url: this.#url, data }, ); } mockClose() { return this.#page.evaluate( ({ url }) => { if (typeof WebSocket.getByURL !== "function") { throw new Error( "WebSocket.getByURL is not a function. Did you forget to call MockWebSocket.init(page)?", ); } WebSocket.getByURL(url).mockClose(); }, { url: this.#url }, ); } }