"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"]}