testcafe
Version:
Automated browser testing for the modern web development stack.
90 lines • 14.7 kB
JavaScript
;
var __importDefault = (this && this.__importDefault) || function (mod) {
return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
const injectables_1 = require("../../assets/injectables");
const debug_loggers_1 = require("../../utils/debug-loggers");
const cdp_1 = require("../utils/cdp");
const constants_1 = require("./constants");
const http_status_codes_1 = require("http-status-codes");
const load_assets_1 = __importDefault(require("../../load-assets"));
const string_1 = require("../utils/string");
const safe_api_1 = require("./safe-api");
async function handleRequestPauseEvent(event, client, sessionId) {
if ((0, cdp_1.isRequest)(event))
await (0, safe_api_1.safeContinueRequest)(client, event, sessionId);
else
await (0, safe_api_1.safeContinueResponse)(client, event, sessionId);
}
const internalRequest = {
condition: (event) => !event.networkId && event.resourceType !== 'Document' && !event.request.url,
handler: async (event, client, isMainWindow, options, sessionId) => {
(0, debug_loggers_1.requestPipelineInternalRequestLogger)('%r', event);
await handleRequestPauseEvent(event, client, sessionId);
},
};
const serviceRequest = {
condition: (event, options, serviceRoutes) => {
const url = event.request.url;
// NOTE: the service 'Error page' should be proxied.
if (url === serviceRoutes.errorPage1
|| url === serviceRoutes.errorPage2)
return false;
return options.serviceDomains.some(domain => url.startsWith(domain));
},
handler: async (event, client, isMainWindow, options, sessionId) => {
(0, debug_loggers_1.requestPipelineServiceRequestLogger)('%r', event);
try {
await handleRequestPauseEvent(event, client, sessionId);
}
catch (err) {
if (isMainWindow) {
(0, debug_loggers_1.requestPipelineServiceRequestLogger)('Failed to process request in main window: %s', event.request.url);
throw err;
}
else {
// NOTE: Sometimes, a child window sends a heartbeat request and then immediately closes.
// In these situations, we need to catch errors because we can't handle this request correctly
// when the cdpClient has already closed.
(0, debug_loggers_1.requestPipelineServiceRequestLogger)('Failed to process request in child window: %s', event.request.url);
}
}
},
};
const defaultFaviconRequest = {
condition: (event) => {
const parsedUrl = new URL(event.request.url);
return parsedUrl.pathname === injectables_1.DEFAULT_FAVICON_PATH;
},
handler: async (event, client, isMainWindow, options, sessionId) => {
(0, debug_loggers_1.requestPipelineLogger)('%r', event);
if ((0, cdp_1.isRequest)(event))
await (0, safe_api_1.safeContinueRequest)(client, event, sessionId);
else {
if (event.responseStatusCode === http_status_codes_1.StatusCodes.NOT_FOUND) { // eslint-disable-line no-lonely-if
const { favIcon } = (0, load_assets_1.default)(options.developmentMode);
await (0, safe_api_1.safeFulfillRequest)(client, {
requestId: event.requestId,
responseCode: http_status_codes_1.StatusCodes.OK,
responseHeaders: [constants_1.FAVICON_CONTENT_TYPE_HEADER],
body: (0, string_1.toBase64String)(favIcon),
}, sessionId);
}
else
await (0, safe_api_1.safeContinueResponse)(client, event, sessionId);
}
},
};
const SPECIAL_REQUEST_HANDLERS = [
internalRequest,
serviceRequest,
defaultFaviconRequest,
];
function getSpecialRequestHandler(event, options, serviceRoutes) {
const specialRequestHandler = SPECIAL_REQUEST_HANDLERS.find(h => h.condition(event, options, serviceRoutes));
return specialRequestHandler ? specialRequestHandler.handler : null;
}
exports.default = getSpecialRequestHandler;
module.exports = exports.default;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"special-handlers.js","sourceRoot":"","sources":["../../../src/native-automation/request-pipeline/special-handlers.ts"],"names":[],"mappings":"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","sourcesContent":["import Protocol from 'devtools-protocol';\nimport RequestPausedEvent = Protocol.Fetch.RequestPausedEvent;\nimport { DEFAULT_FAVICON_PATH } from '../../assets/injectables';\nimport {\n    RequestHandler,\n    SessionId,\n    SpecialServiceRoutes,\n} from '../types';\nimport { ProtocolApi } from 'chrome-remote-interface';\nimport {\n    requestPipelineInternalRequestLogger,\n    requestPipelineLogger,\n    requestPipelineServiceRequestLogger,\n} from '../../utils/debug-loggers';\nimport { NativeAutomationInitOptions } from '../../shared/types';\nimport { isRequest } from '../utils/cdp';\nimport { FAVICON_CONTENT_TYPE_HEADER } from './constants';\nimport { StatusCodes } from 'http-status-codes';\nimport loadAssets from '../../load-assets';\nimport { toBase64String } from '../utils/string';\nimport {\n    safeContinueRequest,\n    safeContinueResponse,\n    safeFulfillRequest,\n} from './safe-api';\n\nasync function handleRequestPauseEvent (event: RequestPausedEvent, client: ProtocolApi, sessionId: SessionId): Promise<void> {\n    if (isRequest(event))\n        await safeContinueRequest(client, event, sessionId);\n    else\n        await safeContinueResponse(client, event, sessionId);\n}\n\n\nconst internalRequest = {\n    condition: (event: RequestPausedEvent): boolean => !event.networkId && event.resourceType !== 'Document' && !event.request.url,\n    handler:   async (event: RequestPausedEvent, client: ProtocolApi, isMainWindow: boolean, options: NativeAutomationInitOptions, sessionId: SessionId): Promise<void> => {\n        requestPipelineInternalRequestLogger('%r', event);\n\n        await handleRequestPauseEvent(event, client, sessionId);\n    },\n} as RequestHandler;\n\nconst serviceRequest = {\n    condition: (event: RequestPausedEvent, options: NativeAutomationInitOptions, serviceRoutes: SpecialServiceRoutes): boolean => {\n        const url = event.request.url;\n\n        // NOTE: the service 'Error page' should be proxied.\n        if (url === serviceRoutes.errorPage1\n            || url === serviceRoutes.errorPage2)\n            return false;\n\n        return options.serviceDomains.some(domain => url.startsWith(domain));\n    },\n    handler: async (event: RequestPausedEvent, client: ProtocolApi, isMainWindow: boolean, options: NativeAutomationInitOptions, sessionId: SessionId): Promise<void> => {\n        requestPipelineServiceRequestLogger('%r', event);\n\n        try {\n            await handleRequestPauseEvent(event, client, sessionId);\n        }\n        catch (err) {\n            if (isMainWindow) {\n                requestPipelineServiceRequestLogger('Failed to process request in main window: %s', event.request.url);\n\n                throw err;\n            }\n            else {\n                // NOTE: Sometimes, a child window sends a heartbeat request and then immediately closes.\n                // In these situations, we need to catch errors because we can't handle this request correctly\n                // when the cdpClient has already closed.\n                requestPipelineServiceRequestLogger('Failed to process request in child window: %s', event.request.url);\n            }\n        }\n    },\n} as RequestHandler;\n\nconst defaultFaviconRequest = {\n    condition: (event: RequestPausedEvent): boolean => {\n        const parsedUrl = new URL(event.request.url);\n\n        return parsedUrl.pathname === DEFAULT_FAVICON_PATH;\n    },\n    handler: async (event: RequestPausedEvent, client: ProtocolApi, isMainWindow: boolean, options: NativeAutomationInitOptions, sessionId: SessionId): Promise<void> => {\n        requestPipelineLogger('%r', event);\n\n        if (isRequest(event))\n            await safeContinueRequest(client, event, sessionId);\n        else {\n            if (event.responseStatusCode === StatusCodes.NOT_FOUND) { // eslint-disable-line no-lonely-if\n                const { favIcon } = loadAssets(options.developmentMode);\n\n                await safeFulfillRequest(client, {\n                    requestId:       event.requestId,\n                    responseCode:    StatusCodes.OK,\n                    responseHeaders: [ FAVICON_CONTENT_TYPE_HEADER ],\n                    body:            toBase64String(favIcon),\n                }, sessionId);\n            }\n            else\n                await safeContinueResponse(client, event, sessionId);\n        }\n    },\n} as RequestHandler;\n\nconst SPECIAL_REQUEST_HANDLERS = [\n    internalRequest,\n    serviceRequest,\n    defaultFaviconRequest,\n];\n\nexport default function getSpecialRequestHandler (event: RequestPausedEvent, options?: NativeAutomationInitOptions, serviceRoutes?: SpecialServiceRoutes): RequestHandler['handler'] | null {\n    const specialRequestHandler = SPECIAL_REQUEST_HANDLERS.find(h => h.condition(event, options, serviceRoutes));\n\n    return specialRequestHandler ? specialRequestHandler.handler : null;\n}\n\n"]}