testcafe
Version:
Automated browser testing for the modern web development stack.
177 lines (174 loc) • 29.5 kB
JavaScript
;
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"]}