230 lines
41 KiB
JavaScript
230 lines
41 KiB
JavaScript
"use strict";
|
|
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
};
|
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
const lodash_1 = require("lodash");
|
|
const event_provider_1 = __importDefault(require("../request-hooks/event-provider"));
|
|
const resource_injector_1 = __importDefault(require("../resource-injector"));
|
|
const headers_1 = require("../utils/headers");
|
|
const cdp_1 = require("../utils/cdp");
|
|
const error_route_1 = __importDefault(require("../error-route"));
|
|
const debug_loggers_1 = require("../../utils/debug-loggers");
|
|
const testcafe_hammerhead_1 = require("testcafe-hammerhead");
|
|
const default_setup_options_1 = __importDefault(require("../default-setup-options"));
|
|
const special_handlers_1 = __importDefault(require("./special-handlers"));
|
|
const safe_api_1 = require("./safe-api");
|
|
const api_base_1 = __importDefault(require("../api-base"));
|
|
const resendAuthRequest_1 = require("./resendAuthRequest");
|
|
const test_run_bridge_1 = __importDefault(require("./test-run-bridge"));
|
|
const context_info_1 = __importDefault(require("./context-info"));
|
|
class ProxylessRequestPipeline extends api_base_1.default {
|
|
constructor(browserId, client) {
|
|
super(browserId, client);
|
|
this._testRunBridge = new test_run_bridge_1.default(browserId);
|
|
this._contextInfo = new context_info_1.default(this._testRunBridge);
|
|
this._specialServiceRoutes = this._getSpecialServiceRoutes();
|
|
this.requestHookEventProvider = new event_provider_1.default();
|
|
this._resourceInjector = new resource_injector_1.default(this._testRunBridge, this._specialServiceRoutes);
|
|
this._options = default_setup_options_1.default;
|
|
this._stopped = false;
|
|
this._currentFrameTree = null;
|
|
this._failedRequestIds = [];
|
|
this.restoringStorages = null;
|
|
this.contextStorage = null;
|
|
}
|
|
_getSpecialServiceRoutes() {
|
|
const browserConnection = this._testRunBridge.getBrowserConnection();
|
|
const proxy = browserConnection.browserConnectionGateway.proxy;
|
|
return {
|
|
errorPage1: proxy.resolveRelativeServiceUrl(error_route_1.default, proxy.server1Info.domain),
|
|
errorPage2: proxy.resolveRelativeServiceUrl(error_route_1.default, proxy.server2Info.domain),
|
|
idlePage: browserConnection.idleUrl,
|
|
openFileProtocolUrl: browserConnection.openFileProtocolUrl,
|
|
};
|
|
}
|
|
async _handleMockErrorIfNecessary(pipelineContext, event) {
|
|
if (!pipelineContext.mock.hasError)
|
|
return;
|
|
await pipelineContext.handleMockError(this.requestHookEventProvider);
|
|
(0, debug_loggers_1.requestPipelineMockLogger)('%s\n%s', event.networkId, pipelineContext.mock.error);
|
|
}
|
|
async _handleMockResponse(mockedResponse, pipelineContext, event) {
|
|
const mockedResponseBodyStr = mockedResponse.getBody().toString();
|
|
const fulfillInfo = {
|
|
requestId: event.requestId,
|
|
responseCode: mockedResponse.statusCode,
|
|
responseHeaders: (0, headers_1.convertToHeaderEntries)(mockedResponse.headers),
|
|
body: mockedResponseBodyStr,
|
|
};
|
|
if (pipelineContext.reqOpts.isAjax)
|
|
await this._resourceInjector.processNonProxiedContent(fulfillInfo, this._client);
|
|
else {
|
|
await this._resourceInjector.processHTMLPageContent(fulfillInfo, {
|
|
isIframe: false,
|
|
contextStorage: this.contextStorage,
|
|
}, this._client);
|
|
}
|
|
(0, debug_loggers_1.requestPipelineMockLogger)(`Mock request ${event.requestId}`);
|
|
}
|
|
_createContinueResponseRequest(event, modified) {
|
|
const continueResponseRequest = {
|
|
requestId: event.requestId,
|
|
};
|
|
if (modified) {
|
|
continueResponseRequest.responseHeaders = event.responseHeaders;
|
|
continueResponseRequest.responseCode = event.responseStatusCode;
|
|
}
|
|
return continueResponseRequest;
|
|
}
|
|
_shouldRedirectToErrorPage(event) {
|
|
return event.resourceType === 'Document'
|
|
&& !this._isIframe(event.frameId);
|
|
}
|
|
async _getUserScripts(event) {
|
|
const { pipelineContext, eventFactory } = this._contextInfo.getContextData(event);
|
|
await pipelineContext.prepareInjectableUserScripts(eventFactory, this._testRunBridge.getUserScripts());
|
|
return pipelineContext.injectableUserScripts;
|
|
}
|
|
async _respondToOtherRequest(event) {
|
|
if ((0, testcafe_hammerhead_1.isRedirectStatusCode)(event.responseStatusCode)) {
|
|
await (0, safe_api_1.safeContinueResponse)(this._client, { requestId: event.requestId });
|
|
return;
|
|
}
|
|
const resourceInfo = await this._resourceInjector.getDocumentResourceInfo(event, this._client);
|
|
if (resourceInfo.error) {
|
|
if (this._shouldRedirectToErrorPage(event)) {
|
|
await this._resourceInjector.redirectToErrorPage(this._client, resourceInfo.error, event.request.url);
|
|
this._contextInfo.dispose((0, cdp_1.getRequestId)(event));
|
|
}
|
|
return;
|
|
}
|
|
const modified = await this.requestHookEventProvider.onResponse(event, resourceInfo.body, this._contextInfo, this._client);
|
|
if (event.resourceType !== 'Document') {
|
|
const continueResponseRequest = this._createContinueResponseRequest(event, modified);
|
|
await (0, safe_api_1.safeContinueResponse)(this._client, continueResponseRequest);
|
|
this._contextInfo.dispose((0, cdp_1.getRequestId)(event));
|
|
}
|
|
else {
|
|
const fulfillInfo = {
|
|
requestId: event.requestId,
|
|
responseHeaders: event.responseHeaders,
|
|
responseCode: event.responseStatusCode,
|
|
body: resourceInfo.body.toString(),
|
|
};
|
|
// NOTE: Strange behavior of the CDP API:
|
|
// if we pass the empty "responseStatusText" value, we get an error 'Invalid status code or phrase'.
|
|
if (event.responseStatusText !== '')
|
|
fulfillInfo.responsePhrase = event.responseStatusText;
|
|
if ((0, cdp_1.isUnauthorized)(event.responseStatusCode))
|
|
await this._tryAuthorizeWithHttpBasicAuthCredentials(event, fulfillInfo);
|
|
const userScripts = await this._getUserScripts(event);
|
|
await this._resourceInjector.processHTMLPageContent(fulfillInfo, {
|
|
isIframe: this._isIframe(event.frameId),
|
|
url: event.request.url,
|
|
restoringStorages: this.restoringStorages,
|
|
contextStorage: this.contextStorage,
|
|
userScripts,
|
|
}, this._client);
|
|
this._contextInfo.dispose((0, cdp_1.getRequestId)(event));
|
|
this.restoringStorages = null;
|
|
}
|
|
}
|
|
async _tryAuthorizeWithHttpBasicAuthCredentials(event, fulfillInfo) {
|
|
const credentials = this._testRun.getAuthCredentials();
|
|
if (!credentials)
|
|
return;
|
|
const authRequest = await (0, resendAuthRequest_1.resendAuthRequest)(event.request, credentials);
|
|
if (typeof authRequest !== 'string' && !(0, cdp_1.isUnauthorized)(authRequest.status)) {
|
|
fulfillInfo.responseCode = authRequest.status;
|
|
fulfillInfo.body = authRequest.body.toString();
|
|
fulfillInfo.responsePhrase = authRequest.statusText;
|
|
}
|
|
}
|
|
async _tryRespondToOtherRequest(event) {
|
|
try {
|
|
await this._respondToOtherRequest(event);
|
|
}
|
|
catch (err) {
|
|
if (event.networkId && this._failedRequestIds.includes(event.networkId)) {
|
|
(0, lodash_1.remove)(this._failedRequestIds, event.networkId);
|
|
return;
|
|
}
|
|
throw err;
|
|
}
|
|
}
|
|
async _handleOtherRequests(event) {
|
|
(0, debug_loggers_1.requestPipelineOtherRequestLogger)('%r', event);
|
|
if (!event.responseErrorReason && ((0, cdp_1.isRequest)(event) || (0, testcafe_hammerhead_1.isRedirectStatusCode)(event.responseStatusCode))) {
|
|
this._contextInfo.init(event);
|
|
await this.requestHookEventProvider.onRequest(event, this._contextInfo);
|
|
const pipelineContext = this._contextInfo.getPipelineContext(event.networkId);
|
|
if (!pipelineContext || !pipelineContext.mock)
|
|
await (0, safe_api_1.safeContinueRequest)(this._client, event);
|
|
else {
|
|
const mockedResponse = await pipelineContext.getMockResponse();
|
|
await this._handleMockErrorIfNecessary(pipelineContext, event);
|
|
const mockedResponseEvent = (0, cdp_1.createRequestPausedEventForResponse)(mockedResponse, event);
|
|
await this.requestHookEventProvider.onResponse(mockedResponseEvent, mockedResponse.getBody(), this._contextInfo, this._client);
|
|
await this._handleMockResponse(mockedResponse, pipelineContext, event);
|
|
this._contextInfo.dispose((0, cdp_1.getRequestId)(event));
|
|
}
|
|
}
|
|
else
|
|
await this._tryRespondToOtherRequest(event);
|
|
}
|
|
_topFrameNavigation(event) {
|
|
return event.type === 'Navigation'
|
|
&& !event.frame.parentId;
|
|
}
|
|
async _updateCurrentFrameTree() {
|
|
// NOTE: Due to CDP restrictions (it hangs), we can't get the frame tree
|
|
// right before injecting service scripts.
|
|
// So, we are forced tracking frames tree.
|
|
const result = await this._client.Page.getFrameTree();
|
|
this._currentFrameTree = result.frameTree;
|
|
}
|
|
_isIframe(frameId) {
|
|
if (!this._currentFrameTree)
|
|
return false;
|
|
return this._currentFrameTree.frame.id !== frameId;
|
|
}
|
|
async init(options) {
|
|
this._options = options;
|
|
this._client.Fetch.on('requestPaused', async (event) => {
|
|
if (this._stopped)
|
|
return;
|
|
const specialRequestHandler = (0, special_handlers_1.default)(event, this._options, this._specialServiceRoutes);
|
|
if (specialRequestHandler)
|
|
await specialRequestHandler(event, this._client, this._options);
|
|
else
|
|
await this._handleOtherRequests(event);
|
|
});
|
|
this._client.Page.on('frameNavigated', async (event) => {
|
|
(0, debug_loggers_1.requestPipelineLogger)('%f', event);
|
|
if (!this._topFrameNavigation(event)
|
|
|| event.frame.url !== testcafe_hammerhead_1.SPECIAL_BLANK_PAGE)
|
|
return;
|
|
this._contextInfo.init(event);
|
|
const userScripts = await this._getUserScripts(event);
|
|
await this._resourceInjector.processAboutBlankPage(event, userScripts, this._client);
|
|
this._contextInfo.dispose((0, cdp_1.getRequestId)(event));
|
|
});
|
|
this._client.Page.on('frameStartedLoading', async () => {
|
|
await this._updateCurrentFrameTree();
|
|
});
|
|
this._client.Network.on('loadingFailed', async (event) => {
|
|
(0, debug_loggers_1.requestPipelineLogger)('%l', event);
|
|
this._failedRequestIds.push(event.requestId);
|
|
if (event.requestId)
|
|
this._contextInfo.dispose(event.requestId);
|
|
});
|
|
await this._client.Page.setBypassCSP({ enabled: true });
|
|
}
|
|
stop() {
|
|
this._stopped = true;
|
|
}
|
|
}
|
|
exports.default = ProxylessRequestPipeline;
|
|
module.exports = exports.default;
|
|
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/proxyless/request-pipeline/index.ts"],"names":[],"mappings":"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","sourcesContent":["import { remove } from 'lodash';\nimport { ProtocolApi } from 'chrome-remote-interface';\nimport Protocol from 'devtools-protocol';\nimport RequestPausedEvent = Protocol.Fetch.RequestPausedEvent;\nimport FrameNavigatedEvent = Protocol.Page.FrameNavigatedEvent;\nimport LoadingFailedEvent = Protocol.Network.LoadingFailedEvent;\nimport ContinueResponseRequest = Protocol.Fetch.ContinueResponseRequest;\nimport FrameTree = Protocol.Page.FrameTree;\nimport FulfillRequestRequest = Protocol.Fetch.FulfillRequestRequest;\nimport ProxylessRequestHookEventProvider from '../request-hooks/event-provider';\nimport ResourceInjector from '../resource-injector';\nimport { convertToHeaderEntries } from '../utils/headers';\n\nimport {\n    createRequestPausedEventForResponse,\n    getRequestId,\n    isRequest,\n    isUnauthorized,\n} from '../utils/cdp';\n\nimport ERROR_ROUTE from '../error-route';\nimport { SessionStorageInfo, SpecialServiceRoutes } from '../types';\nimport {\n    requestPipelineLogger,\n    requestPipelineMockLogger,\n    requestPipelineOtherRequestLogger,\n} from '../../utils/debug-loggers';\n\nimport {\n    IncomingMessageLike,\n    isRedirectStatusCode,\n    SPECIAL_BLANK_PAGE,\n    StoragesSnapshot,\n} from 'testcafe-hammerhead';\n\nimport ProxylessPipelineContext from '../request-hooks/pipeline-context';\nimport { ProxylessSetupOptions } from '../../shared/types';\nimport DEFAULT_PROXYLESS_SETUP_OPTIONS from '../default-setup-options';\nimport getSpecialRequestHandler from './special-handlers';\nimport { safeContinueRequest, safeContinueResponse } from './safe-api';\nimport ProxylessApiBase from '../api-base';\nimport { resendAuthRequest } from './resendAuthRequest';\nimport TestRunBridge from './test-run-bridge';\nimport ProxylessRequestContextInfo from './context-info';\n\n\nexport default class ProxylessRequestPipeline extends ProxylessApiBase {\n    private readonly _testRunBridge: TestRunBridge;\n    private readonly _contextInfo: ProxylessRequestContextInfo;\n    public readonly requestHookEventProvider: ProxylessRequestHookEventProvider;\n    public restoringStorages: StoragesSnapshot | null;\n    public contextStorage: SessionStorageInfo | null;\n    private readonly _resourceInjector: ResourceInjector;\n    private _options: ProxylessSetupOptions;\n    private readonly _specialServiceRoutes: SpecialServiceRoutes;\n    private _stopped: boolean;\n    private _currentFrameTree: FrameTree | null;\n    private readonly _failedRequestIds: string[];\n\n    public constructor (browserId: string, client: ProtocolApi) {\n        super(browserId, client);\n\n        this._testRunBridge           = new TestRunBridge(browserId);\n        this._contextInfo             = new ProxylessRequestContextInfo(this._testRunBridge);\n        this._specialServiceRoutes    = this._getSpecialServiceRoutes();\n        this.requestHookEventProvider = new ProxylessRequestHookEventProvider();\n        this._resourceInjector        = new ResourceInjector(this._testRunBridge, this._specialServiceRoutes);\n        this._options                 = DEFAULT_PROXYLESS_SETUP_OPTIONS;\n        this._stopped                 = false;\n        this._currentFrameTree        = null;\n        this._failedRequestIds        = [];\n        this.restoringStorages        = null;\n        this.contextStorage           = null;\n    }\n\n    private _getSpecialServiceRoutes (): SpecialServiceRoutes {\n        const browserConnection = this._testRunBridge.getBrowserConnection();\n        const proxy             = browserConnection.browserConnectionGateway.proxy;\n\n        return {\n            errorPage1:          proxy.resolveRelativeServiceUrl(ERROR_ROUTE, proxy.server1Info.domain),\n            errorPage2:          proxy.resolveRelativeServiceUrl(ERROR_ROUTE, proxy.server2Info.domain),\n            idlePage:            browserConnection.idleUrl,\n            openFileProtocolUrl: browserConnection.openFileProtocolUrl,\n        };\n    }\n\n    private async _handleMockErrorIfNecessary (pipelineContext: ProxylessPipelineContext, event: RequestPausedEvent): Promise<void> {\n        if (!pipelineContext.mock.hasError)\n            return;\n\n        await pipelineContext.handleMockError(this.requestHookEventProvider);\n\n        requestPipelineMockLogger('%s\\n%s', event.networkId, pipelineContext.mock.error);\n    }\n\n    private async _handleMockResponse (mockedResponse: IncomingMessageLike, pipelineContext: ProxylessPipelineContext, event: RequestPausedEvent): Promise<void> {\n        const mockedResponseBodyStr = (mockedResponse.getBody() as Buffer).toString();\n\n        const fulfillInfo = {\n            requestId:       event.requestId,\n            responseCode:    mockedResponse.statusCode,\n            responseHeaders: convertToHeaderEntries(mockedResponse.headers),\n            body:            mockedResponseBodyStr,\n        };\n\n        if (pipelineContext.reqOpts.isAjax)\n            await this._resourceInjector.processNonProxiedContent(fulfillInfo, this._client);\n        else {\n            await this._resourceInjector.processHTMLPageContent(fulfillInfo, {\n                isIframe:       false,\n                contextStorage: this.contextStorage,\n            }, this._client);\n        }\n\n        requestPipelineMockLogger(`Mock request ${event.requestId}`);\n    }\n\n    private _createContinueResponseRequest (event: RequestPausedEvent, modified: boolean): ContinueResponseRequest {\n        const continueResponseRequest = {\n            requestId: event.requestId,\n        } as ContinueResponseRequest;\n\n        if (modified) {\n            continueResponseRequest.responseHeaders = event.responseHeaders;\n            continueResponseRequest.responseCode    = event.responseStatusCode as number;\n        }\n\n        return continueResponseRequest;\n    }\n\n    private _shouldRedirectToErrorPage (event: RequestPausedEvent): boolean {\n        return event.resourceType === 'Document'\n            && !this._isIframe(event.frameId);\n    }\n\n    private async _getUserScripts (event: RequestPausedEvent | FrameNavigatedEvent): Promise<string[]> {\n        const { pipelineContext, eventFactory } = this._contextInfo.getContextData(event);\n\n        await pipelineContext.prepareInjectableUserScripts(eventFactory, this._testRunBridge.getUserScripts());\n\n        return pipelineContext.injectableUserScripts;\n    }\n\n    private async _respondToOtherRequest (event: RequestPausedEvent): Promise<void> {\n        if (isRedirectStatusCode(event.responseStatusCode)) {\n            await safeContinueResponse(this._client, { requestId: event.requestId });\n\n            return;\n        }\n\n        const resourceInfo = await this._resourceInjector.getDocumentResourceInfo(event, this._client);\n\n        if (resourceInfo.error) {\n            if (this._shouldRedirectToErrorPage(event)) {\n                await this._resourceInjector.redirectToErrorPage(this._client, resourceInfo.error, event.request.url);\n\n                this._contextInfo.dispose(getRequestId(event));\n            }\n\n            return;\n        }\n\n        const modified = await this.requestHookEventProvider.onResponse(event, resourceInfo.body, this._contextInfo, this._client);\n\n        if (event.resourceType !== 'Document') {\n            const continueResponseRequest = this._createContinueResponseRequest(event, modified);\n\n            await safeContinueResponse(this._client, continueResponseRequest);\n\n            this._contextInfo.dispose(getRequestId(event));\n        }\n        else {\n            const fulfillInfo = {\n                requestId:       event.requestId,\n                responseHeaders: event.responseHeaders,\n                responseCode:    event.responseStatusCode as number,\n                body:            (resourceInfo.body as Buffer).toString(),\n            } as FulfillRequestRequest;\n\n            // NOTE: Strange behavior of the CDP API:\n            // if we pass the empty \"responseStatusText\" value, we get an error 'Invalid status code or phrase'.\n            if (event.responseStatusText !== '')\n                fulfillInfo.responsePhrase = event.responseStatusText;\n\n            if (isUnauthorized(event.responseStatusCode as number))\n                await this._tryAuthorizeWithHttpBasicAuthCredentials(event, fulfillInfo);\n\n            const userScripts = await this._getUserScripts(event);\n\n            await this._resourceInjector.processHTMLPageContent(\n                fulfillInfo,\n                {\n                    isIframe:          this._isIframe(event.frameId),\n                    url:               event.request.url,\n                    restoringStorages: this.restoringStorages,\n                    contextStorage:    this.contextStorage,\n                    userScripts,\n                },\n                this._client);\n\n            this._contextInfo.dispose(getRequestId(event));\n\n            this.restoringStorages = null;\n        }\n    }\n\n    private async _tryAuthorizeWithHttpBasicAuthCredentials (event: RequestPausedEvent, fulfillInfo: FulfillRequestRequest): Promise<void> {\n        const credentials = this._testRun.getAuthCredentials();\n\n        if (!credentials)\n            return;\n\n        const authRequest = await resendAuthRequest(event.request, credentials);\n\n        if (typeof authRequest !== 'string' && !isUnauthorized(authRequest.status)) {\n            fulfillInfo.responseCode = authRequest.status;\n            fulfillInfo.body = authRequest.body.toString();\n            fulfillInfo.responsePhrase = authRequest.statusText;\n        }\n    }\n\n    private async _tryRespondToOtherRequest (event: RequestPausedEvent): Promise<void> {\n        try {\n            await this._respondToOtherRequest(event);\n        }\n        catch (err) {\n            if (event.networkId && this._failedRequestIds.includes(event.networkId)) {\n                remove(this._failedRequestIds, event.networkId);\n\n                return;\n            }\n\n            throw err;\n        }\n    }\n\n    private async _handleOtherRequests (event: RequestPausedEvent): Promise<void> {\n        requestPipelineOtherRequestLogger('%r', event);\n\n        if (!event.responseErrorReason && (isRequest(event) || isRedirectStatusCode(event.responseStatusCode))) {\n            this._contextInfo.init(event);\n\n            await this.requestHookEventProvider.onRequest(event, this._contextInfo);\n\n            const pipelineContext = this._contextInfo.getPipelineContext(event.networkId as string);\n\n            if (!pipelineContext || !pipelineContext.mock)\n                await safeContinueRequest(this._client, event);\n            else {\n                const mockedResponse = await pipelineContext.getMockResponse();\n\n                await this._handleMockErrorIfNecessary(pipelineContext, event);\n\n                const mockedResponseEvent = createRequestPausedEventForResponse(mockedResponse, event);\n\n                await this.requestHookEventProvider.onResponse(mockedResponseEvent, mockedResponse.getBody(), this._contextInfo, this._client);\n\n                await this._handleMockResponse(mockedResponse, pipelineContext, event);\n\n                this._contextInfo.dispose(getRequestId(event));\n            }\n        }\n        else\n            await this._tryRespondToOtherRequest(event);\n    }\n\n    private _topFrameNavigation (event: FrameNavigatedEvent): boolean {\n        return event.type === 'Navigation'\n            && !event.frame.parentId;\n    }\n\n    private async _updateCurrentFrameTree (): Promise<void> {\n        // NOTE: Due to CDP restrictions (it hangs), we can't get the frame tree\n        // right before injecting service scripts.\n        // So, we are forced tracking frames tree.\n        const result = await this._client.Page.getFrameTree();\n\n        this._currentFrameTree = result.frameTree;\n    }\n\n    private _isIframe (frameId: string): boolean {\n        if (!this._currentFrameTree)\n            return false;\n\n        return this._currentFrameTree.frame.id !== frameId;\n    }\n\n    public async init (options?: ProxylessSetupOptions): Promise<void> {\n        this._options = options as ProxylessSetupOptions;\n\n        this._client.Fetch.on('requestPaused', async (event: RequestPausedEvent) => {\n            if (this._stopped)\n                return;\n\n            const specialRequestHandler = getSpecialRequestHandler(event, this._options, this._specialServiceRoutes);\n\n            if (specialRequestHandler)\n                await specialRequestHandler(event, this._client, this._options);\n            else\n                await this._handleOtherRequests(event);\n        });\n\n        this._client.Page.on('frameNavigated', async (event: FrameNavigatedEvent) => {\n            requestPipelineLogger('%f', event);\n\n            if (!this._topFrameNavigation(event)\n                || event.frame.url !== SPECIAL_BLANK_PAGE)\n                return;\n\n            this._contextInfo.init(event);\n\n            const userScripts = await this._getUserScripts(event);\n\n            await this._resourceInjector.processAboutBlankPage(event, userScripts, this._client);\n\n            this._contextInfo.dispose(getRequestId(event));\n        });\n\n        this._client.Page.on('frameStartedLoading', async () => {\n            await this._updateCurrentFrameTree();\n        });\n\n        this._client.Network.on('loadingFailed', async (event: LoadingFailedEvent) => {\n            requestPipelineLogger('%l', event);\n\n            this._failedRequestIds.push(event.requestId);\n\n            if (event.requestId)\n                this._contextInfo.dispose(event.requestId);\n        });\n\n        await this._client.Page.setBypassCSP({ enabled: true });\n    }\n\n    public stop (): void {\n        this._stopped = true;\n    }\n}\n"]}
|