From 211ff8d6459e99114a3faddc9d0e69dd1683b0c7 Mon Sep 17 00:00:00 2001 From: Cacie Prins Date: Thu, 18 Jul 2024 11:00:32 -0400 Subject: [PATCH 01/25] Refactor connection logic to cri-connection CDPConnection class instantiates a persistent event emitter, and creates a CDP Client instance via connect(). When this client emits a disconnect event, the CDPConnection will automatically attempt to reconnect, if the connection was not intentionally closed and the connection is created with the reconnect flag. It only listens to 'event' and 'disconnect' events itself, making it easy to unsubscribe from events in the event of a disconnection, helping ease mental load when considering potential memory leaks. Because it uses its own persistent event emitter, event listeners added to the CDPConnection do not need to be re-added when the client reconnects. The CriClient itself no longer handles reconnection logic: it only reacts to connection state changes in order to re-send failed commands and enablements when the CDPConnection restores the CDPClient. --- .../server/lib/browsers/cdp_automation.ts | 2 +- packages/server/lib/browsers/chrome.ts | 4 +- packages/server/lib/browsers/electron.ts | 2 +- packages/server/lib/browsers/firefox-util.ts | 2 +- packages/server/lib/browsers/firefox.ts | 2 +- .../browser-cri-client.ts | 13 +- .../cdp-command-queue.ts | 2 +- .../remote-interface/cdp-connection.ts | 198 +++++++++ .../{ => remote-interface}/cri-client.ts | 419 ++++++------------ .../remote-interface/debug-cdp-connection.ts | 81 ++++ .../lib/browsers/remote-interface/errors.ts | 41 ++ packages/server/lib/browsers/utils.ts | 2 +- .../server/test/integration/cypress_spec.js | 2 +- .../unit/browsers/browser-cri-client_spec.ts | 6 +- .../server/test/unit/browsers/chrome_spec.js | 2 +- .../test/unit/browsers/cri-client_spec.ts | 51 ++- .../test/unit/browsers/electron_spec.js | 2 +- .../server/test/unit/browsers/firefox_spec.ts | 2 +- 18 files changed, 508 insertions(+), 325 deletions(-) rename packages/server/lib/browsers/{ => remote-interface}/browser-cri-client.ts (98%) rename packages/server/lib/browsers/{ => remote-interface}/cdp-command-queue.ts (97%) create mode 100644 packages/server/lib/browsers/remote-interface/cdp-connection.ts rename packages/server/lib/browsers/{ => remote-interface}/cri-client.ts (51%) create mode 100644 packages/server/lib/browsers/remote-interface/debug-cdp-connection.ts create mode 100644 packages/server/lib/browsers/remote-interface/errors.ts diff --git a/packages/server/lib/browsers/cdp_automation.ts b/packages/server/lib/browsers/cdp_automation.ts index 853a14ad1a2e..7e3f6d695f92 100644 --- a/packages/server/lib/browsers/cdp_automation.ts +++ b/packages/server/lib/browsers/cdp_automation.ts @@ -13,7 +13,7 @@ import type { ResourceType, BrowserPreRequest, BrowserResponseReceived } from '@ import type { CDPClient, ProtocolManagerShape, WriteVideoFrame } from '@packages/types' import type { Automation } from '../automation' import { cookieMatches, CyCookie, CyCookieFilter } from '../automation/util' -import { DEFAULT_NETWORK_ENABLE_OPTIONS, CriClient } from './cri-client' +import { DEFAULT_NETWORK_ENABLE_OPTIONS, CriClient } from './remote-interface/cri-client' export type CdpCommand = keyof ProtocolMapping.Commands diff --git a/packages/server/lib/browsers/chrome.ts b/packages/server/lib/browsers/chrome.ts index 33ae2d6643f9..5b8b469d71e3 100644 --- a/packages/server/lib/browsers/chrome.ts +++ b/packages/server/lib/browsers/chrome.ts @@ -15,9 +15,9 @@ import { CdpAutomation, screencastOpts } from './cdp_automation' import * as protocol from './protocol' import utils from './utils' import * as errors from '../errors' -import { BrowserCriClient } from './browser-cri-client' +import { BrowserCriClient } from './remote-interface/browser-cri-client' import type { Browser, BrowserInstance, GracefulShutdownOptions } from './types' -import type { CriClient } from './cri-client' +import type { CriClient } from './remote-interface/cri-client' import type { Automation } from '../automation' import memory from './memory' diff --git a/packages/server/lib/browsers/electron.ts b/packages/server/lib/browsers/electron.ts index 97f318f1bd1b..02c8ebd67312 100644 --- a/packages/server/lib/browsers/electron.ts +++ b/packages/server/lib/browsers/electron.ts @@ -14,7 +14,7 @@ import type { Automation } from '../automation' import type { BrowserLaunchOpts, Preferences, ProtocolManagerShape, RunModeVideoApi } from '@packages/types' import type { CDPSocketServer } from '@packages/socket/lib/cdp-socket' import memory from './memory' -import { BrowserCriClient } from './browser-cri-client' +import { BrowserCriClient } from './remote-interface/browser-cri-client' import { getRemoteDebuggingPort } from '../util/electron-app' // TODO: unmix these two types diff --git a/packages/server/lib/browsers/firefox-util.ts b/packages/server/lib/browsers/firefox-util.ts index 38fff4cf01fa..dfe39a6fd8af 100644 --- a/packages/server/lib/browsers/firefox-util.ts +++ b/packages/server/lib/browsers/firefox-util.ts @@ -7,7 +7,7 @@ import util from 'util' import Foxdriver from '@benmalka/foxdriver' import * as protocol from './protocol' import { CdpAutomation } from './cdp_automation' -import { BrowserCriClient } from './browser-cri-client' +import { BrowserCriClient } from './remote-interface/browser-cri-client' import type { Automation } from '../automation' const errors = require('../errors') diff --git a/packages/server/lib/browsers/firefox.ts b/packages/server/lib/browsers/firefox.ts index 8d78cf89da4b..f460b567c3b6 100644 --- a/packages/server/lib/browsers/firefox.ts +++ b/packages/server/lib/browsers/firefox.ts @@ -16,7 +16,7 @@ import os from 'os' import treeKill from 'tree-kill' import mimeDb from 'mime-db' import { getRemoteDebuggingPort } from './protocol' -import type { BrowserCriClient } from './browser-cri-client' +import type { BrowserCriClient } from './remote-interface/browser-cri-client' import type { Automation } from '../automation' import { getCtx } from '@packages/data-context' import { getError } from '@packages/errors' diff --git a/packages/server/lib/browsers/browser-cri-client.ts b/packages/server/lib/browsers/remote-interface/browser-cri-client.ts similarity index 98% rename from packages/server/lib/browsers/browser-cri-client.ts rename to packages/server/lib/browsers/remote-interface/browser-cri-client.ts index 55fc70739ef4..11d279be337e 100644 --- a/packages/server/lib/browsers/browser-cri-client.ts +++ b/packages/server/lib/browsers/remote-interface/browser-cri-client.ts @@ -3,8 +3,9 @@ import Bluebird from 'bluebird' import CRI from 'chrome-remote-interface' import Debug from 'debug' import type { Protocol } from 'devtools-protocol' -import { _connectAsync, _getDelayMsForRetry } from './protocol' -import * as errors from '../errors' +import { _connectAsync, _getDelayMsForRetry } from '../protocol' +import * as errors from '../../errors' +import type { CypressError } from '@packages/errors' import { CriClient, DEFAULT_NETWORK_ENABLE_OPTIONS } from './cri-client' import { serviceWorkerClientEventHandler, serviceWorkerClientEventHandlerName } from '@packages/proxy/lib/http/util/service-worker-manager' import type { ProtocolManagerShape } from '@packages/types' @@ -23,7 +24,7 @@ type BrowserCriClientOptions = { host: string port: number browserName: string - onAsynchronousError: Function + onAsynchronousError: (err: CypressError) => void protocolManager?: ProtocolManagerShape fullyManageTabs?: boolean onServiceWorkerClientEvent: ServiceWorkerEventHandler @@ -33,7 +34,7 @@ type BrowserCriClientCreateOptions = { browserName: string fullyManageTabs?: boolean hosts: string[] - onAsynchronousError: Function + onAsynchronousError: (err: CypressError) => void onReconnect?: (client: CriClient) => void port: number protocolManager?: ProtocolManagerShape @@ -181,7 +182,7 @@ export class BrowserCriClient { private host: string private port: number private browserName: string - private onAsynchronousError: Function + private onAsynchronousError: (err: CypressError) => void private protocolManager?: ProtocolManagerShape private fullyManageTabs?: boolean onServiceWorkerClientEvent: ServiceWorkerEventHandler @@ -479,7 +480,7 @@ export class BrowserCriClient { browserCriClient.onClose = resolve // or when the browser's CDP ws connection is closed - browserClient.ws.once('close', () => { + browserClient.ws?.once('close', () => { resolve(false) }) }) diff --git a/packages/server/lib/browsers/cdp-command-queue.ts b/packages/server/lib/browsers/remote-interface/cdp-command-queue.ts similarity index 97% rename from packages/server/lib/browsers/cdp-command-queue.ts rename to packages/server/lib/browsers/remote-interface/cdp-command-queue.ts index e79904e4960f..452c04e9c0f5 100644 --- a/packages/server/lib/browsers/cdp-command-queue.ts +++ b/packages/server/lib/browsers/remote-interface/cdp-command-queue.ts @@ -1,6 +1,6 @@ import type ProtocolMapping from 'devtools-protocol/types/protocol-mapping' import pDefer, { DeferredPromise } from 'p-defer' -import type { CdpCommand } from './cdp_automation' +import type { CdpCommand } from '../cdp_automation' import Debug from 'debug' const debug = Debug('cypress:server:browsers:cdp-command-queue') diff --git a/packages/server/lib/browsers/remote-interface/cdp-connection.ts b/packages/server/lib/browsers/remote-interface/cdp-connection.ts new file mode 100644 index 000000000000..bdfb76ed7313 --- /dev/null +++ b/packages/server/lib/browsers/remote-interface/cdp-connection.ts @@ -0,0 +1,198 @@ +import type ProtocolMapping from 'devtools-protocol/types/protocol-mapping' +import Debug from 'debug' +import EventEmitter from 'events' +import CDP from 'chrome-remote-interface' +import type { CypressError } from '@packages/errors' +import { debugCdpConnection, DebuggableCDPClient } from './debug-cdp-connection' +import type { CdpEvent, CdpCommand } from '../cdp_automation' +import { CdpDisconnectedError, CdpTerminatedError } from './errors' +import { asyncRetry } from '../../util/async_retry' +import * as errors from '../../errors' +import type WebSocket from 'ws' + +const debug = Debug('cypress:server:browsers:cdp-connection') + +export type CDPListener = (params: ProtocolMapping.Events[T][0], sessionId?: string) => void + +// CDPClient extends EventEmitter, but does not export that type information from its +// definitelytyped module +type CdpClient = Exclude & CDP.Client + +/** + * There are three error messages we can encounter which should not be re-thrown, but + * should trigger a reconnection attempt if one is not in progress, and enqueue the + * command that errored. This regex is used in client.send to check for: + * - WebSocket connection closed + * - WebSocket not open + * - WebSocket already in CLOSING or CLOSED state + */ +const isWebsocketClosedErrorMessage = (message: string) => { + return /^WebSocket (?:connection closed|is (?:not open|already in CLOSING or CLOSED state))/.test(message) +} + +type CDPConnectionOptions = { + automaticallyReconnect: boolean +} + +type CDPConnectionEventListeners = { + 'cdp-connection-reconnect-error': (err: CypressError) => void + 'cdp-connection-reconnect': () => void + 'cdp-connection-closed': () => void + 'cdp-connection-reconnect-attempt': (attemptNumber: number) => void +} +type CDPConnectionEvent = keyof CDPConnectionEventListeners + +type CDPConnectionEventListener = CDPConnectionEventListeners[T] + +export class CDPConnection { + private _emitter: EventEmitter = new EventEmitter() + private _connection: CdpClient | undefined + private _autoReconnect: boolean + private _terminated: boolean = false + private _reconnection: Promise | undefined + + constructor (private readonly _options: CDP.Options, connectionoptions: CDPConnectionOptions) { + this._autoReconnect = connectionoptions.automaticallyReconnect + } + + get terminated () { + return this._terminated + } + + get ws () { + // this is reached into by browser-cri-client to detect close events - needs rethinking + return (this._connection as { _ws?: WebSocket})._ws + } + + on (event: T, callback: CDPListener) { + debug('attaching event listener to cdp connection', event) + this._emitter.on(event, callback) + } + addConnectionEventListener (event: T, callback: CDPConnectionEventListener) { + this._emitter.on(event, callback) + } + off (event: T, callback: CDPListener) { + this._emitter.off(event, callback) + } + removeConnectionEventListener (event: T, callback: CDPConnectionEventListener) { + this._emitter.on(event, callback) + } + + async connect (): Promise { + if (this._terminated) { + throw new CdpTerminatedError(`Cannot reconnect to CDP. Client target ${this._options.target} has been terminated.`) + } + + if (this._connection) { + await this._gracefullyDisconnect() + } + + this._connection = await CDP(this._options) as CdpClient + + debugCdpConnection(debug.namespace, this._connection as DebuggableCDPClient) + + this._connection.on('event', this._broadcastEvent) + + if (this._autoReconnect) { + this._connection.on('disconnect', this._reconnect) + } + } + + async disconnect () { + if (this._terminated && !this._connection) { + return + } + + this._terminated = true + + await this._gracefullyDisconnect() + + this._emitter.emit('cdp-connection-closed') + } + + private _gracefullyDisconnect = async () => { + this._connection?.off('event', this._broadcastEvent) + this._connection?.off('disconnect', this._reconnect) + await this._connection?.close() + this._connection = undefined + } + + async send ( + command: T, + data?: ProtocolMapping.Commands[T]['paramsType'][0], + sessionId?: string, + ): Promise { + // TODO: what if in the middle of a reconnection attempt? + if (!this._connection || this._terminated) { + throw new CdpDisconnectedError(`${command} will not run as the CRI connection is not available`) + } + + try { + return await this._connection.send(command, data, sessionId) + } catch (e) { + // Clients may wish to determine if the command should be enqueued + // should enqueue logic live in this class tho?? + if (isWebsocketClosedErrorMessage(e.message)) { + throw new CdpDisconnectedError(`${command} failed due to the websocket being disconnected.`, e) + } + + throw e + } + } + + private _reconnect = async () => { + if (this._terminated) { + return + } + + if (this._reconnection) { + return this._reconnection + } + + let attempt = 0 + + this._reconnection = asyncRetry(async () => { + attempt++ + + if (this._terminated) { + throw new CdpTerminatedError(`Cannot reconnect to CDP. Client target ${this._options.target} has been terminated.`) + } + + this._emitter.emit('cdp-connection-reconnect-attempt', attempt) + + await this.connect() + }, { + maxAttempts: 20, + retryDelay: () => { + return 100 + }, + shouldRetry (err) { + return !(err && CdpTerminatedError.isCdpTerminatedError(err)) + }, + })() + + try { + await this._reconnection + this._emitter.emit('cdp-connection-reconnect') + } catch (err) { + debug('error(s) on reconnecting: ', err) + const significantError: Error = err.errors ? (err as AggregateError).errors[err.errors.length - 1] : err + + const retryHaltedDueToClosed = CdpTerminatedError.isCdpTerminatedError(err) || + (err as AggregateError)?.errors?.find((predicate) => CdpTerminatedError.isCdpTerminatedError(predicate)) + + if (!retryHaltedDueToClosed) { + const cdpError = errors.get('CDP_COULD_NOT_RECONNECT', significantError) + + cdpError.isFatalApiErr = true + this._emitter.emit('cdp-connection-reconnect-error', cdpError) + } + } + + this._reconnection = undefined + } + + private _broadcastEvent = ({ method, params, sessionId }: { method: CdpEvent, params: Record, sessionId?: string }) => { + this._emitter.emit(method, params, sessionId) + } +} diff --git a/packages/server/lib/browsers/cri-client.ts b/packages/server/lib/browsers/remote-interface/cri-client.ts similarity index 51% rename from packages/server/lib/browsers/cri-client.ts rename to packages/server/lib/browsers/remote-interface/cri-client.ts index 595fbeb09b08..fd282a71b632 100644 --- a/packages/server/lib/browsers/cri-client.ts +++ b/packages/server/lib/browsers/remote-interface/cri-client.ts @@ -1,34 +1,14 @@ -import CDP from 'chrome-remote-interface' import debugModule from 'debug' -import _ from 'lodash' -import * as errors from '../errors' import { CDPCommandQueue } from './cdp-command-queue' -import { asyncRetry } from '../util/async_retry' +import { CDPConnection, type CDPListener } from './cdp-connection' import type ProtocolMapping from 'devtools-protocol/types/protocol-mapping' -import type EventEmitter from 'events' import type WebSocket from 'ws' - -import type { SendDebuggerCommand, OnFn, OffFn, CdpCommand, CdpEvent } from './cdp_automation' +import type { CypressError } from '@packages/errors' +import type { SendDebuggerCommand, OnFn, OffFn, CdpCommand, CdpEvent } from '../cdp_automation' +import { CdpDisconnectedError } from './errors' import type { ProtocolManagerShape } from '@packages/types' const debug = debugModule('cypress:server:browsers:cri-client') -const debugVerbose = debugModule('cypress-verbose:server:browsers:cri-client') -// debug using cypress-verbose:server:browsers:cri-client:send:* -const debugVerboseSend = debugModule(`${debugVerbose.namespace}:send:[-->]`) -// debug using cypress-verbose:server:browsers:cri-client:recv:* -const debugVerboseReceive = debugModule(`${debugVerbose.namespace}:recv:[<--]`) -// debug using cypress-verbose:server:browsers:cri-client:err:* -const debugVerboseLifecycle = debugModule(`${debugVerbose.namespace}:ws`) - -/** - * There are three error messages we can encounter which should not be re-thrown, but - * should trigger a reconnection attempt if one is not in progress, and enqueue the - * command that errored. This regex is used in client.send to check for: - * - WebSocket connection closed - * - WebSocket not open - * - WebSocket already in CLOSING or CLOSED state - */ -const WEBSOCKET_NOT_OPEN_RE = /^WebSocket (?:connection closed|is (?:not open|already in CLOSING or CLOSED state))/ type QueuedMessages = { enableCommands: EnableCommand[] @@ -56,20 +36,6 @@ type Subscription = { type CmdParams = ProtocolMapping.Commands[TCmd]['paramsType'][0] -interface CDPClient extends CDP.Client { - off: EventEmitter['off'] - _ws: WebSocket -} - -const ConnectionClosedKind: 'CONNECTION_CLOSED' = 'CONNECTION_CLOSED' - -class ConnectionClosedError extends Error { - public readonly kind = ConnectionClosedKind - static isConnectionClosedError (err: Error & { kind?: any }): err is ConnectionClosedError { - return err.kind === ConnectionClosedKind - } -} - export const DEFAULT_NETWORK_ENABLE_OPTIONS = { maxTotalBufferSize: 0, maxResourceBufferSize: 0, @@ -84,7 +50,7 @@ export interface ICriClient { /** * The underlying websocket connection */ - ws: CDPClient['_ws'] + ws?: WebSocket /** * Sends a command to the Chrome remote interface. * @example client.send('Page.navigate', { url }) @@ -118,70 +84,10 @@ export interface ICriClient { off: OffFn } -const maybeDebugCdpMessages = (cri: CDPClient) => { - if (debugVerboseReceive.enabled) { - cri._ws.prependListener('message', (data) => { - data = _ - .chain(JSON.parse(data)) - .tap((data) => { - ([ - 'params.data', // screencast frame data - 'result.data', // screenshot data - ]).forEach((truncatablePath) => { - const str = _.get(data, truncatablePath) - - if (!_.isString(str)) { - return - } - - _.set(data, truncatablePath, _.truncate(str, { - length: 100, - omission: `... [truncated string of total bytes: ${str.length}]`, - })) - }) - - return data - }) - .value() - - debugVerboseReceive('received CDP message %o', data) - }) - } - - if (debugVerboseSend.enabled) { - const send = cri._ws.send - - cri._ws.send = (data, callback) => { - debugVerboseSend('sending CDP command %o', JSON.parse(data)) - - try { - return send.call(cri._ws, data, callback) - } catch (e: any) { - debugVerboseSend('Error sending CDP command %o %O', JSON.parse(data), e) - throw e - } - } - } - - if (debugVerboseLifecycle.enabled) { - cri._ws.addEventListener('open', (event) => { - debugVerboseLifecycle(`[OPEN] %o`, event) - }) - - cri._ws.addEventListener('close', (event) => { - debugVerboseLifecycle(`[CLOSE] %o`, event) - }) - - cri._ws.addEventListener('error', (event) => { - debugVerboseLifecycle(`[ERROR] %o`, event) - }) - } -} - type DeferredPromise = { resolve: Function, reject: Function } type CreateParams = { target: string - onAsynchronousError: Function + onAsynchronousError: (err: CypressError) => void host?: string port?: number onReconnect?: (client: CriClient) => void @@ -202,22 +108,63 @@ export class CriClient implements ICriClient { private _closed = false private _connected = false private crashed = false - private reconnection: Promise | undefined = undefined - - private cri: CDPClient | undefined + private cdpConnection: CDPConnection private constructor ( public targetId: string, - private onAsynchronousError: Function, + onAsynchronousError: (err: CypressError) => void, private host?: string, private port?: number, private onReconnect?: (client: CriClient) => void, private protocolManager?: ProtocolManagerShape, private fullyManageTabs?: boolean, private browserClient?: ICriClient, - private onReconnectAttempt?: (retryIndex: number) => void, - private onCriConnectionClosed?: () => void, - ) {} + onReconnectAttempt?: (retryIndex: number) => void, + onCriConnectionClosed?: () => void, + ) { + // refactor opportunity: + // due to listeners passed in along with connection options, the fns that instantiate this + // class should instantiate and listen to the connection directly rather than having this + // constructor create them. The execution and/or definition of these callbacks is not this + // class' business. + this.cdpConnection = new CDPConnection({ + host: this.host, + port: this.port, + target: this.targetId, + local: true, + useHostName: true, + }, { + // Only automatically reconnect if: this is the root browser cri target (no host), or cy in cy + automaticallyReconnect: !this.host && !process.env.CYPRESS_INTERNAL_E2E_TESTING_SELF, + }) + + this.cdpConnection.addConnectionEventListener('cdp-connection-reconnect-error', onAsynchronousError) + this.cdpConnection.addConnectionEventListener('cdp-connection-reconnect', this._onCdpConnectionReconnect) + + if (onCriConnectionClosed) { + this.cdpConnection.addConnectionEventListener('cdp-connection-closed', onCriConnectionClosed) + } + + if (onReconnectAttempt) { + this.cdpConnection.addConnectionEventListener('cdp-connection-reconnect-attempt', onReconnectAttempt) + } + + debug('attaching crash event listener') + + this.cdpConnection.on('Target.targetCrashed', async (event) => { + debug('crash event detected') + if (event.targetId !== this.targetId) { + return + } + + debug('crash detected') + this.crashed = true + }) + + if (!!this.host && fullyManageTabs) { + this.cdpConnection.on('Target.attachedToTarget', this._onAttachedToTarget) + } + } static async create ({ target, @@ -238,10 +185,6 @@ export class CriClient implements ICriClient { return newClient } - get ws () { - return this.cri!._ws - } - // this property is accessed in a couple different places, but should be refactored to be // private - queues are internal to this class, and should not be exposed get queue () { @@ -257,6 +200,11 @@ export class CriClient implements ICriClient { } } + // this property is accessed by browser-cri-client, to event on websocket closed. + get ws () { + return this.cdpConnection.ws + } + get closed () { return this._closed } @@ -266,82 +214,13 @@ export class CriClient implements ICriClient { } public connect = async () => { - await this.cri?.close() - debug('connecting %o', { connected: this._connected, target: this.targetId }) - /** - * TODO: /~https://github.com/cypress-io/cypress/issues/29744 - * this (`cri` / `this.cri`) symbol is referenced via closure in event listeners added in this method; this - * may prevent old instances of `CDPClient` from being garbage collected. - */ - - const cri = this.cri = await CDP({ - host: this.host, - port: this.port, - target: this.targetId, - local: true, - useHostName: true, - }) as CDPClient + await this.cdpConnection.connect() this._connected = true - debug('connected %o', { connected: this._connected, target: this.targetId, ws: this.cri._ws }) - - maybeDebugCdpMessages(this.cri) - - // Having a host set indicates that this is the child cri target, a.k.a. - // the main Cypress tab (as opposed to the root browser cri target) - const isChildTarget = !!this.host - - // don't reconnect in these circumstances - if ( - // is a child target. we only need to reconnect the root browser target - !isChildTarget - // running cypress in cypress - there are a lot of disconnects that happen - // that we don't want to reconnect on - && !process.env.CYPRESS_INTERNAL_E2E_TESTING_SELF - ) { - this.cri.on('disconnect', this._reconnect) - } - - // We're only interested in child target traffic. Browser cri traffic is - // handled in browser-cri-client.ts. The basic approach here is we attach - // to targets and enable network traffic. We must attach in a paused state - // so that we can enable network traffic before the target starts running. - if (isChildTarget) { - this.cri.on('Target.targetCrashed', async (event) => { - if (event.targetId !== this.targetId) { - return - } - - debug('crash detected') - this.crashed = true - }) - - if (this.fullyManageTabs) { - cri.on('Target.attachedToTarget', async (event) => { - try { - // Service workers get attached at the page and browser level. We only want to handle them at the browser level - // We don't track child tabs/page network traffic. 'other' targets can't have network enabled - if (event.targetInfo.type !== 'service_worker' && event.targetInfo.type !== 'page' && event.targetInfo.type !== 'other') { - await cri.send('Network.enable', this.protocolManager?.networkEnableOptions ?? DEFAULT_NETWORK_ENABLE_OPTIONS, event.sessionId) - } - - if (event.waitingForDebugger) { - await cri.send('Runtime.runIfWaitingForDebugger', undefined, event.sessionId) - } - } catch (error) { - // it's possible that the target was closed before we could enable network and continue, in that case, just ignore - debug('error attaching to target cri', error) - } - }) - - // Ideally we could use filter rather than checking the type above, but that was added relatively recently - await this.cri.send('Target.setAutoAttach', { autoAttach: true, waitForDebuggerOnStart: true, flatten: true }) - await this.cri.send('Target.setDiscoverTargets', { discover: true }) - } - } + debug('connected %o', { connected: this._connected, target: this.targetId }) } public send = async ( @@ -372,17 +251,16 @@ export class CriClient implements ICriClient { this.enableCommands.push(obj) } - if (this._connected) { + if (this._connected && this.cdpConnection) { try { - return await this.cri!.send(command, params, sessionId) + return await this.cdpConnection.send(command, params, sessionId) } catch (err) { debug('Encountered error on send %o', { command, params, sessionId, err }) // This error occurs when the browser has been left open for a long // time and/or the user's computer has been put to sleep. The // socket disconnects and we need to recreate the socket and // connection - if (!WEBSOCKET_NOT_OPEN_RE.test(err.message)) { - debug('error classified as not WEBSOCKET_NOT_OPEN_RE, rethrowing') + if (!CdpDisconnectedError.isCdpDisconnectedError(err)) { throw err } @@ -390,10 +268,8 @@ export class CriClient implements ICriClient { const p = this._enqueueCommand(command, params, sessionId) - await this._reconnect() - // if enqueued commands were wiped out from the reconnect and the socket is already closed, reject the command as it will never be run - if (this.enqueuedCommands.length === 0 && this.closed) { + if (this.enqueuedCommands.length === 0 && this.cdpConnection.terminated) { debug('connection was closed was trying to reconnect') return Promise.reject(new Error(`${command} will not run as browser CRI connection was reset`)) @@ -406,12 +282,15 @@ export class CriClient implements ICriClient { return this._enqueueCommand(command, params, sessionId) } - public on = (eventName: T, cb: (data: ProtocolMapping.Events[T][0], sessionId?: string) => void) => { + public on = (eventName: T, cb: CDPListener) => { + this.cdpConnection?.on(eventName, cb) + + // subscriptions are recorded, but this may no longer be necessary. cdp event listeners + // need only be added to the connection instance, not the (ephemeral) underlying + // CDP.Client instances this.subscriptions.push({ eventName, cb }) debug('registering CDP on event %o', { eventName }) - this.cri!.on(eventName, cb) - if (eventName.startsWith('Network.')) { this.browserClient?.on(eventName, cb) } @@ -422,7 +301,7 @@ export class CriClient implements ICriClient { return sub.eventName === eventName && sub.cb === cb }), 1) - this.cri!.off(eventName, cb) + this.cdpConnection!.off(eventName, cb) // This ensures that we are notified about the browser's network events that have been registered (e.g. service workers) // Long term we should use flat mode entirely across all of chrome remote interface if (eventName.startsWith('Network.')) { @@ -432,8 +311,8 @@ export class CriClient implements ICriClient { public close = async () => { debug('closing') - if (this._closed) { - debug('not closing, cri client is already closed %o', { closed: this._closed, target: this.targetId }) + if (this._closed || this.cdpConnection?.terminated) { + debug('not closing, cri client is already closed %o', { closed: this._closed, target: this.targetId, connection: this.cdpConnection }) return } @@ -443,15 +322,40 @@ export class CriClient implements ICriClient { this._closed = true try { - await this.cri?.close() + await this.cdpConnection?.disconnect() + debug('closed cri client %o', { closed: this._closed, target: this.targetId }) } catch (e) { debug('error closing cri client targeting %s: %o', this.targetId, e) - } finally { - debug('closed cri client %o', { closed: this._closed, target: this.targetId }) - if (this.onCriConnectionClosed) { - this.onCriConnectionClosed() + } + } + + private _onAttachedToTarget = async (event: ProtocolMapping.Events['Target.attachedToTarget'][0]) => { + // We're only interested in child target traffic. Browser cri traffic is + // handled in browser-cri-client.ts. The basic approach here is we attach + // to targets and enable network traffic. We must attach in a paused state + // so that we can enable network traffic before the target starts running. + if (!this.fullyManageTabs || !this.host) { + return + } + + try { + // Service workers get attached at the page and browser level. We only want to handle them at the browser level + // We don't track child tabs/page network traffic. 'other' targets can't have network enabled + if (event.targetInfo.type !== 'service_worker' && event.targetInfo.type !== 'page' && event.targetInfo.type !== 'other') { + await this.cdpConnection.send('Network.enable', this.protocolManager?.networkEnableOptions ?? DEFAULT_NETWORK_ENABLE_OPTIONS, event.sessionId) + } + + if (event.waitingForDebugger) { + await this.cdpConnection.send('Runtime.runIfWaitingForDebugger', undefined, event.sessionId) } + } catch (error) { + // it's possible that the target was closed before we could enable network and continue, in that case, just ignore + debug('error attaching to target cri', error) } + + // Ideally we could use filter rather than checking the type above, but that was added relatively recently + await this.cdpConnection.send('Target.setAutoAttach', { autoAttach: true, waitForDebuggerOnStart: true, flatten: true }) + await this.cdpConnection.send('Target.setDiscoverTargets', { discover: true }) } private _enqueueCommand ( @@ -462,107 +366,26 @@ export class CriClient implements ICriClient { return this._commandQueue.add(command, params, sessionId) } - private _isConnectionError (error: Error) { - return WEBSOCKET_NOT_OPEN_RE.test(error.message) - } - - private _reconnect = async () => { - debug('preparing to reconnect') - if (this.reconnection) { - debug('not reconnecting as there is an active reconnection attempt') - - return this.reconnection - } - - this._connected = false - - if (this._closed) { - debug('Target %s disconnected, not reconnecting because client is closed.', this.targetId) - this._commandQueue.clear() - - return - } - - let attempt = 1 - - try { - this.reconnection = asyncRetry(() => { - if (this._closed) { - throw new ConnectionClosedError('Reconnection halted due to a closed client.') - } - - this.onReconnectAttempt?.(attempt) - attempt++ - - return this.connect() - }, { - maxAttempts: 20, - retryDelay: () => 100, - shouldRetry: (err) => { - debug('error while reconnecting to Target %s: %o', this.targetId, err) - if (err && ConnectionClosedError.isConnectionClosedError(err)) { - return false - } - - debug('Retying reconnection attempt') - - return true - }, - })() - - await this.reconnection - this.reconnection = undefined - debug('reconnected') - } catch (err) { - debug('error(s) on reconnecting: ', err) - const significantError: Error = err.errors ? (err as AggregateError).errors[err.errors.length - 1] : err - - const retryHaltedDueToClosed = ConnectionClosedError.isConnectionClosedError(err) || - (err as AggregateError)?.errors?.find((predicate) => ConnectionClosedError.isConnectionClosedError(predicate)) - - if (!retryHaltedDueToClosed) { - const cdpError = errors.get('CDP_COULD_NOT_RECONNECT', significantError) - - cdpError.isFatalApiErr = true - this.reconnection = undefined - this._commandQueue.clear() - this.onAsynchronousError(cdpError) - } - - // do not re-throw; error handling is done via onAsynchronousError - return - } - + private _onCdpConnectionReconnect = async () => { try { await this._restoreState() await this._drainCommandQueue() await this.protocolManager?.cdpReconnect() - } catch (e) { - if (this._isConnectionError(e)) { - return this._reconnect() - } - - throw e - } - // previous timing of this had it happening before subscriptions/enablements were restored, - // and before any enqueued commands were sent. This made testing problematic. Changing the - // timing may have implications for browsers that wish to update frame tree - that process - // will now be kicked off after state restoration & pending commands, rather then before. - // This warrants extra scrutiny in tests. (convert to PR comment) - if (this.onReconnect) { - this.onReconnect(this) + try { + if (this.onReconnect) { + await this.onReconnect(this) + } + } catch (e) { + debug('uncaught error in CriClient reconnect callback: ', e) + } + } catch (e) { + debug('error re-establishing state on reconnection: ', e) } } private async _restoreState () { - debug('resubscribing to %d subscriptions', this.subscriptions.length) - - this.subscriptions.forEach((sub) => { - this.cri?.on(sub.eventName, sub.cb as any) - }) - // '*.enable' commands need to be resent on reconnect or any events in // that namespace will no longer be received debug('re-enabling %d enablements', this.enableCommands.length) @@ -572,12 +395,15 @@ export class CriClient implements ICriClient { const inFlightCommand = this._commandQueue.extract({ command, params, sessionId }) try { - const response = await this.cri?.send(command, params, sessionId) + const response = await this.cdpConnection.send(command, params, sessionId) inFlightCommand?.deferred.resolve(response) } catch (err) { debug('error re-enabling %s: ', command, err) - if (this._isConnectionError(err)) { + if (CdpDisconnectedError.isCdpDisconnectedError(err)) { + // below comment is no longer accurate - ephemeral connection is wrapped by + // CDPConnection class + // Connection errors are thrown here so that a reconnection attempt // can be made. throw err @@ -600,14 +426,17 @@ export class CriClient implements ICriClient { try { debug('sending enqueued command %s', enqueued.command) - const response = await this.cri!.send(enqueued.command, enqueued.params, enqueued.sessionId) + const response = await this.cdpConnection.send(enqueued.command, enqueued.params, enqueued.sessionId) debug('sent command, received ', { response }) enqueued.deferred.resolve(response) debug('resolved enqueued promise') } catch (e) { debug('enqueued command %s failed:', enqueued.command, e) - if (this._isConnectionError(e)) { + if (CdpDisconnectedError.isCdpDisconnectedError(e)) { + // below comment is no longer accurate - ephemeral connection is wrapped by + // CDPConnection class + // similar to restoring state, connection errors are re-thrown so that // the connection can be restored. The command is queued for re-delivery // upon reconnect. diff --git a/packages/server/lib/browsers/remote-interface/debug-cdp-connection.ts b/packages/server/lib/browsers/remote-interface/debug-cdp-connection.ts new file mode 100644 index 000000000000..0e0b5188807d --- /dev/null +++ b/packages/server/lib/browsers/remote-interface/debug-cdp-connection.ts @@ -0,0 +1,81 @@ +import Debug from 'debug' +import _ from 'lodash' +import type CDP from 'chrome-remote-interface' +import type EventEmitter from 'events' +import type WebSocket from 'ws' + +export interface DebuggableCDPClient extends CDP.Client { + off: EventEmitter['off'] + // ws is defined as optional here, because it is not a declared public property of + // CDP.Client - it may be removed in future minor/patch versions + _ws?: WebSocket +} + +export const debugCdpConnection = (namespace: string, cri: DebuggableCDPClient) => { + // debug using cypress-verbose:server:browsers:cri-client:send:* + const debugVerboseSend = Debug(`${namespace}:send:[-->]`) + // debug using cypress-verbose:server:browsers:cri-client:recv:* + const debugVerboseReceive = Debug(`${namespace}:recv:[<--]`) + // debug using cypress-verbose:server:browsers:cri-client:err:* + const debugVerboseLifecycle = Debug(`${namespace}:ws`) + + if (debugVerboseReceive.enabled) { + cri._ws?.prependListener('message', (data) => { + data = _ + .chain(JSON.parse(data)) + .tap((data) => { + ([ + 'params.data', // screencast frame data + 'result.data', // screenshot data + ]).forEach((truncatablePath) => { + const str = _.get(data, truncatablePath) + + if (!_.isString(str)) { + return + } + + _.set(data, truncatablePath, _.truncate(str, { + length: 100, + omission: `... [truncated string of total bytes: ${str.length}]`, + })) + }) + + return data + }) + .value() + + debugVerboseReceive('received CDP message %o', data) + }) + } + + if (debugVerboseSend.enabled) { + if (cri._ws) { + const send = cri._ws?.send + + cri._ws.send = (data, callback) => { + debugVerboseSend('sending CDP command %o', JSON.parse(data)) + + try { + return send.call(cri._ws, data, callback) + } catch (e: any) { + debugVerboseSend('Error sending CDP command %o %O', JSON.parse(data), e) + throw e + } + } + } + } + + if (debugVerboseLifecycle.enabled) { + cri._ws?.addEventListener('open', (event) => { + debugVerboseLifecycle(`[OPEN] %o`, event) + }) + + cri._ws?.addEventListener('close', (event) => { + debugVerboseLifecycle(`[CLOSE] %o`, event) + }) + + cri._ws?.addEventListener('error', (event) => { + debugVerboseLifecycle(`[ERROR] %o`, event) + }) + } +} diff --git a/packages/server/lib/browsers/remote-interface/errors.ts b/packages/server/lib/browsers/remote-interface/errors.ts new file mode 100644 index 000000000000..ae4f70ab5577 --- /dev/null +++ b/packages/server/lib/browsers/remote-interface/errors.ts @@ -0,0 +1,41 @@ +const Kinds = Object.freeze({ + CDP_CRASHED: 'cdp_crashed', + CDP_DISCONNECTED: 'cdp_disconnected', + CDP_TERMINATED: 'cdp_terminated', +}) + +type CdpErrorKind = typeof Kinds[keyof typeof Kinds] + +type MaybeCdpError = Error & { kind?: CdpErrorKind } + +export class CdpCrashedError extends Error { + public readonly kind = Kinds.CDP_CRASHED + + public static isCdpCrashedError (error: MaybeCdpError): error is CdpCrashedError { + return error.kind === Kinds.CDP_CRASHED + } +} + +export class CdpDisconnectedError extends Error { + public readonly kind = Kinds.CDP_DISCONNECTED + + constructor (message: string, public readonly originalError?: Error) { + super(message) + } + + public static isCdpDisconnectedError (error: MaybeCdpError): error is CdpDisconnectedError { + return error.kind === Kinds.CDP_DISCONNECTED + } +} + +export class CdpTerminatedError extends Error { + public readonly kind = Kinds.CDP_TERMINATED + + constructor (message: string, public readonly originalError?: Error) { + super(message) + } + + public static isCdpTerminatedError (error: MaybeCdpError): error is CdpTerminatedError { + return error.kind === Kinds.CDP_TERMINATED + } +} diff --git a/packages/server/lib/browsers/utils.ts b/packages/server/lib/browsers/utils.ts index 9cc77cf98dc1..d8ad842c0456 100644 --- a/packages/server/lib/browsers/utils.ts +++ b/packages/server/lib/browsers/utils.ts @@ -8,7 +8,7 @@ import { getError } from '@packages/errors' import * as launcher from '@packages/launcher' import type { Automation } from '../automation' import type { Browser } from './types' -import type { CriClient } from './cri-client' +import type { CriClient } from './remote-interface/cri-client' declare global { interface Window { diff --git a/packages/server/test/integration/cypress_spec.js b/packages/server/test/integration/cypress_spec.js index 756d31947c66..37977eb9092d 100644 --- a/packages/server/test/integration/cypress_spec.js +++ b/packages/server/test/integration/cypress_spec.js @@ -44,7 +44,7 @@ const appData = require(`../../lib/util/app_data`) const electronApp = require('../../lib/util/electron-app') const savedState = require(`../../lib/saved_state`) const { getCtx, clearCtx, setCtx, makeDataContext } = require(`../../lib/makeDataContext`) -const { BrowserCriClient } = require(`../../lib/browsers/browser-cri-client`) +const { BrowserCriClient } = require(`../../lib/browsers/remote-interface/browser-cri-client`) const { cloudRecommendationMessage } = require('../../lib/util/print-run') const processVersions = process.versions diff --git a/packages/server/test/unit/browsers/browser-cri-client_spec.ts b/packages/server/test/unit/browsers/browser-cri-client_spec.ts index 510e553dfb02..cc8dc7c6cd22 100644 --- a/packages/server/test/unit/browsers/browser-cri-client_spec.ts +++ b/packages/server/test/unit/browsers/browser-cri-client_spec.ts @@ -1,5 +1,5 @@ -import { BrowserCriClient } from '../../../lib/browsers/browser-cri-client' -import { CriClient } from '../../../lib/browsers/cri-client' +import { BrowserCriClient } from '../../../lib/browsers/remote-interface/browser-cri-client' +import { CriClient } from '../../../lib/browsers/remote-interface/cri-client' import { expect, proxyquire, sinon } from '../../spec_helper' import * as protocol from '../../../lib/browsers/protocol' import { stripAnsi } from '@packages/errors' @@ -55,7 +55,7 @@ describe('lib/browsers/browser-cri-client', function () { close, }) - browserCriClient = proxyquire('../lib/browsers/browser-cri-client', { + browserCriClient = proxyquire('../lib/browsers/remote-interface/browser-cri-client', { 'chrome-remote-interface': criImport, }) diff --git a/packages/server/test/unit/browsers/chrome_spec.js b/packages/server/test/unit/browsers/chrome_spec.js index b09f4de1b3bf..f930cd89a6f6 100644 --- a/packages/server/test/unit/browsers/chrome_spec.js +++ b/packages/server/test/unit/browsers/chrome_spec.js @@ -9,7 +9,7 @@ const plugins = require(`../../../lib/plugins`) const utils = require(`../../../lib/browsers/utils`) const chrome = require(`../../../lib/browsers/chrome`) const { fs } = require(`../../../lib/util/fs`) -const { BrowserCriClient } = require('../../../lib/browsers/browser-cri-client') +const { BrowserCriClient } = require('../../../lib/browsers/remote-interface/browser-cri-client') const { expect } = require('chai') const openOpts = { diff --git a/packages/server/test/unit/browsers/cri-client_spec.ts b/packages/server/test/unit/browsers/cri-client_spec.ts index f57291a80606..91b2bad9ac39 100644 --- a/packages/server/test/unit/browsers/cri-client_spec.ts +++ b/packages/server/test/unit/browsers/cri-client_spec.ts @@ -1,7 +1,8 @@ +import type ProtocolMapping from 'devtools-protocol/types/protocol-mapping' import EventEmitter from 'events' import { ProtocolManagerShape } from '@packages/types' -import type { CriClient } from '../../../lib/browsers/cri-client' - +import type { CriClient } from '../../../lib/browsers/remote-interface/cri-client' +import pDefer from 'p-defer' const { expect, proxyquire, sinon } = require('../../spec_helper') const DEBUGGER_URL = 'http://foo' @@ -11,24 +12,40 @@ const PORT = 50505 describe('lib/browsers/cri-client', function () { let send: sinon.SinonStub let on: sinon.SinonStub + let off: sinon.SinonStub + let criImport: sinon.SinonStub & { New: sinon.SinonStub } let criStub: { send: typeof send on: typeof on + off: typeof off close: sinon.SinonStub _notifier: EventEmitter } let onError: sinon.SinonStub - let getClient: (options?: { host?: string, fullyManageTabs?: boolean, protocolManager?: ProtocolManagerShape }) => ReturnType + let onReconnect: sinon.SinonStub + + let getClient: (options?: { host?: string, fullyManageTabs?: boolean, protocolManager?: ProtocolManagerShape }) => ReturnType + + const fireCDPEvent = (method: T, params: Partial, sessionId?: string) => { + criStub.on.withArgs('event').args[0][1]({ + method, + params, + sessionId, + }) + } beforeEach(function () { send = sinon.stub() onError = sinon.stub() + onReconnect = sinon.stub() on = sinon.stub() + off = sinon.stub() criStub = { on, + off, send, close: sinon.stub().resolves(), _notifier: new EventEmitter(), @@ -43,12 +60,16 @@ describe('lib/browsers/cri-client', function () { criImport.New = sinon.stub().withArgs({ host: HOST, port: PORT, url: 'about:blank' }).resolves({ webSocketDebuggerUrl: 'http://web/socket/url' }) - const { CriClient } = proxyquire('../lib/browsers/cri-client', { + const { CDPConnection } = proxyquire('../lib/browsers/remote-interface/cdp-connection', { 'chrome-remote-interface': criImport, }) + const { CriClient } = proxyquire('../lib/browsers/remote-interface/cri-client', { + './cdp-connection': { CDPConnection }, + }) + getClient = ({ host, fullyManageTabs, protocolManager } = {}): Promise => { - return CriClient.create({ target: DEBUGGER_URL, host, onAsynchronousError: onError, fullyManageTabs, protocolManager }) + return CriClient.create({ target: DEBUGGER_URL, host, onAsynchronousError: onError, fullyManageTabs, protocolManager, onReconnect }) } }) @@ -82,7 +103,8 @@ describe('lib/browsers/cri-client', function () { const command = 'DOM.getDocument' const client = await getClient({ host: '127.0.0.1', fullyManageTabs: true }) - await criStub.on.withArgs('Target.targetCrashed').args[0][1]({ targetId: DEBUGGER_URL }) + fireCDPEvent('Target.targetCrashed', { targetId: DEBUGGER_URL }) + await expect(client.send(command, { depth: -1 })).to.be.rejectedWith(`${command} will not run as the target browser or tab CRI connection has crashed`) }) @@ -91,7 +113,7 @@ describe('lib/browsers/cri-client', function () { await getClient({ host: '127.0.0.1', fullyManageTabs: true }) // This would throw if the error was not caught - await criStub.on.withArgs('Target.attachedToTarget').args[0][1]({ targetInfo: { type: 'worker' } }) + await fireCDPEvent('Target.attachedToTarget', { targetInfo: { type: 'worker', targetId: DEBUGGER_URL, title: '', url: 'https://some_url', attached: true, canAccessOpener: true } }) }) context('retries', () => { @@ -109,8 +131,10 @@ describe('lib/browsers/cri-client', function () { const client = await getClient() - await client.send('DOM.getDocument', { depth: -1 }) + const p = client.send('DOM.getDocument', { depth: -1 }) + await criStub.on.withArgs('disconnect').args[0][1]() + await p expect(send).to.be.calledTwice }) @@ -123,7 +147,11 @@ describe('lib/browsers/cri-client', function () { const client = await getClient() - await client.send('DOM.getDocument', { depth: -1 }) + const getDocumentPromise = client.send('DOM.getDocument', { depth: -1 }) + + await criStub.on.withArgs('disconnect').args[0][1]() + await criStub.on.withArgs('disconnect').args[0][1]() + await getDocumentPromise expect(send).to.have.callCount(3) }) }) @@ -183,6 +211,11 @@ describe('lib/browsers/cri-client', function () { // @ts-ignore await criStub.on.withArgs('disconnect').args[0][1]() + const reconnection = pDefer() + + onReconnect.callsFake(() => reconnection.resolve()) + await reconnection.promise + expect(criStub.send).to.be.calledTwice expect(criStub.send).to.be.calledWith('Page.enable') expect(criStub.send).to.be.calledWith('Network.enable') diff --git a/packages/server/test/unit/browsers/electron_spec.js b/packages/server/test/unit/browsers/electron_spec.js index 2551326d78c5..72f92d489e39 100644 --- a/packages/server/test/unit/browsers/electron_spec.js +++ b/packages/server/test/unit/browsers/electron_spec.js @@ -11,7 +11,7 @@ const Windows = require(`../../../lib/gui/windows`) const electron = require(`../../../lib/browsers/electron`) const savedState = require(`../../../lib/saved_state`) const { Automation } = require(`../../../lib/automation`) -const { BrowserCriClient } = require('../../../lib/browsers/browser-cri-client') +const { BrowserCriClient } = require('../../../lib/browsers/remote-interface/browser-cri-client') const electronApp = require('../../../lib/util/electron-app') const utils = require('../../../lib/browsers/utils') diff --git a/packages/server/test/unit/browsers/firefox_spec.ts b/packages/server/test/unit/browsers/firefox_spec.ts index cbca36a5748f..d1b368fba89c 100644 --- a/packages/server/test/unit/browsers/firefox_spec.ts +++ b/packages/server/test/unit/browsers/firefox_spec.ts @@ -11,7 +11,7 @@ import Foxdriver from '@benmalka/foxdriver' import * as firefox from '../../../lib/browsers/firefox' import firefoxUtil from '../../../lib/browsers/firefox-util' import { CdpAutomation } from '../../../lib/browsers/cdp_automation' -import { BrowserCriClient } from '../../../lib/browsers/browser-cri-client' +import { BrowserCriClient } from '../../../lib/browsers/remote-interface/browser-cri-client' import { ICriClient } from '../../../lib/browsers/cri-client' const path = require('path') From ab3aaceb306b3069d78621c97ceea8c4005a00e1 Mon Sep 17 00:00:00 2001 From: Cacie Prins Date: Tue, 6 Aug 2024 11:20:09 -0400 Subject: [PATCH 02/25] changelog --- cli/CHANGELOG.md | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/cli/CHANGELOG.md b/cli/CHANGELOG.md index 8bf842fe4617..ae9dd523d8b0 100644 --- a/cli/CHANGELOG.md +++ b/cli/CHANGELOG.md @@ -14,7 +14,8 @@ _Released 7/31/2024_ **Performance:** - Fixed a memory leak with command logs with Test Replay enabled. Addressed in [#29939](/~https://github.com/cypress-io/cypress/pull/29939). -- Improved performance of `reduce` in a method within our proxy. Addressed in [#29887](/~https://github.com/cypress-io/cypress/pull/29887). +- Improved performance of `reduce` in a method within our proxy. Addressed in [#29887](/~https://github.com/cypress-io/cypress/pull/29887). +- Fixed a potential memory leak when re-connecting to an unintentionally disconnected CDP connection. Addressed in [#29988](/~https://github.com/cypress-io/cypress/pull/29988) **Bugfixes:** From bb85733777941bf33b44a85ceae9a7a8c9481eb9 Mon Sep 17 00:00:00 2001 From: Cacie Prins Date: Tue, 6 Aug 2024 11:21:15 -0400 Subject: [PATCH 03/25] clean up event listeners before reconnecting --- .../lib/browsers/remote-interface/cdp-connection.ts | 10 ++++++++++ .../server/lib/browsers/remote-interface/cri-client.ts | 6 +++--- 2 files changed, 13 insertions(+), 3 deletions(-) diff --git a/packages/server/lib/browsers/remote-interface/cdp-connection.ts b/packages/server/lib/browsers/remote-interface/cdp-connection.ts index bdfb76ed7313..063578fa0f99 100644 --- a/packages/server/lib/browsers/remote-interface/cdp-connection.ts +++ b/packages/server/lib/browsers/remote-interface/cdp-connection.ts @@ -149,6 +149,16 @@ export class CDPConnection { return this._reconnection } + if (this._connection) { + try { + await this._gracefullyDisconnect() + } catch (e) { + debug('Error cleaning up existing CDP connection before creating a new connection: ', e) + } finally { + this._connection = undefined + } + } + let attempt = 0 this._reconnection = asyncRetry(async () => { diff --git a/packages/server/lib/browsers/remote-interface/cri-client.ts b/packages/server/lib/browsers/remote-interface/cri-client.ts index fd282a71b632..3f33dbcf71b8 100644 --- a/packages/server/lib/browsers/remote-interface/cri-client.ts +++ b/packages/server/lib/browsers/remote-interface/cri-client.ts @@ -99,6 +99,9 @@ type CreateParams = { } export class CriClient implements ICriClient { + // subscriptions are recorded, but this may no longer be necessary. cdp event listeners + // need only be added to the connection instance, not the (ephemeral) underlying + // CDP.Client instances private subscriptions: Subscription[] = [] private enableCommands: EnableCommand[] = [] private enqueuedCommands: EnqueuedCommand[] = [] @@ -285,9 +288,6 @@ export class CriClient implements ICriClient { public on = (eventName: T, cb: CDPListener) => { this.cdpConnection?.on(eventName, cb) - // subscriptions are recorded, but this may no longer be necessary. cdp event listeners - // need only be added to the connection instance, not the (ephemeral) underlying - // CDP.Client instances this.subscriptions.push({ eventName, cb }) debug('registering CDP on event %o', { eventName }) From 326fcdca40bfce533c051d652f79309a5f9eab50 Mon Sep 17 00:00:00 2001 From: Cacie Prins Date: Tue, 6 Aug 2024 11:34:49 -0400 Subject: [PATCH 04/25] changelog --- cli/CHANGELOG.md | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/cli/CHANGELOG.md b/cli/CHANGELOG.md index ae9dd523d8b0..859fd1572a3a 100644 --- a/cli/CHANGELOG.md +++ b/cli/CHANGELOG.md @@ -3,6 +3,10 @@ _Released 8/13/2024 (PENDING)_ +**Performance:** + +- Fixed a potential memory leak when re-connecting to an unintentionally disconnected CDP connection. Addressed in [#29988](/~https://github.com/cypress-io/cypress/pull/29988) + **Misc:** - Updated `cypress open` hints displayed after Cypress binary install. Addresses [#29935](/~https://github.com/cypress-io/cypress/issues/29935). @@ -15,7 +19,6 @@ _Released 7/31/2024_ - Fixed a memory leak with command logs with Test Replay enabled. Addressed in [#29939](/~https://github.com/cypress-io/cypress/pull/29939). - Improved performance of `reduce` in a method within our proxy. Addressed in [#29887](/~https://github.com/cypress-io/cypress/pull/29887). -- Fixed a potential memory leak when re-connecting to an unintentionally disconnected CDP connection. Addressed in [#29988](/~https://github.com/cypress-io/cypress/pull/29988) **Bugfixes:** From 2cf27b458bf2f556f076ab1a3d7842d9c5ca9638 Mon Sep 17 00:00:00 2001 From: Cacie Prins Date: Tue, 6 Aug 2024 11:41:48 -0400 Subject: [PATCH 05/25] changelog --- cli/CHANGELOG.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/cli/CHANGELOG.md b/cli/CHANGELOG.md index 859fd1572a3a..b39e77f84d29 100644 --- a/cli/CHANGELOG.md +++ b/cli/CHANGELOG.md @@ -18,7 +18,7 @@ _Released 7/31/2024_ **Performance:** - Fixed a memory leak with command logs with Test Replay enabled. Addressed in [#29939](/~https://github.com/cypress-io/cypress/pull/29939). -- Improved performance of `reduce` in a method within our proxy. Addressed in [#29887](/~https://github.com/cypress-io/cypress/pull/29887). +- Improved performance of `reduce` in a method within our proxy. Addressed in [#29887](/~https://github.com/cypress-io/cypress/pull/29887). **Bugfixes:** From f441d36e380f150294913afcef6f2a1a59f17c90 Mon Sep 17 00:00:00 2001 From: Cacie Prins Date: Tue, 6 Aug 2024 11:49:02 -0400 Subject: [PATCH 06/25] changelog --- cli/CHANGELOG.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/cli/CHANGELOG.md b/cli/CHANGELOG.md index b39e77f84d29..c7b3fc8c72d8 100644 --- a/cli/CHANGELOG.md +++ b/cli/CHANGELOG.md @@ -5,7 +5,7 @@ _Released 8/13/2024 (PENDING)_ **Performance:** -- Fixed a potential memory leak when re-connecting to an unintentionally disconnected CDP connection. Addressed in [#29988](/~https://github.com/cypress-io/cypress/pull/29988) +- Fixed a potential memory leak when re-connecting to an unintentionally disconnected CDP connection. Fixes [#29744](/~https://github.com/cypress-io/cypress/issues/29744). Addressed in [#29988](/~https://github.com/cypress-io/cypress/pull/29988) **Misc:** From cf81eb9067f7ac17d71ad4ee9a2f303d9256727f Mon Sep 17 00:00:00 2001 From: Cacie Prins Date: Tue, 6 Aug 2024 16:32:31 -0400 Subject: [PATCH 07/25] unit tests --- .../remote-interface/cdp-connection.ts | 48 ++- .../browsers/remote-interface/cri-client.ts | 8 +- .../lib/browsers/remote-interface/errors.ts | 25 +- packages/server/test/integration/cdp_spec.ts | 2 +- .../unit/browsers/cdp-command-queue_spec.ts | 2 +- .../test/unit/browsers/cri-client_spec.ts | 10 +- .../server/test/unit/browsers/firefox_spec.ts | 2 +- .../remote-interface/cdp-connection_spec.ts | 351 ++++++++++++++++++ 8 files changed, 412 insertions(+), 36 deletions(-) create mode 100644 packages/server/test/unit/browsers/remote-interface/cdp-connection_spec.ts diff --git a/packages/server/lib/browsers/remote-interface/cdp-connection.ts b/packages/server/lib/browsers/remote-interface/cdp-connection.ts index 063578fa0f99..4123c7ca58bf 100644 --- a/packages/server/lib/browsers/remote-interface/cdp-connection.ts +++ b/packages/server/lib/browsers/remote-interface/cdp-connection.ts @@ -5,7 +5,7 @@ import CDP from 'chrome-remote-interface' import type { CypressError } from '@packages/errors' import { debugCdpConnection, DebuggableCDPClient } from './debug-cdp-connection' import type { CdpEvent, CdpCommand } from '../cdp_automation' -import { CdpDisconnectedError, CdpTerminatedError } from './errors' +import { CDPDisconnectedError, CDPTerminatedError, CDPAlreadyConnectedError } from './errors' import { asyncRetry } from '../../util/async_retry' import * as errors from '../../errors' import type WebSocket from 'ws' @@ -16,7 +16,7 @@ export type CDPListener = (params: Proto // CDPClient extends EventEmitter, but does not export that type information from its // definitelytyped module -type CdpClient = Exclude & CDP.Client +export type CdpClient = Exclude & CDP.Client /** * There are three error messages we can encounter which should not be re-thrown, but @@ -30,7 +30,7 @@ const isWebsocketClosedErrorMessage = (message: string) => { return /^WebSocket (?:connection closed|is (?:not open|already in CLOSING or CLOSED state))/.test(message) } -type CDPConnectionOptions = { +export type CDPConnectionOptions = { automaticallyReconnect: boolean } @@ -40,7 +40,7 @@ type CDPConnectionEventListeners = { 'cdp-connection-closed': () => void 'cdp-connection-reconnect-attempt': (attemptNumber: number) => void } -type CDPConnectionEvent = keyof CDPConnectionEventListeners +export type CDPConnectionEvent = keyof CDPConnectionEventListeners type CDPConnectionEventListener = CDPConnectionEventListeners[T] @@ -51,8 +51,8 @@ export class CDPConnection { private _terminated: boolean = false private _reconnection: Promise | undefined - constructor (private readonly _options: CDP.Options, connectionoptions: CDPConnectionOptions) { - this._autoReconnect = connectionoptions.automaticallyReconnect + constructor (private readonly _options: CDP.Options, connectionOptions: CDPConnectionOptions) { + this._autoReconnect = connectionOptions.automaticallyReconnect } get terminated () { @@ -69,22 +69,23 @@ export class CDPConnection { this._emitter.on(event, callback) } addConnectionEventListener (event: T, callback: CDPConnectionEventListener) { + debug('adding connection event listener for ', event) this._emitter.on(event, callback) } off (event: T, callback: CDPListener) { this._emitter.off(event, callback) } removeConnectionEventListener (event: T, callback: CDPConnectionEventListener) { - this._emitter.on(event, callback) + this._emitter.off(event, callback) } async connect (): Promise { if (this._terminated) { - throw new CdpTerminatedError(`Cannot reconnect to CDP. Client target ${this._options.target} has been terminated.`) + throw new CDPTerminatedError(`Cannot connect to CDP. Client target ${this._options.target} has been terminated.`) } if (this._connection) { - await this._gracefullyDisconnect() + throw new CDPAlreadyConnectedError(`Cannot connect to CDP. Client target ${this._options.target} is already connected. Did you disconnect first?`) } this._connection = await CDP(this._options) as CdpClient @@ -105,9 +106,11 @@ export class CDPConnection { this._terminated = true - await this._gracefullyDisconnect() + if (this._connection) { + await this._gracefullyDisconnect() - this._emitter.emit('cdp-connection-closed') + this._emitter.emit('cdp-connection-closed') + } } private _gracefullyDisconnect = async () => { @@ -122,9 +125,12 @@ export class CDPConnection { data?: ProtocolMapping.Commands[T]['paramsType'][0], sessionId?: string, ): Promise { - // TODO: what if in the middle of a reconnection attempt? - if (!this._connection || this._terminated) { - throw new CdpDisconnectedError(`${command} will not run as the CRI connection is not available`) + if (this.terminated) { + throw new CDPDisconnectedError(`${command} will not run as the CRI connection to Target ${this._options.target} has been terminated.`) + } + + if (!this._connection) { + throw new CDPDisconnectedError(`${command} will not run as the CRI connection to Target ${this._options.target} has not been established.`) } try { @@ -133,7 +139,7 @@ export class CDPConnection { // Clients may wish to determine if the command should be enqueued // should enqueue logic live in this class tho?? if (isWebsocketClosedErrorMessage(e.message)) { - throw new CdpDisconnectedError(`${command} failed due to the websocket being disconnected.`, e) + throw new CDPDisconnectedError(`${command} failed due to the websocket being disconnected.`, e) } throw e @@ -164,8 +170,11 @@ export class CDPConnection { this._reconnection = asyncRetry(async () => { attempt++ + debug('Reconnection attempt %d for Target %s', attempt, this._options.target) + if (this._terminated) { - throw new CdpTerminatedError(`Cannot reconnect to CDP. Client target ${this._options.target} has been terminated.`) + debug('Not reconnecting, connection to %s has been terminated', this._options.target) + throw new CDPTerminatedError(`Cannot reconnect to CDP. Client target ${this._options.target} has been terminated.`) } this._emitter.emit('cdp-connection-reconnect-attempt', attempt) @@ -177,7 +186,7 @@ export class CDPConnection { return 100 }, shouldRetry (err) { - return !(err && CdpTerminatedError.isCdpTerminatedError(err)) + return !(err && CDPTerminatedError.isCDPTerminatedError(err)) }, })() @@ -188,8 +197,8 @@ export class CDPConnection { debug('error(s) on reconnecting: ', err) const significantError: Error = err.errors ? (err as AggregateError).errors[err.errors.length - 1] : err - const retryHaltedDueToClosed = CdpTerminatedError.isCdpTerminatedError(err) || - (err as AggregateError)?.errors?.find((predicate) => CdpTerminatedError.isCdpTerminatedError(predicate)) + const retryHaltedDueToClosed = CDPTerminatedError.isCDPTerminatedError(err) || + (err as AggregateError)?.errors?.find((predicate) => CDPTerminatedError.isCDPTerminatedError(predicate)) if (!retryHaltedDueToClosed) { const cdpError = errors.get('CDP_COULD_NOT_RECONNECT', significantError) @@ -203,6 +212,7 @@ export class CDPConnection { } private _broadcastEvent = ({ method, params, sessionId }: { method: CdpEvent, params: Record, sessionId?: string }) => { + debug('rebroadcasting event', method, params, sessionId) this._emitter.emit(method, params, sessionId) } } diff --git a/packages/server/lib/browsers/remote-interface/cri-client.ts b/packages/server/lib/browsers/remote-interface/cri-client.ts index 3f33dbcf71b8..490b625d7f3e 100644 --- a/packages/server/lib/browsers/remote-interface/cri-client.ts +++ b/packages/server/lib/browsers/remote-interface/cri-client.ts @@ -5,7 +5,7 @@ import type ProtocolMapping from 'devtools-protocol/types/protocol-mapping' import type WebSocket from 'ws' import type { CypressError } from '@packages/errors' import type { SendDebuggerCommand, OnFn, OffFn, CdpCommand, CdpEvent } from '../cdp_automation' -import { CdpDisconnectedError } from './errors' +import { CDPDisconnectedError } from './errors' import type { ProtocolManagerShape } from '@packages/types' const debug = debugModule('cypress:server:browsers:cri-client') @@ -263,7 +263,7 @@ export class CriClient implements ICriClient { // time and/or the user's computer has been put to sleep. The // socket disconnects and we need to recreate the socket and // connection - if (!CdpDisconnectedError.isCdpDisconnectedError(err)) { + if (!CDPDisconnectedError.isCDPDisconnectedError(err)) { throw err } @@ -400,7 +400,7 @@ export class CriClient implements ICriClient { inFlightCommand?.deferred.resolve(response) } catch (err) { debug('error re-enabling %s: ', command, err) - if (CdpDisconnectedError.isCdpDisconnectedError(err)) { + if (CDPDisconnectedError.isCDPDisconnectedError(err)) { // below comment is no longer accurate - ephemeral connection is wrapped by // CDPConnection class @@ -433,7 +433,7 @@ export class CriClient implements ICriClient { debug('resolved enqueued promise') } catch (e) { debug('enqueued command %s failed:', enqueued.command, e) - if (CdpDisconnectedError.isCdpDisconnectedError(e)) { + if (CDPDisconnectedError.isCDPDisconnectedError(e)) { // below comment is no longer accurate - ephemeral connection is wrapped by // CDPConnection class diff --git a/packages/server/lib/browsers/remote-interface/errors.ts b/packages/server/lib/browsers/remote-interface/errors.ts index ae4f70ab5577..2c05550f7321 100644 --- a/packages/server/lib/browsers/remote-interface/errors.ts +++ b/packages/server/lib/browsers/remote-interface/errors.ts @@ -2,40 +2,53 @@ const Kinds = Object.freeze({ CDP_CRASHED: 'cdp_crashed', CDP_DISCONNECTED: 'cdp_disconnected', CDP_TERMINATED: 'cdp_terminated', + CDP_ALREADY_CONNECTED: 'cdp_already_connected', }) type CdpErrorKind = typeof Kinds[keyof typeof Kinds] type MaybeCdpError = Error & { kind?: CdpErrorKind } -export class CdpCrashedError extends Error { +export class CDPCrashedError extends Error { public readonly kind = Kinds.CDP_CRASHED - public static isCdpCrashedError (error: MaybeCdpError): error is CdpCrashedError { + public static isCDPCrashedError (error: MaybeCdpError): error is CDPCrashedError { return error.kind === Kinds.CDP_CRASHED } } -export class CdpDisconnectedError extends Error { +export class CDPDisconnectedError extends Error { public readonly kind = Kinds.CDP_DISCONNECTED constructor (message: string, public readonly originalError?: Error) { super(message) } - public static isCdpDisconnectedError (error: MaybeCdpError): error is CdpDisconnectedError { + public static isCDPDisconnectedError (error: MaybeCdpError): error is CDPDisconnectedError { return error.kind === Kinds.CDP_DISCONNECTED } } -export class CdpTerminatedError extends Error { +export class CDPTerminatedError extends Error { public readonly kind = Kinds.CDP_TERMINATED constructor (message: string, public readonly originalError?: Error) { super(message) } - public static isCdpTerminatedError (error: MaybeCdpError): error is CdpTerminatedError { + public static isCDPTerminatedError (error: MaybeCdpError): error is CDPTerminatedError { return error.kind === Kinds.CDP_TERMINATED } } + +export class CDPAlreadyConnectedError extends Error { + public readonly kind = Kinds.CDP_ALREADY_CONNECTED + + constructor (message: string, public readonly originalError?: Error) { + super(message) + } + + public static isCDPAlreadyConnectedError (error: MaybeCdpError): error is CDPAlreadyConnectedError { + return error.kind === Kinds.CDP_ALREADY_CONNECTED + } +} diff --git a/packages/server/test/integration/cdp_spec.ts b/packages/server/test/integration/cdp_spec.ts index 2bd2b5bd4422..cdd01ecabb25 100644 --- a/packages/server/test/integration/cdp_spec.ts +++ b/packages/server/test/integration/cdp_spec.ts @@ -4,7 +4,7 @@ import Debug from 'debug' import _ from 'lodash' import WebSocket from 'ws' import { CdpCommand, CdpEvent } from '../../lib/browsers/cdp_automation' -import { CriClient } from '../../lib/browsers/cri-client' +import { CriClient } from '../../lib/browsers/remote-interface/cri-client' import { expect, nock } from '../spec_helper' import pDefer from 'p-defer' import sinon from 'sinon' diff --git a/packages/server/test/unit/browsers/cdp-command-queue_spec.ts b/packages/server/test/unit/browsers/cdp-command-queue_spec.ts index dbb3f4db74cc..0bedd91b8f19 100644 --- a/packages/server/test/unit/browsers/cdp-command-queue_spec.ts +++ b/packages/server/test/unit/browsers/cdp-command-queue_spec.ts @@ -1,4 +1,4 @@ -import { CDPCommandQueue, Command } from '../../../lib/browsers/cdp-command-queue' +import { CDPCommandQueue, Command } from '../../../lib/browsers/remote-interface/cdp-command-queue' import type ProtocolMapping from 'devtools-protocol/types/protocol-mapping' import pDeferred from 'p-defer' import _ from 'lodash' diff --git a/packages/server/test/unit/browsers/cri-client_spec.ts b/packages/server/test/unit/browsers/cri-client_spec.ts index 91b2bad9ac39..5ea732df810c 100644 --- a/packages/server/test/unit/browsers/cri-client_spec.ts +++ b/packages/server/test/unit/browsers/cri-client_spec.ts @@ -9,7 +9,7 @@ const DEBUGGER_URL = 'http://foo' const HOST = '127.0.0.1' const PORT = 50505 -describe('lib/browsers/cri-client', function () { +describe('lib/browsers/remote-interface/cri-client', function () { let send: sinon.SinonStub let on: sinon.SinonStub let off: sinon.SinonStub @@ -223,13 +223,15 @@ describe('lib/browsers/cri-client', function () { }) it('errors if reconnecting fails', async () => { - criStub._notifier.on = sinon.stub() - criStub.close.throws(new Error('could not reconnect')) - await getClient() + + criImport.rejects() + // @ts-ignore await criStub.on.withArgs('disconnect').args[0][1]() + await (new Promise((resolve) => setImmediate(resolve))) + expect(onError).to.be.called const error = onError.lastCall.args[0] diff --git a/packages/server/test/unit/browsers/firefox_spec.ts b/packages/server/test/unit/browsers/firefox_spec.ts index d1b368fba89c..a9056352b665 100644 --- a/packages/server/test/unit/browsers/firefox_spec.ts +++ b/packages/server/test/unit/browsers/firefox_spec.ts @@ -12,7 +12,7 @@ import * as firefox from '../../../lib/browsers/firefox' import firefoxUtil from '../../../lib/browsers/firefox-util' import { CdpAutomation } from '../../../lib/browsers/cdp_automation' import { BrowserCriClient } from '../../../lib/browsers/remote-interface/browser-cri-client' -import { ICriClient } from '../../../lib/browsers/cri-client' +import { ICriClient } from '../../../lib/browsers/remote-interface/cri-client' const path = require('path') const _ = require('lodash') diff --git a/packages/server/test/unit/browsers/remote-interface/cdp-connection_spec.ts b/packages/server/test/unit/browsers/remote-interface/cdp-connection_spec.ts new file mode 100644 index 000000000000..a0138745a6e9 --- /dev/null +++ b/packages/server/test/unit/browsers/remote-interface/cdp-connection_spec.ts @@ -0,0 +1,351 @@ +import type CDP from 'chrome-remote-interface' +import type { CdpClient, CDPConnection, CDPConnectionOptions } from '../../../../lib/browsers/remote-interface/cdp-connection' +import type { debugCdpConnection } from '../../../../lib/browsers/remote-interface/debug-cdp-connection' +import type { CdpEvent, CdpCommand } from '../../../../lib/browsers/cdp_automation' +import type ProtocolMapping from 'devtools-protocol/types/protocol-mapping' +import { CDPTerminatedError, CDPAlreadyConnectedError, CDPDisconnectedError } from '../../../../lib/browsers/remote-interface/errors' +import WebSocket from 'ws' +import pDefer, { DeferredPromise } from 'p-defer' +const { expect, proxyquire, sinon } = require('../../../spec_helper') + +const DEBUGGER_URL = 'http://foo' + +type CDPConnectionCtor = new (_options: CDP.Options, connectionOptions: CDPConnectionOptions) => CDPConnection + +describe('CDPConnection', () => { + let stubbedCDPClient: sinon.SinonStubbedInstance & { _ws?: sinon.SinonStubbedInstance }> + let stubbedWebSocket: sinon.SinonStubbedInstance + let stubbedDebugger: sinon.SinonStub, ReturnType> + + let CDPConnection: CDPConnectionCtor + let CDPImport: sinon.SinonStub + + let cdpConnection: CDPConnection + + let onReconnectCb: sinon.SinonStub + let onReconnectAttemptCb: sinon.SinonStub + let onReconnectErrCb: sinon.SinonStub + let onConnectionClosedCb: sinon.SinonStub + + const createStubbedCdpClient = () => { + return { + send: sinon.stub(), + on: sinon.stub(), + off: sinon.stub(), + close: sinon.stub().resolves(), + _ws: stubbedWebSocket, + } + } + + beforeEach(() => { + stubbedWebSocket = sinon.createStubInstance(WebSocket) + + stubbedCDPClient = createStubbedCdpClient() + + stubbedDebugger = sinon.stub().callsFake(() => {}) + + CDPImport = sinon.stub() + + CDPConnection = proxyquire('../lib/browsers/remote-interface/cdp-connection', { + 'chrome-remote-interface': CDPImport, + './debug-cdp-connection': stubbedDebugger, + }).CDPConnection + + cdpConnection = new CDPConnection({ + target: DEBUGGER_URL, + local: true, + }, { automaticallyReconnect: false }) + + onReconnectCb = sinon.stub() + onReconnectAttemptCb = sinon.stub() + onReconnectErrCb = sinon.stub() + onConnectionClosedCb = sinon.stub() + + cdpConnection.addConnectionEventListener('cdp-connection-reconnect', onReconnectCb) + cdpConnection.addConnectionEventListener('cdp-connection-reconnect-attempt', onReconnectAttemptCb) + cdpConnection.addConnectionEventListener('cdp-connection-reconnect-error', onReconnectErrCb) + cdpConnection.addConnectionEventListener('cdp-connection-closed', onConnectionClosedCb) + }) + + describe('.connect()', () => { + describe('when CDP connects', () => { + beforeEach(() => { + CDPImport.withArgs({ target: DEBUGGER_URL, local: true }).resolves(stubbedCDPClient) + }) + + it('resolves', async () => { + await expect(cdpConnection.connect()).to.be.fulfilled + }) + + describe('when there is already an active connection', () => { + beforeEach(async () => { + await cdpConnection.connect() + }) + + it('rejects with a CDPAlreadyConnectedError', async () => { + await expect(cdpConnection.connect()).to.be.rejectedWith(CDPAlreadyConnectedError, DEBUGGER_URL) + }) + }) + }) + + describe('when CDP fails to connect', () => { + let someErr: Error + + beforeEach(() => { + someErr = new Error('some error') + CDPImport.withArgs({ target: DEBUGGER_URL, local: true }).rejects(someErr) + }) + + it('rejects', async () => { + await expect(cdpConnection.connect()).to.be.rejectedWith(someErr) + }) + }) + + describe('when the connection has been terminated', () => { + beforeEach(async () => { + await cdpConnection.disconnect() + }) + + it('rejects with a CdpTerminatedError', async () => { + await expect(cdpConnection.connect()).to.be.rejectedWith(CDPTerminatedError, DEBUGGER_URL) + }) + }) + + describe('when CDP disconnects', () => { + beforeEach(async () => { + CDPImport.withArgs({ target: DEBUGGER_URL, local: true }).resolves(stubbedCDPClient) + await cdpConnection.connect() + }) + + it('does not add a reconnect listener', async () => { + //@ts-expect-error + expect(stubbedCDPClient.on?.withArgs('disconnect')).not.to.have.been.called + }) + }) + }) + + describe('.disconnect()', () => { + describe('when the connection has not been terminated and there is an active connection', () => { + beforeEach(async () => { + CDPImport.withArgs({ target: DEBUGGER_URL, local: true }).resolves(stubbedCDPClient) + + await cdpConnection.connect() + }) + + it('removes any event listeners that have been added', async () => { + await cdpConnection.disconnect() + + const calls = stubbedCDPClient.on?.getCalls() + + if (calls?.length) { + for (const call of calls) { + const [event, listener] = call.args + + expect(stubbedCDPClient.off).to.have.been.calledWith(event, listener) + } + } + }) + + it('closes the CDP connection', async () => { + await cdpConnection.disconnect() + expect(stubbedCDPClient.close).to.have.been.called + }) + + it('marks this connection as terminated', async () => { + await cdpConnection.disconnect() + expect(cdpConnection.terminated).to.be.true + }) + + it('emits the cdp-connection-closed connection event', async () => { + await cdpConnection.disconnect() + await new Promise((resolve) => setImmediate(resolve)) + expect(onConnectionClosedCb).to.have.been.called + }) + + describe('when the connection has already been terminated', () => { + beforeEach(async () => { + await cdpConnection.disconnect() + stubbedCDPClient.close?.resetHistory() + onConnectionClosedCb.reset() + }) + + it('does not emit a lifecycle event, remove listeners, etc', async () => { + await expect(cdpConnection.disconnect()).to.be.fulfilled + expect(onConnectionClosedCb).not.to.have.been.called + expect(cdpConnection.terminated).to.be.true + expect(stubbedCDPClient.close).not.to.have.been.called + }) + }) + }) + + describe('when there is no active connection', () => { + it('does not throw, emit a lifecycle event, remove listeners, or call close on the cdp client', async () => { + await expect(cdpConnection.disconnect()).to.be.fulfilled + expect(onConnectionClosedCb).not.to.have.been.called + expect(cdpConnection.terminated).to.be.true + expect(stubbedCDPClient.close).not.to.have.been.called + }) + }) + }) + + describe('.send()', () => { + const method: CdpCommand = 'Runtime.runScript' + const params = { scriptId: 'efg' } + const sessionId = 'abc' + + describe('when the connection has been established', () => { + beforeEach(async () => { + CDPImport.withArgs({ target: DEBUGGER_URL, local: true }).resolves(stubbedCDPClient) + await cdpConnection.connect() + }) + + describe('when the CDP command resolves', () => { + const resolve: ProtocolMapping.Commands['Runtime.runScript']['returnType'] = { + result: { + type: 'undefined', + }, + } + + beforeEach(() => { + stubbedCDPClient.send?.withArgs(method, params, sessionId).resolves(resolve) + }) + + it('resolves with the same value', async () => { + await expect(cdpConnection.send(method, params, sessionId)).to.eventually.eq(resolve) + }) + }) + + describe('when the CDP command rejects with a general error', () => { + const err = new Error('some err') + + beforeEach(() => { + stubbedCDPClient.send?.rejects(err) + }) + + it('rejects with the same error', async () => { + await expect(cdpConnection.send(method, params, sessionId)).to.be.rejectedWith(err) + }) + }) + + describe('when the CDP command rejects with a websocket disconnection error message', () => { + ['WebSocket connection closed', 'WebSocket is not open', 'WebSocket is already in CLOSING or CLOSED state'].forEach((msg) => { + it(` it rejects "${msg}" with a CDPDisconnectedError`, async () => { + const err = new Error(msg) + + stubbedCDPClient.send?.rejects(err) + await expect(cdpConnection.send(method, params, sessionId)).to.be.rejectedWith(CDPDisconnectedError) + }) + }) + }) + }) + + describe('when the connection has yet to be established', () => { + it('rejects with a CDPDisconnectedError', async () => { + await expect(cdpConnection.send(method, params, sessionId)).to.be.rejectedWith(CDPDisconnectedError, 'has not been established') + }) + }) + + describe('when the connection has been terminated', () => { + it('rejects with a CDPDisconnectedError', async () => { + CDPImport.withArgs({ target: DEBUGGER_URL, local: true }).resolves(stubbedCDPClient) + await cdpConnection.connect() + await cdpConnection.disconnect() + await expect(cdpConnection.send(method, params, sessionId)).to.be.rejectedWith(CDPDisconnectedError, 'terminated') + }) + }) + }) + + describe('.on()', () => { + const event: CdpEvent = 'Browser.downloadProgress' + const params = { some: 'params' } + + it('calls the callback when cdp client broadcasts the event', async () => { + const cb = sinon.stub() + + CDPImport.withArgs({ target: DEBUGGER_URL, local: true }).resolves(stubbedCDPClient) + + await cdpConnection.connect() + cdpConnection.on(event, cb) + + //@ts-expect-error + stubbedCDPClient.on?.withArgs('event').args[0][1]({ + method: event, + params, + }) + + await (new Promise((resolve) => setImmediate(resolve))) + expect(cb).to.have.been.calledWith(params) + }) + }) + + describe('.off()', () => { + const event: CdpEvent = 'Browser.downloadProgress' + const params = { some: 'params' } + + it('no longer calls the callback when cdp client broadcasts the event', async () => { + const cb = sinon.stub() + + cdpConnection.on(event, cb) + CDPImport.withArgs({ target: DEBUGGER_URL, local: true }).resolves(stubbedCDPClient) + + await cdpConnection.connect() + cdpConnection.off(event, cb) + + //@ts-expect-error + stubbedCDPClient.on?.withArgs('event').args[0][1]({ + method: event, + params, + }) + + await (new Promise((resolve) => setImmediate(resolve))) + expect(cb).not.to.have.been.calledWith(params) + }) + }) + + describe('when created with auto reconnect behavior enabled and disconnected', () => { + let deferredReconnection: DeferredPromise + + beforeEach(async () => { + cdpConnection = new CDPConnection({ + target: DEBUGGER_URL, + local: true, + }, { automaticallyReconnect: true }) + + cdpConnection.addConnectionEventListener('cdp-connection-reconnect', onReconnectCb) + cdpConnection.addConnectionEventListener('cdp-connection-reconnect-attempt', onReconnectAttemptCb) + cdpConnection.addConnectionEventListener('cdp-connection-reconnect-error', onReconnectErrCb) + + deferredReconnection = pDefer() + onReconnectCb.callsFake(() => deferredReconnection.resolve()) + onReconnectErrCb.callsFake(() => deferredReconnection.reject()) + CDPImport.withArgs({ target: DEBUGGER_URL, local: true }).resolves(stubbedCDPClient) + + await cdpConnection.connect() + // @ts-expect-error + stubbedCDPClient.on?.withArgs('disconnect').args[0][1]() + CDPImport.reset() + }) + + it('reconnects when disconnected and reconnection succeeds on the third try', async () => { + CDPImport.onFirstCall().rejects() + CDPImport.onSecondCall().rejects() + + const newCDPClientStub = createStubbedCdpClient() + + CDPImport.onThirdCall().resolves(newCDPClientStub) + + await deferredReconnection.promise + + expect(onReconnectAttemptCb).to.have.callCount(3) + expect(onReconnectCb).to.be.called + expect(onReconnectErrCb).not.to.be.called + }) + + it('rejects when disconnected and reconnection fails after 20 attempts', async () => { + CDPImport.rejects() + await expect(deferredReconnection.promise).to.be.rejected + expect(onReconnectErrCb).to.be.called + expect(onReconnectAttemptCb).to.have.callCount(20) + }) + }) +}) From 254c18eccb7292faeae1e0cf5b53de69e1220133 Mon Sep 17 00:00:00 2001 From: Cacie Prins Date: Wed, 7 Aug 2024 10:04:27 -0400 Subject: [PATCH 08/25] lift out of the remote-interface dir --- .../browsers/{remote-interface => }/browser-cri-client.ts | 0 .../browsers/{remote-interface => }/cdp-command-queue.ts | 0 .../lib/browsers/{remote-interface => }/cdp-connection.ts | 2 +- .../lib/browsers/{remote-interface => }/cri-client.ts | 2 +- .../{remote-interface/errors.ts => cri-errors.ts} | 0 .../{remote-interface => }/debug-cdp-connection.ts | 0 packages/server/test/integration/cdp_spec.ts | 2 +- packages/server/test/integration/cypress_spec.js | 2 +- .../server/test/unit/browsers/browser-cri-client_spec.ts | 6 +++--- .../server/test/unit/browsers/cdp-command-queue_spec.ts | 2 +- packages/server/test/unit/browsers/chrome_spec.js | 2 +- packages/server/test/unit/browsers/cri-client_spec.ts | 8 ++++---- packages/server/test/unit/browsers/electron_spec.js | 2 +- packages/server/test/unit/browsers/firefox_spec.ts | 4 ++-- .../unit/browsers/remote-interface/cdp-connection_spec.ts | 8 ++++---- 15 files changed, 20 insertions(+), 20 deletions(-) rename packages/server/lib/browsers/{remote-interface => }/browser-cri-client.ts (100%) rename packages/server/lib/browsers/{remote-interface => }/cdp-command-queue.ts (100%) rename packages/server/lib/browsers/{remote-interface => }/cdp-connection.ts (99%) rename packages/server/lib/browsers/{remote-interface => }/cri-client.ts (99%) rename packages/server/lib/browsers/{remote-interface/errors.ts => cri-errors.ts} (100%) rename packages/server/lib/browsers/{remote-interface => }/debug-cdp-connection.ts (100%) diff --git a/packages/server/lib/browsers/remote-interface/browser-cri-client.ts b/packages/server/lib/browsers/browser-cri-client.ts similarity index 100% rename from packages/server/lib/browsers/remote-interface/browser-cri-client.ts rename to packages/server/lib/browsers/browser-cri-client.ts diff --git a/packages/server/lib/browsers/remote-interface/cdp-command-queue.ts b/packages/server/lib/browsers/cdp-command-queue.ts similarity index 100% rename from packages/server/lib/browsers/remote-interface/cdp-command-queue.ts rename to packages/server/lib/browsers/cdp-command-queue.ts diff --git a/packages/server/lib/browsers/remote-interface/cdp-connection.ts b/packages/server/lib/browsers/cdp-connection.ts similarity index 99% rename from packages/server/lib/browsers/remote-interface/cdp-connection.ts rename to packages/server/lib/browsers/cdp-connection.ts index 4123c7ca58bf..56a7f04064f1 100644 --- a/packages/server/lib/browsers/remote-interface/cdp-connection.ts +++ b/packages/server/lib/browsers/cdp-connection.ts @@ -5,7 +5,7 @@ import CDP from 'chrome-remote-interface' import type { CypressError } from '@packages/errors' import { debugCdpConnection, DebuggableCDPClient } from './debug-cdp-connection' import type { CdpEvent, CdpCommand } from '../cdp_automation' -import { CDPDisconnectedError, CDPTerminatedError, CDPAlreadyConnectedError } from './errors' +import { CDPDisconnectedError, CDPTerminatedError, CDPAlreadyConnectedError } from './cri-errors' import { asyncRetry } from '../../util/async_retry' import * as errors from '../../errors' import type WebSocket from 'ws' diff --git a/packages/server/lib/browsers/remote-interface/cri-client.ts b/packages/server/lib/browsers/cri-client.ts similarity index 99% rename from packages/server/lib/browsers/remote-interface/cri-client.ts rename to packages/server/lib/browsers/cri-client.ts index 490b625d7f3e..87f53b5c50a3 100644 --- a/packages/server/lib/browsers/remote-interface/cri-client.ts +++ b/packages/server/lib/browsers/cri-client.ts @@ -5,7 +5,7 @@ import type ProtocolMapping from 'devtools-protocol/types/protocol-mapping' import type WebSocket from 'ws' import type { CypressError } from '@packages/errors' import type { SendDebuggerCommand, OnFn, OffFn, CdpCommand, CdpEvent } from '../cdp_automation' -import { CDPDisconnectedError } from './errors' +import { CDPDisconnectedError } from './cri-errors' import type { ProtocolManagerShape } from '@packages/types' const debug = debugModule('cypress:server:browsers:cri-client') diff --git a/packages/server/lib/browsers/remote-interface/errors.ts b/packages/server/lib/browsers/cri-errors.ts similarity index 100% rename from packages/server/lib/browsers/remote-interface/errors.ts rename to packages/server/lib/browsers/cri-errors.ts diff --git a/packages/server/lib/browsers/remote-interface/debug-cdp-connection.ts b/packages/server/lib/browsers/debug-cdp-connection.ts similarity index 100% rename from packages/server/lib/browsers/remote-interface/debug-cdp-connection.ts rename to packages/server/lib/browsers/debug-cdp-connection.ts diff --git a/packages/server/test/integration/cdp_spec.ts b/packages/server/test/integration/cdp_spec.ts index cdd01ecabb25..2bd2b5bd4422 100644 --- a/packages/server/test/integration/cdp_spec.ts +++ b/packages/server/test/integration/cdp_spec.ts @@ -4,7 +4,7 @@ import Debug from 'debug' import _ from 'lodash' import WebSocket from 'ws' import { CdpCommand, CdpEvent } from '../../lib/browsers/cdp_automation' -import { CriClient } from '../../lib/browsers/remote-interface/cri-client' +import { CriClient } from '../../lib/browsers/cri-client' import { expect, nock } from '../spec_helper' import pDefer from 'p-defer' import sinon from 'sinon' diff --git a/packages/server/test/integration/cypress_spec.js b/packages/server/test/integration/cypress_spec.js index 37977eb9092d..756d31947c66 100644 --- a/packages/server/test/integration/cypress_spec.js +++ b/packages/server/test/integration/cypress_spec.js @@ -44,7 +44,7 @@ const appData = require(`../../lib/util/app_data`) const electronApp = require('../../lib/util/electron-app') const savedState = require(`../../lib/saved_state`) const { getCtx, clearCtx, setCtx, makeDataContext } = require(`../../lib/makeDataContext`) -const { BrowserCriClient } = require(`../../lib/browsers/remote-interface/browser-cri-client`) +const { BrowserCriClient } = require(`../../lib/browsers/browser-cri-client`) const { cloudRecommendationMessage } = require('../../lib/util/print-run') const processVersions = process.versions diff --git a/packages/server/test/unit/browsers/browser-cri-client_spec.ts b/packages/server/test/unit/browsers/browser-cri-client_spec.ts index cc8dc7c6cd22..510e553dfb02 100644 --- a/packages/server/test/unit/browsers/browser-cri-client_spec.ts +++ b/packages/server/test/unit/browsers/browser-cri-client_spec.ts @@ -1,5 +1,5 @@ -import { BrowserCriClient } from '../../../lib/browsers/remote-interface/browser-cri-client' -import { CriClient } from '../../../lib/browsers/remote-interface/cri-client' +import { BrowserCriClient } from '../../../lib/browsers/browser-cri-client' +import { CriClient } from '../../../lib/browsers/cri-client' import { expect, proxyquire, sinon } from '../../spec_helper' import * as protocol from '../../../lib/browsers/protocol' import { stripAnsi } from '@packages/errors' @@ -55,7 +55,7 @@ describe('lib/browsers/browser-cri-client', function () { close, }) - browserCriClient = proxyquire('../lib/browsers/remote-interface/browser-cri-client', { + browserCriClient = proxyquire('../lib/browsers/browser-cri-client', { 'chrome-remote-interface': criImport, }) diff --git a/packages/server/test/unit/browsers/cdp-command-queue_spec.ts b/packages/server/test/unit/browsers/cdp-command-queue_spec.ts index 0bedd91b8f19..dbb3f4db74cc 100644 --- a/packages/server/test/unit/browsers/cdp-command-queue_spec.ts +++ b/packages/server/test/unit/browsers/cdp-command-queue_spec.ts @@ -1,4 +1,4 @@ -import { CDPCommandQueue, Command } from '../../../lib/browsers/remote-interface/cdp-command-queue' +import { CDPCommandQueue, Command } from '../../../lib/browsers/cdp-command-queue' import type ProtocolMapping from 'devtools-protocol/types/protocol-mapping' import pDeferred from 'p-defer' import _ from 'lodash' diff --git a/packages/server/test/unit/browsers/chrome_spec.js b/packages/server/test/unit/browsers/chrome_spec.js index f930cd89a6f6..b09f4de1b3bf 100644 --- a/packages/server/test/unit/browsers/chrome_spec.js +++ b/packages/server/test/unit/browsers/chrome_spec.js @@ -9,7 +9,7 @@ const plugins = require(`../../../lib/plugins`) const utils = require(`../../../lib/browsers/utils`) const chrome = require(`../../../lib/browsers/chrome`) const { fs } = require(`../../../lib/util/fs`) -const { BrowserCriClient } = require('../../../lib/browsers/remote-interface/browser-cri-client') +const { BrowserCriClient } = require('../../../lib/browsers/browser-cri-client') const { expect } = require('chai') const openOpts = { diff --git a/packages/server/test/unit/browsers/cri-client_spec.ts b/packages/server/test/unit/browsers/cri-client_spec.ts index 5ea732df810c..da6828160e09 100644 --- a/packages/server/test/unit/browsers/cri-client_spec.ts +++ b/packages/server/test/unit/browsers/cri-client_spec.ts @@ -1,7 +1,7 @@ import type ProtocolMapping from 'devtools-protocol/types/protocol-mapping' import EventEmitter from 'events' import { ProtocolManagerShape } from '@packages/types' -import type { CriClient } from '../../../lib/browsers/remote-interface/cri-client' +import type { CriClient } from '../../../lib/browsers/cri-client' import pDefer from 'p-defer' const { expect, proxyquire, sinon } = require('../../spec_helper') @@ -9,7 +9,7 @@ const DEBUGGER_URL = 'http://foo' const HOST = '127.0.0.1' const PORT = 50505 -describe('lib/browsers/remote-interface/cri-client', function () { +describe('lib/browsers/cri-client', function () { let send: sinon.SinonStub let on: sinon.SinonStub let off: sinon.SinonStub @@ -60,11 +60,11 @@ describe('lib/browsers/remote-interface/cri-client', function () { criImport.New = sinon.stub().withArgs({ host: HOST, port: PORT, url: 'about:blank' }).resolves({ webSocketDebuggerUrl: 'http://web/socket/url' }) - const { CDPConnection } = proxyquire('../lib/browsers/remote-interface/cdp-connection', { + const { CDPConnection } = proxyquire('../lib/browsers/cdp-connection', { 'chrome-remote-interface': criImport, }) - const { CriClient } = proxyquire('../lib/browsers/remote-interface/cri-client', { + const { CriClient } = proxyquire('../lib/browsers/cri-client', { './cdp-connection': { CDPConnection }, }) diff --git a/packages/server/test/unit/browsers/electron_spec.js b/packages/server/test/unit/browsers/electron_spec.js index 72f92d489e39..2551326d78c5 100644 --- a/packages/server/test/unit/browsers/electron_spec.js +++ b/packages/server/test/unit/browsers/electron_spec.js @@ -11,7 +11,7 @@ const Windows = require(`../../../lib/gui/windows`) const electron = require(`../../../lib/browsers/electron`) const savedState = require(`../../../lib/saved_state`) const { Automation } = require(`../../../lib/automation`) -const { BrowserCriClient } = require('../../../lib/browsers/remote-interface/browser-cri-client') +const { BrowserCriClient } = require('../../../lib/browsers/browser-cri-client') const electronApp = require('../../../lib/util/electron-app') const utils = require('../../../lib/browsers/utils') diff --git a/packages/server/test/unit/browsers/firefox_spec.ts b/packages/server/test/unit/browsers/firefox_spec.ts index a9056352b665..cbca36a5748f 100644 --- a/packages/server/test/unit/browsers/firefox_spec.ts +++ b/packages/server/test/unit/browsers/firefox_spec.ts @@ -11,8 +11,8 @@ import Foxdriver from '@benmalka/foxdriver' import * as firefox from '../../../lib/browsers/firefox' import firefoxUtil from '../../../lib/browsers/firefox-util' import { CdpAutomation } from '../../../lib/browsers/cdp_automation' -import { BrowserCriClient } from '../../../lib/browsers/remote-interface/browser-cri-client' -import { ICriClient } from '../../../lib/browsers/remote-interface/cri-client' +import { BrowserCriClient } from '../../../lib/browsers/browser-cri-client' +import { ICriClient } from '../../../lib/browsers/cri-client' const path = require('path') const _ = require('lodash') diff --git a/packages/server/test/unit/browsers/remote-interface/cdp-connection_spec.ts b/packages/server/test/unit/browsers/remote-interface/cdp-connection_spec.ts index a0138745a6e9..7307d55b7e17 100644 --- a/packages/server/test/unit/browsers/remote-interface/cdp-connection_spec.ts +++ b/packages/server/test/unit/browsers/remote-interface/cdp-connection_spec.ts @@ -1,9 +1,9 @@ import type CDP from 'chrome-remote-interface' -import type { CdpClient, CDPConnection, CDPConnectionOptions } from '../../../../lib/browsers/remote-interface/cdp-connection' -import type { debugCdpConnection } from '../../../../lib/browsers/remote-interface/debug-cdp-connection' +import type { CdpClient, CDPConnection, CDPConnectionOptions } from '../../../../lib/browsers/cdp-connection' +import type { debugCdpConnection } from '../../../../lib/browsers/debug-cdp-connection' import type { CdpEvent, CdpCommand } from '../../../../lib/browsers/cdp_automation' import type ProtocolMapping from 'devtools-protocol/types/protocol-mapping' -import { CDPTerminatedError, CDPAlreadyConnectedError, CDPDisconnectedError } from '../../../../lib/browsers/remote-interface/errors' +import { CDPTerminatedError, CDPAlreadyConnectedError, CDPDisconnectedError } from '../../../../lib/browsers/cri-errors' import WebSocket from 'ws' import pDefer, { DeferredPromise } from 'p-defer' const { expect, proxyquire, sinon } = require('../../../spec_helper') @@ -46,7 +46,7 @@ describe('CDPConnection', () => { CDPImport = sinon.stub() - CDPConnection = proxyquire('../lib/browsers/remote-interface/cdp-connection', { + CDPConnection = proxyquire('../lib/browsers/cdp-connection', { 'chrome-remote-interface': CDPImport, './debug-cdp-connection': stubbedDebugger, }).CDPConnection From 17f5bcdb5601762f3e39494cd89f1dfee7ba483e Mon Sep 17 00:00:00 2001 From: Cacie Prins Date: Wed, 7 Aug 2024 12:27:40 -0400 Subject: [PATCH 09/25] complete lift from /remote-interface --- packages/server/lib/browsers/cdp_automation.ts | 2 +- packages/server/lib/browsers/chrome.ts | 4 ++-- packages/server/lib/browsers/electron.ts | 2 +- packages/server/lib/browsers/firefox-util.ts | 2 +- packages/server/lib/browsers/firefox.ts | 2 +- packages/server/lib/browsers/utils.ts | 2 +- 6 files changed, 7 insertions(+), 7 deletions(-) diff --git a/packages/server/lib/browsers/cdp_automation.ts b/packages/server/lib/browsers/cdp_automation.ts index 7e3f6d695f92..853a14ad1a2e 100644 --- a/packages/server/lib/browsers/cdp_automation.ts +++ b/packages/server/lib/browsers/cdp_automation.ts @@ -13,7 +13,7 @@ import type { ResourceType, BrowserPreRequest, BrowserResponseReceived } from '@ import type { CDPClient, ProtocolManagerShape, WriteVideoFrame } from '@packages/types' import type { Automation } from '../automation' import { cookieMatches, CyCookie, CyCookieFilter } from '../automation/util' -import { DEFAULT_NETWORK_ENABLE_OPTIONS, CriClient } from './remote-interface/cri-client' +import { DEFAULT_NETWORK_ENABLE_OPTIONS, CriClient } from './cri-client' export type CdpCommand = keyof ProtocolMapping.Commands diff --git a/packages/server/lib/browsers/chrome.ts b/packages/server/lib/browsers/chrome.ts index 5b8b469d71e3..33ae2d6643f9 100644 --- a/packages/server/lib/browsers/chrome.ts +++ b/packages/server/lib/browsers/chrome.ts @@ -15,9 +15,9 @@ import { CdpAutomation, screencastOpts } from './cdp_automation' import * as protocol from './protocol' import utils from './utils' import * as errors from '../errors' -import { BrowserCriClient } from './remote-interface/browser-cri-client' +import { BrowserCriClient } from './browser-cri-client' import type { Browser, BrowserInstance, GracefulShutdownOptions } from './types' -import type { CriClient } from './remote-interface/cri-client' +import type { CriClient } from './cri-client' import type { Automation } from '../automation' import memory from './memory' diff --git a/packages/server/lib/browsers/electron.ts b/packages/server/lib/browsers/electron.ts index 02c8ebd67312..97f318f1bd1b 100644 --- a/packages/server/lib/browsers/electron.ts +++ b/packages/server/lib/browsers/electron.ts @@ -14,7 +14,7 @@ import type { Automation } from '../automation' import type { BrowserLaunchOpts, Preferences, ProtocolManagerShape, RunModeVideoApi } from '@packages/types' import type { CDPSocketServer } from '@packages/socket/lib/cdp-socket' import memory from './memory' -import { BrowserCriClient } from './remote-interface/browser-cri-client' +import { BrowserCriClient } from './browser-cri-client' import { getRemoteDebuggingPort } from '../util/electron-app' // TODO: unmix these two types diff --git a/packages/server/lib/browsers/firefox-util.ts b/packages/server/lib/browsers/firefox-util.ts index dfe39a6fd8af..38fff4cf01fa 100644 --- a/packages/server/lib/browsers/firefox-util.ts +++ b/packages/server/lib/browsers/firefox-util.ts @@ -7,7 +7,7 @@ import util from 'util' import Foxdriver from '@benmalka/foxdriver' import * as protocol from './protocol' import { CdpAutomation } from './cdp_automation' -import { BrowserCriClient } from './remote-interface/browser-cri-client' +import { BrowserCriClient } from './browser-cri-client' import type { Automation } from '../automation' const errors = require('../errors') diff --git a/packages/server/lib/browsers/firefox.ts b/packages/server/lib/browsers/firefox.ts index f460b567c3b6..8d78cf89da4b 100644 --- a/packages/server/lib/browsers/firefox.ts +++ b/packages/server/lib/browsers/firefox.ts @@ -16,7 +16,7 @@ import os from 'os' import treeKill from 'tree-kill' import mimeDb from 'mime-db' import { getRemoteDebuggingPort } from './protocol' -import type { BrowserCriClient } from './remote-interface/browser-cri-client' +import type { BrowserCriClient } from './browser-cri-client' import type { Automation } from '../automation' import { getCtx } from '@packages/data-context' import { getError } from '@packages/errors' diff --git a/packages/server/lib/browsers/utils.ts b/packages/server/lib/browsers/utils.ts index d8ad842c0456..9cc77cf98dc1 100644 --- a/packages/server/lib/browsers/utils.ts +++ b/packages/server/lib/browsers/utils.ts @@ -8,7 +8,7 @@ import { getError } from '@packages/errors' import * as launcher from '@packages/launcher' import type { Automation } from '../automation' import type { Browser } from './types' -import type { CriClient } from './remote-interface/cri-client' +import type { CriClient } from './cri-client' declare global { interface Window { From 08428b13edf6eeee156354cca57b0360beeac509 Mon Sep 17 00:00:00 2001 From: Cacie Prins Date: Wed, 7 Aug 2024 15:05:37 -0400 Subject: [PATCH 10/25] further fix imports --- packages/server/lib/browsers/cdp-command-queue.ts | 2 +- packages/server/lib/browsers/cdp-connection.ts | 6 +++--- packages/server/lib/browsers/cri-client.ts | 2 +- 3 files changed, 5 insertions(+), 5 deletions(-) diff --git a/packages/server/lib/browsers/cdp-command-queue.ts b/packages/server/lib/browsers/cdp-command-queue.ts index 452c04e9c0f5..e79904e4960f 100644 --- a/packages/server/lib/browsers/cdp-command-queue.ts +++ b/packages/server/lib/browsers/cdp-command-queue.ts @@ -1,6 +1,6 @@ import type ProtocolMapping from 'devtools-protocol/types/protocol-mapping' import pDefer, { DeferredPromise } from 'p-defer' -import type { CdpCommand } from '../cdp_automation' +import type { CdpCommand } from './cdp_automation' import Debug from 'debug' const debug = Debug('cypress:server:browsers:cdp-command-queue') diff --git a/packages/server/lib/browsers/cdp-connection.ts b/packages/server/lib/browsers/cdp-connection.ts index 56a7f04064f1..4dda68b8571e 100644 --- a/packages/server/lib/browsers/cdp-connection.ts +++ b/packages/server/lib/browsers/cdp-connection.ts @@ -4,10 +4,10 @@ import EventEmitter from 'events' import CDP from 'chrome-remote-interface' import type { CypressError } from '@packages/errors' import { debugCdpConnection, DebuggableCDPClient } from './debug-cdp-connection' -import type { CdpEvent, CdpCommand } from '../cdp_automation' +import type { CdpEvent, CdpCommand } from './cdp_automation' import { CDPDisconnectedError, CDPTerminatedError, CDPAlreadyConnectedError } from './cri-errors' -import { asyncRetry } from '../../util/async_retry' -import * as errors from '../../errors' +import { asyncRetry } from '../util/async_retry' +import * as errors from '../errors' import type WebSocket from 'ws' const debug = Debug('cypress:server:browsers:cdp-connection') diff --git a/packages/server/lib/browsers/cri-client.ts b/packages/server/lib/browsers/cri-client.ts index 87f53b5c50a3..54a458dc9da1 100644 --- a/packages/server/lib/browsers/cri-client.ts +++ b/packages/server/lib/browsers/cri-client.ts @@ -4,7 +4,7 @@ import { CDPConnection, type CDPListener } from './cdp-connection' import type ProtocolMapping from 'devtools-protocol/types/protocol-mapping' import type WebSocket from 'ws' import type { CypressError } from '@packages/errors' -import type { SendDebuggerCommand, OnFn, OffFn, CdpCommand, CdpEvent } from '../cdp_automation' +import type { SendDebuggerCommand, OnFn, OffFn, CdpCommand, CdpEvent } from './cdp_automation' import { CDPDisconnectedError } from './cri-errors' import type { ProtocolManagerShape } from '@packages/types' From 44b888ff1927154be6eb61a319a7b1d76a404575 Mon Sep 17 00:00:00 2001 From: Cacie Prins Date: Wed, 7 Aug 2024 16:07:33 -0400 Subject: [PATCH 11/25] import --- packages/server/lib/browsers/browser-cri-client.ts | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/packages/server/lib/browsers/browser-cri-client.ts b/packages/server/lib/browsers/browser-cri-client.ts index 11d279be337e..8aadb945d790 100644 --- a/packages/server/lib/browsers/browser-cri-client.ts +++ b/packages/server/lib/browsers/browser-cri-client.ts @@ -3,8 +3,8 @@ import Bluebird from 'bluebird' import CRI from 'chrome-remote-interface' import Debug from 'debug' import type { Protocol } from 'devtools-protocol' -import { _connectAsync, _getDelayMsForRetry } from '../protocol' -import * as errors from '../../errors' +import { _connectAsync, _getDelayMsForRetry } from './protocol' +import * as errors from '../errors' import type { CypressError } from '@packages/errors' import { CriClient, DEFAULT_NETWORK_ENABLE_OPTIONS } from './cri-client' import { serviceWorkerClientEventHandler, serviceWorkerClientEventHandlerName } from '@packages/proxy/lib/http/util/service-worker-manager' From 57d6c9ee24b3b5737cfddb20cea0e0a185a07f54 Mon Sep 17 00:00:00 2001 From: Cacie Prins Date: Thu, 8 Aug 2024 10:35:28 -0400 Subject: [PATCH 12/25] fix disconnect behavior to make integration test pass --- packages/server/lib/browsers/cdp-connection.ts | 13 +++++++++---- packages/server/test/integration/cdp_spec.ts | 7 ++++--- .../{remote-interface => }/cdp-connection_spec.ts | 10 +++++----- 3 files changed, 18 insertions(+), 12 deletions(-) rename packages/server/test/unit/browsers/{remote-interface => }/cdp-connection_spec.ts (97%) diff --git a/packages/server/lib/browsers/cdp-connection.ts b/packages/server/lib/browsers/cdp-connection.ts index 4dda68b8571e..75aa0a3af438 100644 --- a/packages/server/lib/browsers/cdp-connection.ts +++ b/packages/server/lib/browsers/cdp-connection.ts @@ -100,6 +100,7 @@ export class CDPConnection { } async disconnect () { + debug('disconnect of target %s requested.', this._options.target, { terminated: this._terminated, connection: !!this._connection, reconnection: !!this._reconnection }) if (this._terminated && !this._connection) { return } @@ -108,7 +109,6 @@ export class CDPConnection { if (this._connection) { await this._gracefullyDisconnect() - this._emitter.emit('cdp-connection-closed') } } @@ -198,9 +198,14 @@ export class CDPConnection { const significantError: Error = err.errors ? (err as AggregateError).errors[err.errors.length - 1] : err const retryHaltedDueToClosed = CDPTerminatedError.isCDPTerminatedError(err) || - (err as AggregateError)?.errors?.find((predicate) => CDPTerminatedError.isCDPTerminatedError(predicate)) - - if (!retryHaltedDueToClosed) { + (err as AggregateError)?.errors?.find((predicate) => CDPTerminatedError.isCDPTerminatedError(predicate)) + + // if .disconnect() was called while trying to reconnect, there will be no active connection + // so the .disconnect() method will not emit the connection closed event. However, we do + // want to emit that once the reconnection attempts cease due to being closed. + if (retryHaltedDueToClosed) { + this._emitter.emit('cdp-connection-closed') + } else { const cdpError = errors.get('CDP_COULD_NOT_RECONNECT', significantError) cdpError.isFatalApiErr = true diff --git a/packages/server/test/integration/cdp_spec.ts b/packages/server/test/integration/cdp_spec.ts index 2bd2b5bd4422..e8b496fd7a68 100644 --- a/packages/server/test/integration/cdp_spec.ts +++ b/packages/server/test/integration/cdp_spec.ts @@ -29,7 +29,7 @@ describe('CDP Clients', () => { let criClient: CriClient let messages: object[] let onMessage: sinon.SinonStub - let messageResponse: ReturnType + let messageResponse: ReturnType | undefined let neverAck: boolean const startWsServer = async (onConnection?: OnWSConnection): Promise => { @@ -97,7 +97,7 @@ describe('CDP Clients', () => { const clientDisconnected = () => { return new Promise((resolve, reject) => { - criClient.ws.once('close', resolve) + criClient.ws?.once('close', resolve) }) } @@ -198,7 +198,8 @@ describe('CDP Clients', () => { }) await haltedReconnection - + // onCriConnectionClosed + await (new Promise((resolve) => setImmediate(resolve))) expect(onCriConnectionClosed).to.have.been.called }) diff --git a/packages/server/test/unit/browsers/remote-interface/cdp-connection_spec.ts b/packages/server/test/unit/browsers/cdp-connection_spec.ts similarity index 97% rename from packages/server/test/unit/browsers/remote-interface/cdp-connection_spec.ts rename to packages/server/test/unit/browsers/cdp-connection_spec.ts index 7307d55b7e17..45c67c48a2e6 100644 --- a/packages/server/test/unit/browsers/remote-interface/cdp-connection_spec.ts +++ b/packages/server/test/unit/browsers/cdp-connection_spec.ts @@ -1,12 +1,12 @@ import type CDP from 'chrome-remote-interface' -import type { CdpClient, CDPConnection, CDPConnectionOptions } from '../../../../lib/browsers/cdp-connection' -import type { debugCdpConnection } from '../../../../lib/browsers/debug-cdp-connection' -import type { CdpEvent, CdpCommand } from '../../../../lib/browsers/cdp_automation' +import type { CdpClient, CDPConnection, CDPConnectionOptions } from '../../../lib/browsers/cdp-connection' +import type { debugCdpConnection } from '../../../lib/browsers/debug-cdp-connection' +import type { CdpEvent, CdpCommand } from '../../../lib/browsers/cdp_automation' import type ProtocolMapping from 'devtools-protocol/types/protocol-mapping' -import { CDPTerminatedError, CDPAlreadyConnectedError, CDPDisconnectedError } from '../../../../lib/browsers/cri-errors' +import { CDPTerminatedError, CDPAlreadyConnectedError, CDPDisconnectedError } from '../../../lib/browsers/cri-errors' import WebSocket from 'ws' import pDefer, { DeferredPromise } from 'p-defer' -const { expect, proxyquire, sinon } = require('../../../spec_helper') +const { expect, proxyquire, sinon } = require('../../spec_helper') const DEBUGGER_URL = 'http://foo' From 66af7ed9ef4b5d88fb35ef3ea68b53a6fa520b94 Mon Sep 17 00:00:00 2001 From: Cacie Prins Date: Thu, 8 Aug 2024 10:53:04 -0400 Subject: [PATCH 13/25] Update packages/server/test/integration/cdp_spec.ts comment --- packages/server/test/integration/cdp_spec.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/server/test/integration/cdp_spec.ts b/packages/server/test/integration/cdp_spec.ts index e8b496fd7a68..5baa2c7b0e58 100644 --- a/packages/server/test/integration/cdp_spec.ts +++ b/packages/server/test/integration/cdp_spec.ts @@ -198,7 +198,7 @@ describe('CDP Clients', () => { }) await haltedReconnection - // onCriConnectionClosed + // Macrotask queue needs to flush for the event to trigger await (new Promise((resolve) => setImmediate(resolve))) expect(onCriConnectionClosed).to.have.been.called }) From 284a2102971c574e164a136f6198915d0601e7a0 Mon Sep 17 00:00:00 2001 From: Cacie Prins Date: Thu, 8 Aug 2024 11:04:31 -0400 Subject: [PATCH 14/25] fix snapgen --- packages/server/lib/browsers/cri-client.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/server/lib/browsers/cri-client.ts b/packages/server/lib/browsers/cri-client.ts index 54a458dc9da1..bf2717372cd1 100644 --- a/packages/server/lib/browsers/cri-client.ts +++ b/packages/server/lib/browsers/cri-client.ts @@ -1,6 +1,6 @@ import debugModule from 'debug' import { CDPCommandQueue } from './cdp-command-queue' -import { CDPConnection, type CDPListener } from './cdp-connection' +import { CDPConnection, CDPListener } from './cdp-connection' import type ProtocolMapping from 'devtools-protocol/types/protocol-mapping' import type WebSocket from 'ws' import type { CypressError } from '@packages/errors' From 3d537cc102dafe1710a8c53367c594b4e15464e1 Mon Sep 17 00:00:00 2001 From: Cacie Prins Date: Thu, 8 Aug 2024 15:02:25 -0400 Subject: [PATCH 15/25] improve debug --- packages/server/lib/browsers/cdp-connection.ts | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/packages/server/lib/browsers/cdp-connection.ts b/packages/server/lib/browsers/cdp-connection.ts index 75aa0a3af438..807c5d36a8a4 100644 --- a/packages/server/lib/browsers/cdp-connection.ts +++ b/packages/server/lib/browsers/cdp-connection.ts @@ -11,6 +11,7 @@ import * as errors from '../errors' import type WebSocket from 'ws' const debug = Debug('cypress:server:browsers:cdp-connection') +const verboseDebug = Debug('cypress-verbose:server:browsers:cdp-connection') export type CDPListener = (params: ProtocolMapping.Events[T][0], sessionId?: string) => void @@ -90,7 +91,7 @@ export class CDPConnection { this._connection = await CDP(this._options) as CdpClient - debugCdpConnection(debug.namespace, this._connection as DebuggableCDPClient) + debugCdpConnection(verboseDebug.namespace, this._connection as DebuggableCDPClient) this._connection.on('event', this._broadcastEvent) @@ -125,6 +126,7 @@ export class CDPConnection { data?: ProtocolMapping.Commands[T]['paramsType'][0], sessionId?: string, ): Promise { + debug('preparing to send CDP command:', command, data) if (this.terminated) { throw new CDPDisconnectedError(`${command} will not run as the CRI connection to Target ${this._options.target} has been terminated.`) } @@ -147,6 +149,7 @@ export class CDPConnection { } private _reconnect = async () => { + debug('Reconnection requested') if (this._terminated) { return } @@ -217,7 +220,7 @@ export class CDPConnection { } private _broadcastEvent = ({ method, params, sessionId }: { method: CdpEvent, params: Record, sessionId?: string }) => { - debug('rebroadcasting event', method, params, sessionId) + verboseDebug('rebroadcasting event', method, params, sessionId) this._emitter.emit(method, params, sessionId) } } From 750ca37426df06402aba42b96698f1836947363d Mon Sep 17 00:00:00 2001 From: Cacie Prins Date: Thu, 8 Aug 2024 15:37:11 -0400 Subject: [PATCH 16/25] further debug improvements --- .../server/lib/browsers/cdp-connection.ts | 33 +++++++++++-------- packages/server/lib/browsers/chrome.ts | 2 ++ packages/server/lib/browsers/cri-client.ts | 6 +++- packages/server/lib/modes/run.ts | 8 +++-- 4 files changed, 33 insertions(+), 16 deletions(-) diff --git a/packages/server/lib/browsers/cdp-connection.ts b/packages/server/lib/browsers/cdp-connection.ts index 807c5d36a8a4..778a678c4b82 100644 --- a/packages/server/lib/browsers/cdp-connection.ts +++ b/packages/server/lib/browsers/cdp-connection.ts @@ -10,8 +10,7 @@ import { asyncRetry } from '../util/async_retry' import * as errors from '../errors' import type WebSocket from 'ws' -const debug = Debug('cypress:server:browsers:cdp-connection') -const verboseDebug = Debug('cypress-verbose:server:browsers:cdp-connection') +const verboseDebugNs = 'cypress-verbose:server:browsers:cdp-connection' export type CDPListener = (params: ProtocolMapping.Events[T][0], sessionId?: string) => void @@ -51,9 +50,13 @@ export class CDPConnection { private _autoReconnect: boolean private _terminated: boolean = false private _reconnection: Promise | undefined + private debug: Debug.Debugger + private verboseDebug: Debug.Debugger constructor (private readonly _options: CDP.Options, connectionOptions: CDPConnectionOptions) { this._autoReconnect = connectionOptions.automaticallyReconnect + this.debug = Debug(`cypress:server:browsers:cdp-connection:${_options.target}`) + this.verboseDebug = Debug(`${verboseDebugNs}:${_options.target}`) } get terminated () { @@ -66,11 +69,11 @@ export class CDPConnection { } on (event: T, callback: CDPListener) { - debug('attaching event listener to cdp connection', event) + this.debug('attaching event listener to cdp connection', event) this._emitter.on(event, callback) } addConnectionEventListener (event: T, callback: CDPConnectionEventListener) { - debug('adding connection event listener for ', event) + this.debug('adding connection event listener for ', event) this._emitter.on(event, callback) } off (event: T, callback: CDPListener) { @@ -91,7 +94,7 @@ export class CDPConnection { this._connection = await CDP(this._options) as CdpClient - debugCdpConnection(verboseDebug.namespace, this._connection as DebuggableCDPClient) + debugCdpConnection(this.verboseDebug.namespace, this._connection as DebuggableCDPClient) this._connection.on('event', this._broadcastEvent) @@ -101,7 +104,7 @@ export class CDPConnection { } async disconnect () { - debug('disconnect of target %s requested.', this._options.target, { terminated: this._terminated, connection: !!this._connection, reconnection: !!this._reconnection }) + this.debug('disconnect of target %s requested.', this._options.target, { terminated: this._terminated, connection: !!this._connection, reconnection: !!this._reconnection }) if (this._terminated && !this._connection) { return } @@ -126,7 +129,7 @@ export class CDPConnection { data?: ProtocolMapping.Commands[T]['paramsType'][0], sessionId?: string, ): Promise { - debug('preparing to send CDP command:', command, data) + this.debug('preparing to send CDP command:', command) if (this.terminated) { throw new CDPDisconnectedError(`${command} will not run as the CRI connection to Target ${this._options.target} has been terminated.`) } @@ -149,7 +152,7 @@ export class CDPConnection { } private _reconnect = async () => { - debug('Reconnection requested') + this.debug('Reconnection requested') if (this._terminated) { return } @@ -162,7 +165,7 @@ export class CDPConnection { try { await this._gracefullyDisconnect() } catch (e) { - debug('Error cleaning up existing CDP connection before creating a new connection: ', e) + this.debug('Error cleaning up existing CDP connection before creating a new connection: ', e) } finally { this._connection = undefined } @@ -173,10 +176,10 @@ export class CDPConnection { this._reconnection = asyncRetry(async () => { attempt++ - debug('Reconnection attempt %d for Target %s', attempt, this._options.target) + this.debug('Reconnection attempt %d for Target %s', attempt, this._options.target) if (this._terminated) { - debug('Not reconnecting, connection to %s has been terminated', this._options.target) + this.debug('Not reconnecting, connection to %s has been terminated', this._options.target) throw new CDPTerminatedError(`Cannot reconnect to CDP. Client target ${this._options.target} has been terminated.`) } @@ -197,7 +200,7 @@ export class CDPConnection { await this._reconnection this._emitter.emit('cdp-connection-reconnect') } catch (err) { - debug('error(s) on reconnecting: ', err) + this.debug('error(s) on reconnecting: ', err) const significantError: Error = err.errors ? (err as AggregateError).errors[err.errors.length - 1] : err const retryHaltedDueToClosed = CDPTerminatedError.isCDPTerminatedError(err) || @@ -220,7 +223,11 @@ export class CDPConnection { } private _broadcastEvent = ({ method, params, sessionId }: { method: CdpEvent, params: Record, sessionId?: string }) => { - verboseDebug('rebroadcasting event', method, params, sessionId) + this.verboseDebug('rebroadcasting event', method, params, sessionId) + if (method === 'Target.targetCrashed') { + this.debug('broadcasting crash event') + } + this._emitter.emit(method, params, sessionId) } } diff --git a/packages/server/lib/browsers/chrome.ts b/packages/server/lib/browsers/chrome.ts index 33ae2d6643f9..d386dded7604 100644 --- a/packages/server/lib/browsers/chrome.ts +++ b/packages/server/lib/browsers/chrome.ts @@ -445,7 +445,9 @@ export = { const browserCriClient = this._getBrowserCriClient() // Handle chrome tab crashes. + debug('attaching crash handler to target ', pageCriClient.targetId) pageCriClient.on('Target.targetCrashed', async (event) => { + debug('target crashed!', event) if (event.targetId !== browserCriClient?.currentlyAttachedTarget?.targetId) { return } diff --git a/packages/server/lib/browsers/cri-client.ts b/packages/server/lib/browsers/cri-client.ts index bf2717372cd1..bf93cf7a275b 100644 --- a/packages/server/lib/browsers/cri-client.ts +++ b/packages/server/lib/browsers/cri-client.ts @@ -155,7 +155,7 @@ export class CriClient implements ICriClient { debug('attaching crash event listener') this.cdpConnection.on('Target.targetCrashed', async (event) => { - debug('crash event detected') + debug('crash event detected', event) if (event.targetId !== this.targetId) { return } @@ -286,6 +286,10 @@ export class CriClient implements ICriClient { } public on = (eventName: T, cb: CDPListener) => { + if (eventName === 'Target.targetCrashed') { + debug('attaching crash listener') + } + this.cdpConnection?.on(eventName, cb) this.subscriptions.push({ eventName, cb }) diff --git a/packages/server/lib/modes/run.ts b/packages/server/lib/modes/run.ts index 66a626057e64..3dd24d2ae741 100644 --- a/packages/server/lib/modes/run.ts +++ b/packages/server/lib/modes/run.ts @@ -500,6 +500,7 @@ async function waitForBrowserToConnect (options: { project: Project, socketId: s telemetry.getSpan(`waitForBrowserToConnect:attempt:${browserLaunchAttempt}`)?.end() }) .catch(Bluebird.TimeoutError, async (err) => { + debug('Catch on waitForBrowserToConnect') telemetry.getSpan(`waitForBrowserToConnect:attempt:${browserLaunchAttempt}`)?.end() console.log('') @@ -936,7 +937,10 @@ async function runSpec (config, spec: SpecWithRelativeRoot, options: { project: project, browser, screenshots, - onError, + onError: (...args) => { + debug('onError from runSpec') + onError(...args) + }, videoRecording, socketId: options.socketId, webSecurity: options.webSecurity, @@ -996,7 +1000,7 @@ async function ready (options: ReadyOptions) { // TODO: refactor this so we don't need to extend options const onError = options.onError = (err) => { - debug('onError') + debug('onError', new Error().stack) earlyExitTerminator.exitEarly(err) } From e998290d92fc64923b36e664d7d256f996dcf9d1 Mon Sep 17 00:00:00 2001 From: Cacie Prins Date: Thu, 8 Aug 2024 16:33:57 -0400 Subject: [PATCH 17/25] do not attach crash handlers on browser level cri clients, add tests --- packages/server/lib/browsers/cri-client.ts | 50 +++++++----- .../test/unit/browsers/cri-client_spec.ts | 76 ++++++++++++++++++- 2 files changed, 106 insertions(+), 20 deletions(-) diff --git a/packages/server/lib/browsers/cri-client.ts b/packages/server/lib/browsers/cri-client.ts index bf93cf7a275b..91270bdfafe3 100644 --- a/packages/server/lib/browsers/cri-client.ts +++ b/packages/server/lib/browsers/cri-client.ts @@ -110,7 +110,9 @@ export class CriClient implements ICriClient { private _closed = false private _connected = false - private crashed = false + private _isChildTarget = false + + private _crashed = false private cdpConnection: CDPConnection private constructor ( @@ -125,6 +127,10 @@ export class CriClient implements ICriClient { onReconnectAttempt?: (retryIndex: number) => void, onCriConnectionClosed?: () => void, ) { + debug('creating cri client with', { + host, port, targetId, + }) + // refactor opportunity: // due to listeners passed in along with connection options, the fns that instantiate this // class should instantiate and listen to the connection directly rather than having this @@ -152,20 +158,24 @@ export class CriClient implements ICriClient { this.cdpConnection.addConnectionEventListener('cdp-connection-reconnect-attempt', onReconnectAttempt) } - debug('attaching crash event listener') + this._isChildTarget = !!this.host - this.cdpConnection.on('Target.targetCrashed', async (event) => { - debug('crash event detected', event) - if (event.targetId !== this.targetId) { - return - } + if (this._isChildTarget) { + // If crash listeners are added at the browser level, tabs/page connections do not + // emit them. + this.cdpConnection.on('Target.targetCrashed', async (event) => { + debug('crash event detected', event) + if (event.targetId !== this.targetId) { + return + } - debug('crash detected') - this.crashed = true - }) + debug('crash detected') + this._crashed = true + }) - if (!!this.host && fullyManageTabs) { - this.cdpConnection.on('Target.attachedToTarget', this._onAttachedToTarget) + if (fullyManageTabs) { + this.cdpConnection.on('Target.attachedToTarget', this._onAttachedToTarget) + } } } @@ -216,6 +226,10 @@ export class CriClient implements ICriClient { return this._connected } + get crashed () { + return this._crashed + } + public connect = async () => { debug('connecting %o', { connected: this._connected, target: this.targetId }) @@ -223,6 +237,12 @@ export class CriClient implements ICriClient { this._connected = true + if (this._isChildTarget) { + // Ideally we could use filter rather than checking the type above, but that was added relatively recently + await this.cdpConnection.send('Target.setAutoAttach', { autoAttach: true, waitForDebuggerOnStart: true, flatten: true }) + await this.cdpConnection.send('Target.setDiscoverTargets', { discover: true }) + } + debug('connected %o', { connected: this._connected, target: this.targetId }) } @@ -231,7 +251,7 @@ export class CriClient implements ICriClient { params?: CmdParams, sessionId?: string, ): Promise => { - if (this.crashed) { + if (this._crashed) { return Promise.reject(new Error(`${command} will not run as the target browser or tab CRI connection has crashed`)) } @@ -356,10 +376,6 @@ export class CriClient implements ICriClient { // it's possible that the target was closed before we could enable network and continue, in that case, just ignore debug('error attaching to target cri', error) } - - // Ideally we could use filter rather than checking the type above, but that was added relatively recently - await this.cdpConnection.send('Target.setAutoAttach', { autoAttach: true, waitForDebuggerOnStart: true, flatten: true }) - await this.cdpConnection.send('Target.setDiscoverTargets', { discover: true }) } private _enqueueCommand ( diff --git a/packages/server/test/unit/browsers/cri-client_spec.ts b/packages/server/test/unit/browsers/cri-client_spec.ts index da6828160e09..424d6776fcda 100644 --- a/packages/server/test/unit/browsers/cri-client_spec.ts +++ b/packages/server/test/unit/browsers/cri-client_spec.ts @@ -60,12 +60,12 @@ describe('lib/browsers/cri-client', function () { criImport.New = sinon.stub().withArgs({ host: HOST, port: PORT, url: 'about:blank' }).resolves({ webSocketDebuggerUrl: 'http://web/socket/url' }) - const { CDPConnection } = proxyquire('../lib/browsers/cdp-connection', { + const CDPConnectionRef = proxyquire('../lib/browsers/cdp-connection', { 'chrome-remote-interface': criImport, - }) + }).CDPConnection const { CriClient } = proxyquire('../lib/browsers/cri-client', { - './cdp-connection': { CDPConnection }, + './cdp-connection': { CDPConnection: CDPConnectionRef }, }) getClient = ({ host, fullyManageTabs, protocolManager } = {}): Promise => { @@ -80,6 +80,76 @@ describe('lib/browsers/cri-client', function () { expect(client.send).to.be.instanceOf(Function) }) + describe('when it has a host', () => { + it('adds a crash listener', async () => { + const client = await getClient({ host: HOST }) + + fireCDPEvent('Target.targetCrashed', { targetId: DEBUGGER_URL }) + expect(client.crashed).to.be.true + }) + }) + + describe('when it does not have a host', () => { + it('does not add a crash listener', async () => { + const client = await getClient() + + fireCDPEvent('Target.targetCrashed', { targetId: DEBUGGER_URL }) + expect(client.crashed).to.be.false + }) + }) + + describe('when it has a host and is fully managed and receives an attachedToTarget event', () => { + beforeEach(async () => { + await getClient({ host: HOST, fullyManageTabs: true }) + criStub.send.resolves() + }) + + describe('target type is service worker, page, or other', async () => { + it('does not enable network', async () => { + await Promise.all(['service_worker', 'page', 'other'].map((type) => { + return fireCDPEvent('Target.attachedToTarget', { + // do not need entire event payload for this test + // @ts-ignore + targetInfo: { + type, + }, + }) + })) + + expect(criStub.send).not.to.have.been.calledWith('Network.enable') + }) + }) + + describe('target type is something other than service worker, page, or other', () => { + it('enables network', async () => { + await fireCDPEvent('Target.attachedToTarget', { + // do not need entire event payload for this test + // @ts-ignore + targetInfo: { + type: 'somethin else', + }, + }) + + expect(criStub.send).to.have.been.calledWith('Network.enable') + }) + }) + + describe('target is waiting for debugger', () => { + it('sends Runtime.runIfWaitingForDebugger', async () => { + const sessionId = 'abc123' + + await fireCDPEvent('Target.attachedToTarget', { + waitingForDebugger: true, + sessionId, + // @ts-ignore + targetInfo: { type: 'service_worker' }, + }) + + expect(criStub.send).to.have.been.calledWith('Runtime.runIfWaitingForDebugger', undefined, sessionId) + }) + }) + }) + context('#send', function () { it('calls cri.send with command and data', async function () { send.resolves() From 9fbebddc7e2f891d56da6e66a69f32dfe50e031f Mon Sep 17 00:00:00 2001 From: Cacie Prins Date: Fri, 9 Aug 2024 16:05:37 -0400 Subject: [PATCH 18/25] adds bindingCalled listeners discretely, rather than relying on 'event', which does not trigger for them --- .../server/lib/browsers/cdp-connection.ts | 29 +++++++++++++++++-- 1 file changed, 26 insertions(+), 3 deletions(-) diff --git a/packages/server/lib/browsers/cdp-connection.ts b/packages/server/lib/browsers/cdp-connection.ts index 778a678c4b82..330c88a71aac 100644 --- a/packages/server/lib/browsers/cdp-connection.ts +++ b/packages/server/lib/browsers/cdp-connection.ts @@ -52,6 +52,7 @@ export class CDPConnection { private _reconnection: Promise | undefined private debug: Debug.Debugger private verboseDebug: Debug.Debugger + private _bindingCalledListeners: { event: string, callback: CDPListener}[] = [] constructor (private readonly _options: CDP.Options, connectionOptions: CDPConnectionOptions) { this._autoReconnect = connectionOptions.automaticallyReconnect @@ -70,14 +71,29 @@ export class CDPConnection { on (event: T, callback: CDPListener) { this.debug('attaching event listener to cdp connection', event) - this._emitter.on(event, callback) + + if (event.startsWith('Runtime.bindingCalled')) { + // these events do not get emitted by cdp clients primary 'event' hook, and must be + // registered separately + this._connection?.on(event, callback) + this._bindingCalledListeners.push({ event, callback }) + } else { + this._emitter.on(event, callback) + } } addConnectionEventListener (event: T, callback: CDPConnectionEventListener) { this.debug('adding connection event listener for ', event) this._emitter.on(event, callback) } off (event: T, callback: CDPListener) { - this._emitter.off(event, callback) + if (event.startsWith('Runtime.bindingCalled')) { + this._connection?.off(event, callback) + this._bindingCalledListeners = this._bindingCalledListeners.filter((pair) => { + return pair.callback !== callback && pair.event !== event + }) + } else { + this._emitter.off(event, callback) + } } removeConnectionEventListener (event: T, callback: CDPConnectionEventListener) { this._emitter.off(event, callback) @@ -120,6 +136,14 @@ export class CDPConnection { private _gracefullyDisconnect = async () => { this._connection?.off('event', this._broadcastEvent) this._connection?.off('disconnect', this._reconnect) + while (this._bindingCalledListeners.length) { + const pair = this._bindingCalledListeners.pop() + + if (pair) { + this._connection?.off(pair.event, pair.callback) + } + } + await this._connection?.close() this._connection = undefined } @@ -129,7 +153,6 @@ export class CDPConnection { data?: ProtocolMapping.Commands[T]['paramsType'][0], sessionId?: string, ): Promise { - this.debug('preparing to send CDP command:', command) if (this.terminated) { throw new CDPDisconnectedError(`${command} will not run as the CRI connection to Target ${this._options.target} has been terminated.`) } From 4855120d4278f7c57c756890c8d6d42633282dc6 Mon Sep 17 00:00:00 2001 From: Cacie Prins Date: Mon, 12 Aug 2024 16:41:20 -0400 Subject: [PATCH 19/25] back out of main `event` listener, use discrete listeners, add integration test for service worker bindingCalled --- .../server/lib/browsers/browser-cri-client.ts | 7 ++- .../server/lib/browsers/cdp-connection.ts | 44 ++++++------------- packages/server/test/integration/cdp_spec.ts | 43 ++++++++++++++++++ 3 files changed, 62 insertions(+), 32 deletions(-) diff --git a/packages/server/lib/browsers/browser-cri-client.ts b/packages/server/lib/browsers/browser-cri-client.ts index 8aadb945d790..940d18b22f02 100644 --- a/packages/server/lib/browsers/browser-cri-client.ts +++ b/packages/server/lib/browsers/browser-cri-client.ts @@ -323,7 +323,12 @@ export class BrowserCriClient { try { // attach a binding to the runtime so that we can listen for service worker events if (event.targetInfo.type === 'service_worker') { - browserClient.on(`Runtime.bindingCalled.${event.sessionId}` as 'Runtime.bindingCalled', serviceWorkerClientEventHandler(browserCriClient.onServiceWorkerClientEvent)) + debug(`adding Runtime.bindingCalled.${event.sessionId} callback`) + browserClient.on(`Runtime.bindingCalled.${event.sessionId}` as 'Runtime.bindingCalled', (payload, sessionId) => { + debug('Runtime.bindinCalled.%s %o', sessionId, payload) + serviceWorkerClientEventHandler(browserCriClient.onServiceWorkerClientEvent)(payload) + }) + await browserClient.send('Runtime.addBinding', { name: serviceWorkerClientEventHandlerName }, event.sessionId) } } catch (error) { diff --git a/packages/server/lib/browsers/cdp-connection.ts b/packages/server/lib/browsers/cdp-connection.ts index 330c88a71aac..4c925c10774c 100644 --- a/packages/server/lib/browsers/cdp-connection.ts +++ b/packages/server/lib/browsers/cdp-connection.ts @@ -52,7 +52,7 @@ export class CDPConnection { private _reconnection: Promise | undefined private debug: Debug.Debugger private verboseDebug: Debug.Debugger - private _bindingCalledListeners: { event: string, callback: CDPListener}[] = [] + private _eventListeners: { event: string, callback: CDPListener}[] = [] constructor (private readonly _options: CDP.Options, connectionOptions: CDPConnectionOptions) { this._autoReconnect = connectionOptions.automaticallyReconnect @@ -72,28 +72,18 @@ export class CDPConnection { on (event: T, callback: CDPListener) { this.debug('attaching event listener to cdp connection', event) - if (event.startsWith('Runtime.bindingCalled')) { - // these events do not get emitted by cdp clients primary 'event' hook, and must be - // registered separately - this._connection?.on(event, callback) - this._bindingCalledListeners.push({ event, callback }) - } else { - this._emitter.on(event, callback) - } + this._connection?.on(event, callback) + this._eventListeners.push({ event, callback }) } addConnectionEventListener (event: T, callback: CDPConnectionEventListener) { this.debug('adding connection event listener for ', event) this._emitter.on(event, callback) } off (event: T, callback: CDPListener) { - if (event.startsWith('Runtime.bindingCalled')) { - this._connection?.off(event, callback) - this._bindingCalledListeners = this._bindingCalledListeners.filter((pair) => { - return pair.callback !== callback && pair.event !== event - }) - } else { - this._emitter.off(event, callback) - } + this._connection?.off(event, callback) + this._eventListeners = this._eventListeners.filter((pair) => { + return pair.callback !== callback && pair.event !== event + }) } removeConnectionEventListener (event: T, callback: CDPConnectionEventListener) { this._emitter.off(event, callback) @@ -110,9 +100,11 @@ export class CDPConnection { this._connection = await CDP(this._options) as CdpClient - debugCdpConnection(this.verboseDebug.namespace, this._connection as DebuggableCDPClient) + this._eventListeners.forEach(({ event, callback }) => { + this._connection?.on(event, callback) + }) - this._connection.on('event', this._broadcastEvent) + debugCdpConnection(this.verboseDebug.namespace, this._connection as DebuggableCDPClient) if (this._autoReconnect) { this._connection.on('disconnect', this._reconnect) @@ -134,10 +126,9 @@ export class CDPConnection { } private _gracefullyDisconnect = async () => { - this._connection?.off('event', this._broadcastEvent) this._connection?.off('disconnect', this._reconnect) - while (this._bindingCalledListeners.length) { - const pair = this._bindingCalledListeners.pop() + while (this._eventListeners.length) { + const pair = this._eventListeners.pop() if (pair) { this._connection?.off(pair.event, pair.callback) @@ -244,13 +235,4 @@ export class CDPConnection { this._reconnection = undefined } - - private _broadcastEvent = ({ method, params, sessionId }: { method: CdpEvent, params: Record, sessionId?: string }) => { - this.verboseDebug('rebroadcasting event', method, params, sessionId) - if (method === 'Target.targetCrashed') { - this.debug('broadcasting crash event') - } - - this._emitter.emit(method, params, sessionId) - } } diff --git a/packages/server/test/integration/cdp_spec.ts b/packages/server/test/integration/cdp_spec.ts index 5baa2c7b0e58..ebe58ddda197 100644 --- a/packages/server/test/integration/cdp_spec.ts +++ b/packages/server/test/integration/cdp_spec.ts @@ -31,6 +31,7 @@ describe('CDP Clients', () => { let onMessage: sinon.SinonStub let messageResponse: ReturnType | undefined let neverAck: boolean + let webSocket: WebSocket const startWsServer = async (onConnection?: OnWSConnection): Promise => { return new Promise((resolve, reject) => { @@ -39,6 +40,7 @@ describe('CDP Clients', () => { }) srv.on('connection', (ws) => { + webSocket = ws if (onConnection) { onConnection(ws) } @@ -435,4 +437,45 @@ describe('CDP Clients', () => { }) }) }) + + /** + * Service worker bindings emit a Runtime.bindingCalled.${session_id} event; + * this event is integral to request correlation, but the `method.session_id` pattern + * is not emitted via the 'event' event + */ + context('nonstandard event listeners', () => { + it('calls the callback when emitted', async () => { + const deferred = pDefer() + const sessionId = 'abc123' + const bindingCalledCallback = sinon.stub().callsFake(deferred.resolve) + + criClient = await CriClient.create({ + target: `ws://127.0.0.1:${wsServerPort}`, + onAsynchronousError: deferred.reject.bind(deferred), + }) + + criClient.on(`Runtime.bindingCalled.${sessionId}` as 'Runtime.bindingCalled', bindingCalledCallback) + + await new Promise((resolve, reject) => { + webSocket.send(JSON.stringify({ + method: 'Runtime.bindingCalled', + params: { + name: '__cypressServiceWorkerClientEvent', + payload: {}, + executionContextId: 1, + }, + sessionId, + }), (err) => { + if (err) { + reject(err) + } else { + resolve() + } + }) + }) + + await (deferred.promise) + expect(bindingCalledCallback).to.have.been.called + }) + }) }) From a5c0fc47cdee725295eb384943747d0590fa5e6c Mon Sep 17 00:00:00 2001 From: Cacie Prins Date: Tue, 13 Aug 2024 09:01:17 -0400 Subject: [PATCH 20/25] Revert "back out of main `event` listener, use discrete listeners, add integration test for service worker bindingCalled" This reverts commit 4855120d4278f7c57c756890c8d6d42633282dc6. --- .../server/lib/browsers/browser-cri-client.ts | 7 +-- .../server/lib/browsers/cdp-connection.ts | 44 +++++++++++++------ packages/server/test/integration/cdp_spec.ts | 43 ------------------ 3 files changed, 32 insertions(+), 62 deletions(-) diff --git a/packages/server/lib/browsers/browser-cri-client.ts b/packages/server/lib/browsers/browser-cri-client.ts index 940d18b22f02..8aadb945d790 100644 --- a/packages/server/lib/browsers/browser-cri-client.ts +++ b/packages/server/lib/browsers/browser-cri-client.ts @@ -323,12 +323,7 @@ export class BrowserCriClient { try { // attach a binding to the runtime so that we can listen for service worker events if (event.targetInfo.type === 'service_worker') { - debug(`adding Runtime.bindingCalled.${event.sessionId} callback`) - browserClient.on(`Runtime.bindingCalled.${event.sessionId}` as 'Runtime.bindingCalled', (payload, sessionId) => { - debug('Runtime.bindinCalled.%s %o', sessionId, payload) - serviceWorkerClientEventHandler(browserCriClient.onServiceWorkerClientEvent)(payload) - }) - + browserClient.on(`Runtime.bindingCalled.${event.sessionId}` as 'Runtime.bindingCalled', serviceWorkerClientEventHandler(browserCriClient.onServiceWorkerClientEvent)) await browserClient.send('Runtime.addBinding', { name: serviceWorkerClientEventHandlerName }, event.sessionId) } } catch (error) { diff --git a/packages/server/lib/browsers/cdp-connection.ts b/packages/server/lib/browsers/cdp-connection.ts index 4c925c10774c..330c88a71aac 100644 --- a/packages/server/lib/browsers/cdp-connection.ts +++ b/packages/server/lib/browsers/cdp-connection.ts @@ -52,7 +52,7 @@ export class CDPConnection { private _reconnection: Promise | undefined private debug: Debug.Debugger private verboseDebug: Debug.Debugger - private _eventListeners: { event: string, callback: CDPListener}[] = [] + private _bindingCalledListeners: { event: string, callback: CDPListener}[] = [] constructor (private readonly _options: CDP.Options, connectionOptions: CDPConnectionOptions) { this._autoReconnect = connectionOptions.automaticallyReconnect @@ -72,18 +72,28 @@ export class CDPConnection { on (event: T, callback: CDPListener) { this.debug('attaching event listener to cdp connection', event) - this._connection?.on(event, callback) - this._eventListeners.push({ event, callback }) + if (event.startsWith('Runtime.bindingCalled')) { + // these events do not get emitted by cdp clients primary 'event' hook, and must be + // registered separately + this._connection?.on(event, callback) + this._bindingCalledListeners.push({ event, callback }) + } else { + this._emitter.on(event, callback) + } } addConnectionEventListener (event: T, callback: CDPConnectionEventListener) { this.debug('adding connection event listener for ', event) this._emitter.on(event, callback) } off (event: T, callback: CDPListener) { - this._connection?.off(event, callback) - this._eventListeners = this._eventListeners.filter((pair) => { - return pair.callback !== callback && pair.event !== event - }) + if (event.startsWith('Runtime.bindingCalled')) { + this._connection?.off(event, callback) + this._bindingCalledListeners = this._bindingCalledListeners.filter((pair) => { + return pair.callback !== callback && pair.event !== event + }) + } else { + this._emitter.off(event, callback) + } } removeConnectionEventListener (event: T, callback: CDPConnectionEventListener) { this._emitter.off(event, callback) @@ -100,12 +110,10 @@ export class CDPConnection { this._connection = await CDP(this._options) as CdpClient - this._eventListeners.forEach(({ event, callback }) => { - this._connection?.on(event, callback) - }) - debugCdpConnection(this.verboseDebug.namespace, this._connection as DebuggableCDPClient) + this._connection.on('event', this._broadcastEvent) + if (this._autoReconnect) { this._connection.on('disconnect', this._reconnect) } @@ -126,9 +134,10 @@ export class CDPConnection { } private _gracefullyDisconnect = async () => { + this._connection?.off('event', this._broadcastEvent) this._connection?.off('disconnect', this._reconnect) - while (this._eventListeners.length) { - const pair = this._eventListeners.pop() + while (this._bindingCalledListeners.length) { + const pair = this._bindingCalledListeners.pop() if (pair) { this._connection?.off(pair.event, pair.callback) @@ -235,4 +244,13 @@ export class CDPConnection { this._reconnection = undefined } + + private _broadcastEvent = ({ method, params, sessionId }: { method: CdpEvent, params: Record, sessionId?: string }) => { + this.verboseDebug('rebroadcasting event', method, params, sessionId) + if (method === 'Target.targetCrashed') { + this.debug('broadcasting crash event') + } + + this._emitter.emit(method, params, sessionId) + } } diff --git a/packages/server/test/integration/cdp_spec.ts b/packages/server/test/integration/cdp_spec.ts index ebe58ddda197..5baa2c7b0e58 100644 --- a/packages/server/test/integration/cdp_spec.ts +++ b/packages/server/test/integration/cdp_spec.ts @@ -31,7 +31,6 @@ describe('CDP Clients', () => { let onMessage: sinon.SinonStub let messageResponse: ReturnType | undefined let neverAck: boolean - let webSocket: WebSocket const startWsServer = async (onConnection?: OnWSConnection): Promise => { return new Promise((resolve, reject) => { @@ -40,7 +39,6 @@ describe('CDP Clients', () => { }) srv.on('connection', (ws) => { - webSocket = ws if (onConnection) { onConnection(ws) } @@ -437,45 +435,4 @@ describe('CDP Clients', () => { }) }) }) - - /** - * Service worker bindings emit a Runtime.bindingCalled.${session_id} event; - * this event is integral to request correlation, but the `method.session_id` pattern - * is not emitted via the 'event' event - */ - context('nonstandard event listeners', () => { - it('calls the callback when emitted', async () => { - const deferred = pDefer() - const sessionId = 'abc123' - const bindingCalledCallback = sinon.stub().callsFake(deferred.resolve) - - criClient = await CriClient.create({ - target: `ws://127.0.0.1:${wsServerPort}`, - onAsynchronousError: deferred.reject.bind(deferred), - }) - - criClient.on(`Runtime.bindingCalled.${sessionId}` as 'Runtime.bindingCalled', bindingCalledCallback) - - await new Promise((resolve, reject) => { - webSocket.send(JSON.stringify({ - method: 'Runtime.bindingCalled', - params: { - name: '__cypressServiceWorkerClientEvent', - payload: {}, - executionContextId: 1, - }, - sessionId, - }), (err) => { - if (err) { - reject(err) - } else { - resolve() - } - }) - }) - - await (deferred.promise) - expect(bindingCalledCallback).to.have.been.called - }) - }) }) From 4f61f3c914023ca53be14d85eaf119efafd28fc1 Mon Sep 17 00:00:00 2001 From: Cacie Prins Date: Wed, 14 Aug 2024 10:12:05 -0400 Subject: [PATCH 21/25] changelog --- cli/CHANGELOG.md | 8 ++++++-- 1 file changed, 6 insertions(+), 2 deletions(-) diff --git a/cli/CHANGELOG.md b/cli/CHANGELOG.md index e13c7ee1f0c7..15505d5a954c 100644 --- a/cli/CHANGELOG.md +++ b/cli/CHANGELOG.md @@ -1,12 +1,16 @@ -## 13.13.3 +## 13.13.4 -_Released 8/13/2024_ +_Released 8/27/2024 (PENDING)_ **Performance:** - Fixed a potential memory leak when re-connecting to an unintentionally disconnected CDP connection. Fixes [#29744](/~https://github.com/cypress-io/cypress/issues/29744). Addressed in [#29988](/~https://github.com/cypress-io/cypress/pull/29988) +## 13.13.3 + +_Released 8/13/2024_ + **Bugfixes:** - A console error will no longer display in Chrome about a deprecated unload call originating from jQuery. Addressed in [#29944](/~https://github.com/cypress-io/cypress/pull/29944). From 0295d35ed362d4fc3e839e929844e76ba440b40a Mon Sep 17 00:00:00 2001 From: Cacie Prins Date: Wed, 14 Aug 2024 13:20:24 -0400 Subject: [PATCH 22/25] adds integration test for various ways to add cdp event listeners --- .../server/lib/browsers/cdp-connection.ts | 31 ++----------- packages/server/test/integration/cdp_spec.ts | 46 ++++++++++++++++++- 2 files changed, 49 insertions(+), 28 deletions(-) diff --git a/packages/server/lib/browsers/cdp-connection.ts b/packages/server/lib/browsers/cdp-connection.ts index 330c88a71aac..c26a940a79ec 100644 --- a/packages/server/lib/browsers/cdp-connection.ts +++ b/packages/server/lib/browsers/cdp-connection.ts @@ -52,7 +52,6 @@ export class CDPConnection { private _reconnection: Promise | undefined private debug: Debug.Debugger private verboseDebug: Debug.Debugger - private _bindingCalledListeners: { event: string, callback: CDPListener}[] = [] constructor (private readonly _options: CDP.Options, connectionOptions: CDPConnectionOptions) { this._autoReconnect = connectionOptions.automaticallyReconnect @@ -72,28 +71,14 @@ export class CDPConnection { on (event: T, callback: CDPListener) { this.debug('attaching event listener to cdp connection', event) - if (event.startsWith('Runtime.bindingCalled')) { - // these events do not get emitted by cdp clients primary 'event' hook, and must be - // registered separately - this._connection?.on(event, callback) - this._bindingCalledListeners.push({ event, callback }) - } else { - this._emitter.on(event, callback) - } + this._emitter.on(event, callback) } addConnectionEventListener (event: T, callback: CDPConnectionEventListener) { this.debug('adding connection event listener for ', event) this._emitter.on(event, callback) } off (event: T, callback: CDPListener) { - if (event.startsWith('Runtime.bindingCalled')) { - this._connection?.off(event, callback) - this._bindingCalledListeners = this._bindingCalledListeners.filter((pair) => { - return pair.callback !== callback && pair.event !== event - }) - } else { - this._emitter.off(event, callback) - } + this._emitter.off(event, callback) } removeConnectionEventListener (event: T, callback: CDPConnectionEventListener) { this._emitter.off(event, callback) @@ -136,13 +121,6 @@ export class CDPConnection { private _gracefullyDisconnect = async () => { this._connection?.off('event', this._broadcastEvent) this._connection?.off('disconnect', this._reconnect) - while (this._bindingCalledListeners.length) { - const pair = this._bindingCalledListeners.pop() - - if (pair) { - this._connection?.off(pair.event, pair.callback) - } - } await this._connection?.close() this._connection = undefined @@ -247,10 +225,9 @@ export class CDPConnection { private _broadcastEvent = ({ method, params, sessionId }: { method: CdpEvent, params: Record, sessionId?: string }) => { this.verboseDebug('rebroadcasting event', method, params, sessionId) - if (method === 'Target.targetCrashed') { - this.debug('broadcasting crash event') - } + this._emitter.emit('event', { method, params, sessionId }) this._emitter.emit(method, params, sessionId) + this._emitter.emit(`${method}.${sessionId}`, params) } } diff --git a/packages/server/test/integration/cdp_spec.ts b/packages/server/test/integration/cdp_spec.ts index 5baa2c7b0e58..8717800039e6 100644 --- a/packages/server/test/integration/cdp_spec.ts +++ b/packages/server/test/integration/cdp_spec.ts @@ -27,6 +27,7 @@ describe('CDP Clients', () => { let wsSrv: WebSocket.Server let criClient: CriClient + let wsClient: WebSocket let messages: object[] let onMessage: sinon.SinonStub let messageResponse: ReturnType | undefined @@ -109,7 +110,9 @@ describe('CDP Clients', () => { nock.enableNetConnect() - wsSrv = await startWsServer() + wsSrv = await startWsServer((client) => { + wsClient = client + }) }) afterEach(async () => { @@ -118,6 +121,47 @@ describe('CDP Clients', () => { await closeWsServer() }) + it('properly handles various ways to add event listeners', async () => { + const sessionId = 'abc123' + const method = `Network.responseReceived` + const sessionDeferred = pDefer() + const sessionCb = sinon.stub().callsFake(sessionDeferred.resolve) + const eventDeferred = pDefer() + const eventCb = sinon.stub().callsFake(eventDeferred.resolve) + const globalDeferred = pDefer() + const globalCb = sinon.stub().callsFake(globalDeferred.resolve) + const params = { foo: 'bar' } + + criClient = await CriClient.create({ + target: `ws://127.0.0.1:${wsServerPort}`, + onAsynchronousError: (err) => { + sessionDeferred.reject(err) + eventDeferred.reject(err) + globalDeferred.reject(err) + }, + }) + + criClient.on(`${method}.${sessionId}` as CdpEvent, sessionCb) + criClient.on(method, eventCb) + criClient.on('event' as CdpEvent, globalCb) + + wsClient.send(JSON.stringify({ + method, + params, + sessionId, + })) + + await Promise.all([ + sessionDeferred.promise, + eventDeferred.promise, + globalDeferred.promise, + ]) + + expect(sessionCb).to.have.been.calledWith(params) + expect(eventCb).to.have.been.calledWith(params, sessionId) + expect(globalCb).to.have.been.calledWith({ method, params, sessionId }) + }) + context('reconnect after disconnect', () => { it('retries to connect', async () => { const stub = sinon.stub() From c80aec802817da8eb82acbd962d7c859e22558ec Mon Sep 17 00:00:00 2001 From: Cacie Prins Date: Wed, 14 Aug 2024 13:30:42 -0400 Subject: [PATCH 23/25] note in changelog this is a cypress server fix --- cli/CHANGELOG.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/cli/CHANGELOG.md b/cli/CHANGELOG.md index 15505d5a954c..87f04f6547a1 100644 --- a/cli/CHANGELOG.md +++ b/cli/CHANGELOG.md @@ -5,7 +5,7 @@ _Released 8/27/2024 (PENDING)_ **Performance:** -- Fixed a potential memory leak when re-connecting to an unintentionally disconnected CDP connection. Fixes [#29744](/~https://github.com/cypress-io/cypress/issues/29744). Addressed in [#29988](/~https://github.com/cypress-io/cypress/pull/29988) +- Fixed a potential memory leak in Cypress server when re-connecting to an unintentionally disconnected CDP connection. Fixes [#29744](/~https://github.com/cypress-io/cypress/issues/29744). Addressed in [#29988](/~https://github.com/cypress-io/cypress/pull/29988) ## 13.13.3 From 1fb6a52b09d8bd6ee6e245976a196c795d7bbc55 Mon Sep 17 00:00:00 2001 From: Cacie Prins Date: Wed, 14 Aug 2024 15:55:51 -0400 Subject: [PATCH 24/25] ensure enablement promises resolve after reconnect --- packages/server/lib/browsers/cri-client.ts | 20 +++++++++---------- .../test/unit/browsers/cri-client_spec.ts | 19 ++++++++++++++++++ 2 files changed, 28 insertions(+), 11 deletions(-) diff --git a/packages/server/lib/browsers/cri-client.ts b/packages/server/lib/browsers/cri-client.ts index 91270bdfafe3..36e123eb3f12 100644 --- a/packages/server/lib/browsers/cri-client.ts +++ b/packages/server/lib/browsers/cri-client.ts @@ -387,6 +387,7 @@ export class CriClient implements ICriClient { } private _onCdpConnectionReconnect = async () => { + debug('cdp connection reconnected') try { await this._restoreState() await this._drainCommandQueue() @@ -421,11 +422,14 @@ export class CriClient implements ICriClient { } catch (err) { debug('error re-enabling %s: ', command, err) if (CDPDisconnectedError.isCDPDisconnectedError(err)) { - // below comment is no longer accurate - ephemeral connection is wrapped by - // CDPConnection class + // this error is caught in _onCdpConnectionReconnect + // because this is a connection error, the enablement will be re-attempted + // when _onCdpConnectionReconnect is called again. We do need to ensure the + // original in-flight command, if present, is re-enqueued. + if (inFlightCommand) { + this._commandQueue.unshift(inFlightCommand) + } - // Connection errors are thrown here so that a reconnection attempt - // can be made. throw err } else { // non-connection errors are appropriate for rejecting the original command promise @@ -454,13 +458,7 @@ export class CriClient implements ICriClient { } catch (e) { debug('enqueued command %s failed:', enqueued.command, e) if (CDPDisconnectedError.isCDPDisconnectedError(e)) { - // below comment is no longer accurate - ephemeral connection is wrapped by - // CDPConnection class - - // similar to restoring state, connection errors are re-thrown so that - // the connection can be restored. The command is queued for re-delivery - // upon reconnect. - debug('re-enqueuing command and re-throwing') + debug('command failed due to disconnection; enqueuing for resending once reconnected') this._commandQueue.unshift(enqueued) throw e } else { diff --git a/packages/server/test/unit/browsers/cri-client_spec.ts b/packages/server/test/unit/browsers/cri-client_spec.ts index 424d6776fcda..608cfcf2e55b 100644 --- a/packages/server/test/unit/browsers/cri-client_spec.ts +++ b/packages/server/test/unit/browsers/cri-client_spec.ts @@ -224,6 +224,23 @@ describe('lib/browsers/cri-client', function () { await getDocumentPromise expect(send).to.have.callCount(3) }) + + it(`with two '${msg}' message it retries enablements twice`, async () => { + const err = new Error(msg) + + send.onFirstCall().rejects(err) + send.onSecondCall().rejects(err) + send.onThirdCall().resolves() + + const client = await getClient() + + const enableNetworkPromise = client.send('Network.enable') + + await criStub.on.withArgs('disconnect').args[0][1]() + await criStub.on.withArgs('disconnect').args[0][1]() + await enableNetworkPromise + expect(send).to.have.callCount(3) + }) }) }) @@ -290,6 +307,8 @@ describe('lib/browsers/cri-client', function () { expect(criStub.send).to.be.calledWith('Page.enable') expect(criStub.send).to.be.calledWith('Network.enable') expect(protocolManager.cdpReconnect).to.be.called + + await criStub.on.withArgs('disconnect').args[0][1]() }) it('errors if reconnecting fails', async () => { From 6b82a7c0392b73074683dbd27d09059dbffea14f Mon Sep 17 00:00:00 2001 From: Cacie Prins Date: Thu, 15 Aug 2024 10:07:33 -0400 Subject: [PATCH 25/25] Update cli/CHANGELOG.md Co-authored-by: Matt Schile --- cli/CHANGELOG.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/cli/CHANGELOG.md b/cli/CHANGELOG.md index 7afde96fd470..a99d834fa9e3 100644 --- a/cli/CHANGELOG.md +++ b/cli/CHANGELOG.md @@ -5,7 +5,7 @@ _Released 8/27/2024 (PENDING)_ **Performance:** -- Fixed a potential memory leak in Cypress server when re-connecting to an unintentionally disconnected CDP connection. Fixes [#29744](/~https://github.com/cypress-io/cypress/issues/29744). Addressed in [#29988](/~https://github.com/cypress-io/cypress/pull/29988) +- Fixed a potential memory leak in the Cypress server when re-connecting to an unintentionally disconnected CDP connection. Fixes [#29744](/~https://github.com/cypress-io/cypress/issues/29744). Addressed in [#29988](/~https://github.com/cypress-io/cypress/pull/29988) **Dependency Updates:**