Innovenergy_trunk/frontend/node_modules/testcafe/lib/proxyless/resource-injector.js

162 lines
26 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 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 RESPONSE_REMOVED_HEADERS = [
'cross-origin-embedder-policy',
'cross-origin-opener-policy',
'cross-origin-resource-policy',
];
class ResourceInjector {
constructor(testRunBridge, specialServiceRoutes) {
this._specialServiceRoutes = specialServiceRoutes;
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, '%proxylessContextStorage%', { 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,
],
scripts: [
...testcafe_hammerhead_1.INJECTABLE_SCRIPTS.map(hs => (0, testcafe_hammerhead_1.getAssetPath)(hs, proxy.options.developmentMode)),
...injectables_1.SCRIPTS.map(s => (0, testcafe_hammerhead_1.getAssetPath)(s, proxy.options.developmentMode)),
],
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) {
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),
});
}
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._specialServiceRoutes.errorPage1);
}
async getDocumentResourceInfo(event, client) {
const { requestId, request, responseErrorReason, resourceType, } = event;
if (resourceType !== 'Document') {
return {
error: null,
body: null,
};
}
try {
if (responseErrorReason === 'NameNotResolved') {
const err = new Error(`Failed to find a DNS-record for the resource at "${event.request.url}"`);
return {
error: err,
body: null,
};
}
const responseObj = await client.Fetch.getResponseBody({ requestId });
const responseStr = (0, string_1.getResponseAsString)(responseObj);
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, 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 });
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) {
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._specialServiceRoutes.idlePage);
else {
const updatedResponseStr = (0, testcafe_hammerhead_1.injectResources)(fulfillRequestInfo.body, injectableResources, this._getPageInjectableResourcesOptions(injectableResourcesOptions));
await this._fulfillRequest(client, fulfillRequestInfo, updatedResponseStr);
}
}
async processNonProxiedContent(fulfillRequestInfo, client) {
await this._fulfillRequest(client, fulfillRequestInfo, fulfillRequestInfo.body);
}
_getPageInjectableResourcesOptions(injectableResourcesOptions) {
const { url, restoringStorages } = injectableResourcesOptions;
if (url && restoringStorages) {
return {
host: new URL(url).host,
sessionId: this._testRunBridge.getSessionId(),
};
}
return void 0;
}
}
exports.default = ResourceInjector;
module.exports = exports.default;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"resource-injector.js","sourceRoot":"","sources":["../../src/proxyless/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    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';\n\nconst RESPONSE_REMOVED_HEADERS = [\n    'cross-origin-embedder-policy',\n    'cross-origin-opener-policy',\n    'cross-origin-resource-policy',\n];\n\nexport default class ResourceInjector {\n    private readonly _specialServiceRoutes: SpecialServiceRoutes;\n    private readonly _testRunBridge: TestRunBridge;\n\n    public constructor (testRunBridge: TestRunBridge, specialServiceRoutes: SpecialServiceRoutes) {\n        this._specialServiceRoutes = specialServiceRoutes;\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, '%proxylessContextStorage%', { 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            ],\n            scripts: [\n                ...HAMMERHEAD_INJECTABLE_SCRIPTS.map(hs => getAssetPath(hs, proxy.options.developmentMode)),\n                ...SCRIPTS.map(s => getAssetPath(s, proxy.options.developmentMode)),\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): 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),\n        });\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._specialServiceRoutes.errorPage1);\n    }\n\n    public async getDocumentResourceInfo (event: RequestPausedEvent, client: ProtocolApi): 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 = new Error(`Failed to find a DNS-record for the resource at \"${event.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);\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[], 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 }) 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): 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._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);\n        }\n    }\n\n    public async processNonProxiedContent (fulfillRequestInfo: FulfillRequestRequest, client: ProtocolApi): Promise<void> {\n        await this._fulfillRequest(client, fulfillRequestInfo, fulfillRequestInfo.body as string);\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"]}