UNPKG

testcafe

Version:

Automated browser testing for the modern web development stack.

177 lines (174 loc) 29.5 kB
"use strict"; var __importDefault = (this && this.__importDefault) || function (mod) { return (mod && mod.__esModule) ? mod : { "default": mod }; }; Object.defineProperty(exports, "__esModule", { value: true }); const testcafe_hammerhead_1 = require("testcafe-hammerhead"); const injectables_1 = require("../assets/injectables"); const empty_page_markup_1 = __importDefault(require("./empty-page-markup")); const http_status_codes_1 = require("http-status-codes"); const test_run_1 = require("../errors/test-run"); const cdp_1 = require("./utils/cdp"); const debug_loggers_1 = require("../utils/debug-loggers"); const string_1 = require("./utils/string"); const safe_api_1 = require("./request-pipeline/safe-api"); const errors_1 = require("./errors"); const RESPONSE_REMOVED_HEADERS = [ 'cross-origin-embedder-policy', 'cross-origin-opener-policy', 'cross-origin-resource-policy', ]; const DEFAULT_RESOURCE_INJECTOR_OPTIONS = { specialServiceRoutes: { errorPage1: '', errorPage2: '', openFileProtocolUrl: '', idlePage: '', }, developmentMode: false, }; class ResourceInjector { constructor(testRunBridge) { this._options = DEFAULT_RESOURCE_INJECTOR_OPTIONS; this._testRunBridge = testRunBridge; } _getRestoreContextStorageScript(contextStorage) { const currentTestRun = this._testRunBridge.getCurrentTestRun(); const value = JSON.stringify((contextStorage === null || contextStorage === void 0 ? void 0 : contextStorage[currentTestRun.id]) || ''); return `Object.defineProperty(window, '%nativeAutomationContextStorage%', { configurable: true, value: ${value} });`; } _getRestoreStoragesScript(restoringStorages) { if (!restoringStorages) return '(function() {})()'; return `(function() { window.localStorage.clear(); window.sessionStorage.clear(); const snapshot = ${JSON.stringify(restoringStorages)}; const ls = JSON.parse(snapshot.localStorage); const ss = JSON.parse(snapshot.sessionStorage); for (let i = 0; i < ls[0].length; i++) window.localStorage.setItem(ls[0][i], ls[1][i]); for (let i = 0; i < ss[0].length; i++) window.sessionStorage.setItem(ss[0][i], ss[1][i]); })(); `; } _resolveRelativeUrls(proxy, relativeUrls) { return relativeUrls.map(url => proxy.resolveRelativeServiceUrl(url)); } async _prepareInjectableResources({ isIframe, restoringStorages, contextStorage, userScripts }) { if (!this._testRunBridge.getCurrentTestRun()) return null; const taskScript = await this._testRunBridge.getTaskScript({ isIframe, restoringStorages, contextStorage, userScripts }); const proxy = this._testRunBridge.getBrowserConnection().browserConnectionGateway.proxy; const injectableResources = { stylesheets: [ injectables_1.TESTCAFE_UI_STYLES, ...this._testRunBridge.getInjectableStyles(), ], scripts: [ ...testcafe_hammerhead_1.INJECTABLE_SCRIPTS.map(hs => (0, testcafe_hammerhead_1.getAssetPath)(hs, this._options.developmentMode)), ...injectables_1.SCRIPTS.map(s => (0, testcafe_hammerhead_1.getAssetPath)(s, this._options.developmentMode)), ...this._testRunBridge.getInjectableScripts(), ], embeddedScripts: [this._getRestoreStoragesScript(restoringStorages), this._getRestoreContextStorageScript(contextStorage), taskScript], userScripts: userScripts || [], }; injectableResources.scripts = this._resolveRelativeUrls(proxy, injectableResources.scripts); injectableResources.userScripts = this._resolveRelativeUrls(proxy, injectableResources.userScripts); injectableResources.stylesheets = this._resolveRelativeUrls(proxy, injectableResources.stylesheets); return injectableResources; } _processResponseHeaders(headers) { if (!headers) return []; headers = headers.filter(header => !RESPONSE_REMOVED_HEADERS.includes(header.name.toLowerCase())); return (0, string_1.stringifyHeaderValues)(headers); } async _fulfillRequest(client, fulfillRequestInfo, body, sessionId, contentType) { await (0, safe_api_1.safeFulfillRequest)(client, { requestId: fulfillRequestInfo.requestId, responseCode: fulfillRequestInfo.responseCode || http_status_codes_1.StatusCodes.OK, responsePhrase: fulfillRequestInfo.responsePhrase, responseHeaders: this._processResponseHeaders(fulfillRequestInfo.responseHeaders), body: (0, string_1.toBase64String)(body, contentType), }, sessionId); } async redirectToErrorPage(client, err, url) { const currentTestRun = this._testRunBridge.getCurrentTestRun(); if (!currentTestRun) return; currentTestRun.pendingPageError = new test_run_1.PageLoadError(err, url); await (0, cdp_1.navigateTo)(client, this._options.specialServiceRoutes.errorPage1); } async getDocumentResourceInfo(event, client, contentType) { const { requestId, request, responseErrorReason, resourceType, } = event; if (resourceType !== 'Document') { return { error: null, body: null, }; } try { if (responseErrorReason === 'NameNotResolved') { const err = (0, errors_1.failedToFindDNSError)(request.url); return { error: err, body: null, }; } const responseObj = await client.Fetch.getResponseBody({ requestId }); const responseStr = (0, string_1.getResponseAsString)(responseObj, contentType); return { error: null, body: Buffer.from(responseStr), }; } catch (err) { (0, debug_loggers_1.resourceInjectorLogger)('Failed to process request: %s', request.url); return { error: err, body: null, }; } } async processAboutBlankPage(event, userScripts, contextStorage, client) { (0, debug_loggers_1.resourceInjectorLogger)('Handle page as about:blank. Origin url: %s', event.frame.url); const injectableResources = await this._prepareInjectableResources({ isIframe: false, userScripts, contextStorage }); const html = (0, testcafe_hammerhead_1.injectResources)(empty_page_markup_1.default, injectableResources); await client.Page.setDocumentContent({ frameId: event.frame.id, html, }); } async processHTMLPageContent(fulfillRequestInfo, injectableResourcesOptions, client, sessionId, contentType) { const injectableResources = await this._prepareInjectableResources(injectableResourcesOptions); // NOTE: an unhandled exception interrupts the test execution, // and we are force to redirect manually to the idle page. if (!injectableResources) await (0, cdp_1.redirect)(client, fulfillRequestInfo.requestId, this._options.specialServiceRoutes.idlePage); else { const updatedResponseStr = (0, testcafe_hammerhead_1.injectResources)(fulfillRequestInfo.body, injectableResources, this._getPageInjectableResourcesOptions(injectableResourcesOptions)); await this._fulfillRequest(client, fulfillRequestInfo, updatedResponseStr, sessionId, contentType); } } async processNonProxiedContent(fulfillRequestInfo, client, sessionId) { await this._fulfillRequest(client, fulfillRequestInfo, fulfillRequestInfo.body, sessionId); } _getPageInjectableResourcesOptions(injectableResourcesOptions) { const { url, restoringStorages } = injectableResourcesOptions; if (url && restoringStorages) { return { host: new URL(url).host, sessionId: this._testRunBridge.getSessionId(), }; } return void 0; } setOptions(options) { this._options = options; } } exports.default = ResourceInjector; module.exports = exports.default; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"resource-injector.js","sourceRoot":"","sources":["../../src/native-automation/resource-injector.ts"],"names":[],"mappings":"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","sourcesContent":["import { ProtocolApi } from 'chrome-remote-interface';\nimport Protocol from 'devtools-protocol';\nimport RequestPausedEvent = Protocol.Fetch.RequestPausedEvent;\nimport FrameNavigatedEvent = Protocol.Page.FrameNavigatedEvent;\nimport FulfillRequestRequest = Protocol.Fetch.FulfillRequestRequest;\nimport HeaderEntry = Protocol.Fetch.HeaderEntry;\nimport {\n    injectResources,\n    PageInjectableResources,\n    INJECTABLE_SCRIPTS as HAMMERHEAD_INJECTABLE_SCRIPTS,\n    getAssetPath,\n    PageRestoreStoragesOptions,\n    StoragesSnapshot,\n    Proxy,\n} from 'testcafe-hammerhead';\nimport { SCRIPTS, TESTCAFE_UI_STYLES } from '../assets/injectables';\nimport EMPTY_PAGE_MARKUP from './empty-page-markup';\nimport { StatusCodes } from 'http-status-codes';\nimport { PageLoadError } from '../errors/test-run';\nimport { redirect, navigateTo } from './utils/cdp';\n\nimport {\n    DocumentResourceInfo,\n    InjectableResourcesOptions,\n    SessionId,\n    SessionStorageInfo,\n    SpecialServiceRoutes,\n} from './types';\n\nimport { resourceInjectorLogger } from '../utils/debug-loggers';\nimport {\n    getResponseAsString,\n    stringifyHeaderValues,\n    toBase64String,\n} from './utils/string';\nimport { safeFulfillRequest } from './request-pipeline/safe-api';\nimport TestRunBridge from './request-pipeline/test-run-bridge';\nimport { failedToFindDNSError } from './errors';\n\nconst RESPONSE_REMOVED_HEADERS = [\n    'cross-origin-embedder-policy',\n    'cross-origin-opener-policy',\n    'cross-origin-resource-policy',\n];\n\nexport interface ResourceInjectorOptions {\n    specialServiceRoutes: SpecialServiceRoutes;\n    developmentMode: boolean;\n}\n\nconst DEFAULT_RESOURCE_INJECTOR_OPTIONS = {\n    specialServiceRoutes: {\n        errorPage1:          '',\n        errorPage2:          '',\n        openFileProtocolUrl: '',\n        idlePage:            '',\n    },\n\n    developmentMode: false,\n};\n\nexport default class ResourceInjector {\n    private _options: ResourceInjectorOptions;\n    private readonly _testRunBridge: TestRunBridge;\n\n    public constructor (testRunBridge: TestRunBridge) {\n        this._options       = DEFAULT_RESOURCE_INJECTOR_OPTIONS;\n        this._testRunBridge = testRunBridge;\n    }\n\n    private _getRestoreContextStorageScript (contextStorage?: SessionStorageInfo | null): string {\n        const currentTestRun = this._testRunBridge.getCurrentTestRun();\n        const value = JSON.stringify(contextStorage?.[currentTestRun.id] || '');\n\n        return `Object.defineProperty(window, '%nativeAutomationContextStorage%', { configurable: true, value: ${value} });`;\n    }\n\n    private _getRestoreStoragesScript (restoringStorages: StoragesSnapshot | null | undefined): string {\n        if (!restoringStorages)\n            return '(function() {})()';\n\n        return `(function() {\n            window.localStorage.clear();\n            window.sessionStorage.clear();\n\n            const snapshot = ${JSON.stringify(restoringStorages)};\n            const ls       = JSON.parse(snapshot.localStorage);\n            const ss       = JSON.parse(snapshot.sessionStorage);\n\n            for (let i = 0; i < ls[0].length; i++)\n                window.localStorage.setItem(ls[0][i], ls[1][i]);\n\n            for (let i = 0; i < ss[0].length; i++)\n                window.sessionStorage.setItem(ss[0][i], ss[1][i]);\n        })();\n        `;\n    }\n\n    private _resolveRelativeUrls (proxy: Proxy, relativeUrls: string[]): string[] {\n        return relativeUrls.map(url => proxy.resolveRelativeServiceUrl(url));\n    }\n\n    private async _prepareInjectableResources ({ isIframe, restoringStorages, contextStorage, userScripts }: InjectableResourcesOptions): Promise<PageInjectableResources | null> {\n        if (!this._testRunBridge.getCurrentTestRun())\n            return null;\n\n        const taskScript = await this._testRunBridge.getTaskScript({ isIframe, restoringStorages, contextStorage, userScripts });\n        const proxy      = this._testRunBridge.getBrowserConnection().browserConnectionGateway.proxy;\n\n        const injectableResources = {\n            stylesheets: [\n                TESTCAFE_UI_STYLES,\n                ...this._testRunBridge.getInjectableStyles(),\n            ],\n            scripts: [\n                ...HAMMERHEAD_INJECTABLE_SCRIPTS.map(hs => getAssetPath(hs, this._options.developmentMode)),\n                ...SCRIPTS.map(s => getAssetPath(s, this._options.developmentMode)),\n                ...this._testRunBridge.getInjectableScripts(),\n            ],\n            embeddedScripts: [ this._getRestoreStoragesScript(restoringStorages), this._getRestoreContextStorageScript(contextStorage), taskScript],\n            userScripts:     userScripts || [],\n        };\n\n        injectableResources.scripts     = this._resolveRelativeUrls(proxy, injectableResources.scripts);\n        injectableResources.userScripts = this._resolveRelativeUrls(proxy, injectableResources.userScripts);\n        injectableResources.stylesheets = this._resolveRelativeUrls(proxy, injectableResources.stylesheets);\n\n        return injectableResources;\n    }\n\n    private _processResponseHeaders (headers: HeaderEntry[] | undefined): HeaderEntry[] {\n        if (!headers)\n            return [];\n\n        headers = headers.filter(header => !RESPONSE_REMOVED_HEADERS.includes(header.name.toLowerCase()));\n\n        return stringifyHeaderValues(headers);\n    }\n\n    private async _fulfillRequest (client: ProtocolApi, fulfillRequestInfo: FulfillRequestRequest, body: string, sessionId: SessionId, contentType?: string): Promise<void> {\n        await safeFulfillRequest(client, {\n            requestId:       fulfillRequestInfo.requestId,\n            responseCode:    fulfillRequestInfo.responseCode || StatusCodes.OK,\n            responsePhrase:  fulfillRequestInfo.responsePhrase,\n            responseHeaders: this._processResponseHeaders(fulfillRequestInfo.responseHeaders),\n            body:            toBase64String(body, contentType),\n        }, sessionId);\n    }\n\n    public async redirectToErrorPage (client: ProtocolApi, err: Error, url: string): Promise<void> {\n        const currentTestRun = this._testRunBridge.getCurrentTestRun();\n\n        if (!currentTestRun)\n            return;\n\n        currentTestRun.pendingPageError = new PageLoadError(err, url);\n\n        await navigateTo(client, this._options.specialServiceRoutes.errorPage1);\n    }\n\n    public async getDocumentResourceInfo (event: RequestPausedEvent, client: ProtocolApi, contentType?: string): Promise<DocumentResourceInfo> {\n        const {\n            requestId,\n            request,\n            responseErrorReason,\n            resourceType,\n        } = event;\n\n        if (resourceType !== 'Document') {\n            return {\n                error: null,\n                body:  null,\n            };\n        }\n\n        try {\n            if (responseErrorReason === 'NameNotResolved') {\n                const err = failedToFindDNSError(request.url);\n\n                return {\n                    error: err,\n                    body:  null,\n                };\n            }\n\n            const responseObj = await client.Fetch.getResponseBody({ requestId });\n            const responseStr = getResponseAsString(responseObj, contentType);\n\n            return {\n                error: null,\n                body:  Buffer.from(responseStr),\n            };\n        }\n        catch (err) {\n            resourceInjectorLogger('Failed to process request: %s', request.url);\n\n            return {\n                error: err,\n                body:  null,\n            };\n        }\n    }\n\n    public async processAboutBlankPage (event: FrameNavigatedEvent, userScripts: string[], contextStorage: SessionStorageInfo | null, client: ProtocolApi): Promise<void> {\n        resourceInjectorLogger('Handle page as about:blank. Origin url: %s', event.frame.url);\n\n        const injectableResources = await this._prepareInjectableResources({ isIframe: false, userScripts, contextStorage }) as PageInjectableResources;\n        const html                = injectResources(EMPTY_PAGE_MARKUP, injectableResources);\n\n        await client.Page.setDocumentContent({\n            frameId: event.frame.id,\n            html,\n        });\n    }\n\n    public async processHTMLPageContent (fulfillRequestInfo: FulfillRequestRequest, injectableResourcesOptions: InjectableResourcesOptions, client: ProtocolApi, sessionId: SessionId, contentType?: string): Promise<void> {\n        const injectableResources = await this._prepareInjectableResources(injectableResourcesOptions);\n\n        // NOTE: an unhandled exception interrupts the test execution,\n        // and we are force to redirect manually to the idle page.\n        if (!injectableResources)\n            await redirect(client, fulfillRequestInfo.requestId, this._options.specialServiceRoutes.idlePage);\n        else {\n            const updatedResponseStr = injectResources(\n                fulfillRequestInfo.body as string,\n                injectableResources,\n                this._getPageInjectableResourcesOptions(injectableResourcesOptions),\n            );\n\n            await this._fulfillRequest(client, fulfillRequestInfo, updatedResponseStr, sessionId, contentType);\n        }\n    }\n\n    public async processNonProxiedContent (fulfillRequestInfo: FulfillRequestRequest, client: ProtocolApi, sessionId: SessionId): Promise<void> {\n        await this._fulfillRequest(client, fulfillRequestInfo, fulfillRequestInfo.body as string, sessionId);\n    }\n\n    private _getPageInjectableResourcesOptions (injectableResourcesOptions: InjectableResourcesOptions): PageRestoreStoragesOptions | undefined {\n        const { url, restoringStorages } = injectableResourcesOptions;\n\n        if (url && restoringStorages) {\n            return {\n                host:      new URL(url).host,\n                sessionId: this._testRunBridge.getSessionId(),\n            };\n        }\n\n        return void 0;\n    }\n\n    public setOptions (options: ResourceInjectorOptions): void {\n        this._options = options;\n    }\n}\n"]}