UNPKG

testcafe

Version:

Automated browser testing for the modern web development stack.

110 lines 16.8 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 hook_1 = __importDefault(require("./hook")); const useragent_1 = require("useragent"); const test_run_tracker_1 = __importDefault(require("../test-run-tracker")); const re_executable_promise_1 = __importDefault(require("../../utils/re-executable-promise")); const runtime_1 = require("../../errors/runtime"); const types_1 = require("../../errors/types"); const DEFAULT_OPTIONS = { logRequestHeaders: false, logRequestBody: false, stringifyRequestBody: false, logResponseHeaders: false, logResponseBody: false, stringifyResponseBody: false }; class RequestLoggerImplementation extends hook_1.default { constructor(requestFilterRuleInit, options) { options = Object.assign({}, DEFAULT_OPTIONS, options); RequestLoggerImplementation._assertLogOptions(options); const configureResponseEventOptions = new testcafe_hammerhead_1.ConfigureResponseEventOptions(options.logResponseHeaders, options.logResponseBody); super(requestFilterRuleInit, configureResponseEventOptions); this.options = options; this._internalRequests = {}; } static _assertLogOptions(logOptions) { if (!logOptions.logRequestBody && logOptions.stringifyRequestBody) throw new runtime_1.APIError('RequestLogger', types_1.RUNTIME_ERRORS.requestHookConfigureAPIError, 'RequestLogger', 'Cannot stringify the request body because it is not logged. Specify { logRequestBody: true } in log options.'); if (!logOptions.logResponseBody && logOptions.stringifyResponseBody) throw new runtime_1.APIError('RequestLogger', types_1.RUNTIME_ERRORS.requestHookConfigureAPIError, 'RequestLogger', 'Cannot stringify the response body because it is not logged. Specify { logResponseBody: true } in log options.'); } async onRequest(event) { const userAgent = useragent_1.parse(event._requestInfo.userAgent).toString(); const loggedReq = { id: event._requestInfo.requestId, testRunId: event._requestInfo.sessionId, userAgent, request: { url: event._requestInfo.url, method: event._requestInfo.method, } }; if (this.options.logRequestHeaders) loggedReq.request.headers = Object.assign({}, event._requestInfo.headers); if (this.options.logRequestBody) loggedReq.request.body = this.options.stringifyRequestBody ? event._requestInfo.body.toString() : event._requestInfo.body; this._internalRequests[loggedReq.id] = loggedReq; } async onResponse(event) { const loggerReq = this._internalRequests[event.requestId]; // NOTE: If the 'clear' method is called during a long running request, // we should not save a response part - request part has been already removed. if (!loggerReq) return; loggerReq.response = {}; loggerReq.response.statusCode = event.statusCode; if (this.options.logResponseHeaders) loggerReq.response.headers = Object.assign({}, event.headers); if (this.options.logResponseBody) { loggerReq.response.body = this.options.stringifyResponseBody && event.body ? event.body.toString() : event.body; } } _prepareInternalRequestInfo() { const testRun = test_run_tracker_1.default.resolveContextTestRun(); let preparedRequests = Object.values(this._internalRequests); if (testRun) preparedRequests = preparedRequests.filter(r => r.testRunId === testRun.id); return preparedRequests; } _getCompletedRequests() { return this._prepareInternalRequestInfo().filter(r => r.response); } // API contains(predicate) { return re_executable_promise_1.default.fromFn(async () => { return !!this._getCompletedRequests().find(predicate); }); } count(predicate) { return re_executable_promise_1.default.fromFn(async () => { return this._getCompletedRequests().filter(predicate).length; }); } clear() { const testRun = test_run_tracker_1.default.resolveContextTestRun(); if (testRun) { Object.keys(this._internalRequests).forEach(id => { if (this._internalRequests[id].testRunId === testRun.id) delete this._internalRequests[id]; }); } else this._internalRequests = {}; } get requests() { return this._prepareInternalRequestInfo(); } } function createRequestLogger(requestFilterRuleInit, logOptions) { return new RequestLoggerImplementation(requestFilterRuleInit, logOptions); } exports.default = createRequestLogger; module.exports = exports.default; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"request-logger.js","sourceRoot":"","sources":["../../../src/api/request-hooks/request-logger.js"],"names":[],"mappings":";;;;;AAAA,6DAAoE;AACpE,kDAAiC;AACjC,yCAAoD;AACpD,2EAAiD;AACjD,8FAAoE;AACpE,kDAAgD;AAChD,8CAAoD;AAEpD,MAAM,eAAe,GAAG;IACpB,iBAAiB,EAAM,KAAK;IAC5B,cAAc,EAAS,KAAK;IAC5B,oBAAoB,EAAG,KAAK;IAC5B,kBAAkB,EAAK,KAAK;IAC5B,eAAe,EAAQ,KAAK;IAC5B,qBAAqB,EAAE,KAAK;CAC/B,CAAC;AAEF,MAAM,2BAA4B,SAAQ,cAAW;IACjD,YAAa,qBAAqB,EAAE,OAAO;QACvC,OAAO,GAAG,MAAM,CAAC,MAAM,CAAC,EAAE,EAAE,eAAe,EAAE,OAAO,CAAC,CAAC;QACtD,2BAA2B,CAAC,iBAAiB,CAAC,OAAO,CAAC,CAAC;QAEvD,MAAM,6BAA6B,GAAG,IAAI,mDAA6B,CAAC,OAAO,CAAC,kBAAkB,EAAE,OAAO,CAAC,eAAe,CAAC,CAAC;QAE7H,KAAK,CAAC,qBAAqB,EAAE,6BAA6B,CAAC,CAAC;QAE5D,IAAI,CAAC,OAAO,GAAG,OAAO,CAAC;QAEvB,IAAI,CAAC,iBAAiB,GAAG,EAAE,CAAC;IAChC,CAAC;IAED,MAAM,CAAC,iBAAiB,CAAE,UAAU;QAChC,IAAI,CAAC,UAAU,CAAC,cAAc,IAAI,UAAU,CAAC,oBAAoB;YAC7D,MAAM,IAAI,kBAAQ,CAAC,eAAe,EAAE,sBAAc,CAAC,4BAA4B,EAAE,eAAe,EAAE,8GAA8G,CAAC,CAAC;QAEtN,IAAI,CAAC,UAAU,CAAC,eAAe,IAAI,UAAU,CAAC,qBAAqB;YAC/D,MAAM,IAAI,kBAAQ,CAAC,eAAe,EAAE,sBAAc,CAAC,4BAA4B,EAAE,eAAe,EAAE,gHAAgH,CAAC,CAAC;IAC5N,CAAC;IAED,KAAK,CAAC,SAAS,CAAE,KAAK;QAClB,MAAM,SAAS,GAAG,iBAAc,CAAC,KAAK,CAAC,YAAY,CAAC,SAAS,CAAC,CAAC,QAAQ,EAAE,CAAC;QAE1E,MAAM,SAAS,GAAG;YACd,EAAE,EAAS,KAAK,CAAC,YAAY,CAAC,SAAS;YACvC,SAAS,EAAE,KAAK,CAAC,YAAY,CAAC,SAAS;YACvC,SAAS;YACT,OAAO,EAAI;gBACP,GAAG,EAAK,KAAK,CAAC,YAAY,CAAC,GAAG;gBAC9B,MAAM,EAAE,KAAK,CAAC,YAAY,CAAC,MAAM;aACpC;SACJ,CAAC;QAEF,IAAI,IAAI,CAAC,OAAO,CAAC,iBAAiB;YAC9B,SAAS,CAAC,OAAO,CAAC,OAAO,GAAG,MAAM,CAAC,MAAM,CAAC,EAAE,EAAE,KAAK,CAAC,YAAY,CAAC,OAAO,CAAC,CAAC;QAE9E,IAAI,IAAI,CAAC,OAAO,CAAC,cAAc;YAC3B,SAAS,CAAC,OAAO,CAAC,IAAI,GAAG,IAAI,CAAC,OAAO,CAAC,oBAAoB,CAAC,CAAC,CAAC,KAAK,CAAC,YAAY,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC,CAAC,CAAC,KAAK,CAAC,YAAY,CAAC,IAAI,CAAC;QAE9H,IAAI,CAAC,iBAAiB,CAAC,SAAS,CAAC,EAAE,CAAC,GAAG,SAAS,CAAC;IACrD,CAAC;IAED,KAAK,CAAC,UAAU,CAAE,KAAK;QACnB,MAAM,SAAS,GAAG,IAAI,CAAC,iBAAiB,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;QAE1D,uEAAuE;QACvE,8EAA8E;QAC9E,IAAI,CAAC,SAAS;YACV,OAAO;QAEX,SAAS,CAAC,QAAQ,GAAc,EAAE,CAAC;QACnC,SAAS,CAAC,QAAQ,CAAC,UAAU,GAAG,KAAK,CAAC,UAAU,CAAC;QAEjD,IAAI,IAAI,CAAC,OAAO,CAAC,kBAAkB;YAC/B,SAAS,CAAC,QAAQ,CAAC,OAAO,GAAG,MAAM,CAAC,MAAM,CAAC,EAAE,EAAE,KAAK,CAAC,OAAO,CAAC,CAAC;QAElE,IAAI,IAAI,CAAC,OAAO,CAAC,eAAe,EAAE;YAC9B,SAAS,CAAC,QAAQ,CAAC,IAAI,GAAG,IAAI,CAAC,OAAO,CAAC,qBAAqB,IAAI,KAAK,CAAC,IAAI;gBACtE,CAAC,CAAC,KAAK,CAAC,IAAI,CAAC,QAAQ,EAAE;gBACvB,CAAC,CAAC,KAAK,CAAC,IAAI,CAAC;SACpB;IACL,CAAC;IAED,2BAA2B;QACvB,MAAM,OAAO,GAAU,0BAAc,CAAC,qBAAqB,EAAE,CAAC;QAC9D,IAAI,gBAAgB,GAAG,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC;QAE7D,IAAI,OAAO;YACP,gBAAgB,GAAG,gBAAgB,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,SAAS,KAAK,OAAO,CAAC,EAAE,CAAC,CAAC;QAEhF,OAAO,gBAAgB,CAAC;IAC5B,CAAC;IAED,qBAAqB;QACjB,OAAO,IAAI,CAAC,2BAA2B,EAAE,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC;IACtE,CAAC;IAED,MAAM;IACN,QAAQ,CAAE,SAAS;QACf,OAAO,+BAAmB,CAAC,MAAM,CAAC,KAAK,IAAI,EAAE;YACzC,OAAO,CAAC,CAAC,IAAI,CAAC,qBAAqB,EAAE,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QAC1D,CAAC,CAAC,CAAC;IACP,CAAC;IAED,KAAK,CAAE,SAAS;QACZ,OAAO,+BAAmB,CAAC,MAAM,CAAC,KAAK,IAAI,EAAE;YACzC,OAAO,IAAI,CAAC,qBAAqB,EAAE,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC,MAAM,CAAC;QACjE,CAAC,CAAC,CAAC;IACP,CAAC;IAED,KAAK;QACD,MAAM,OAAO,GAAG,0BAAc,CAAC,qBAAqB,EAAE,CAAC;QAEvD,IAAI,OAAO,EAAE;YACT,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC,OAAO,CAAC,EAAE,CAAC,EAAE;gBAC7C,IAAI,IAAI,CAAC,iBAAiB,CAAC,EAAE,CAAC,CAAC,SAAS,KAAK,OAAO,CAAC,EAAE;oBACnD,OAAO,IAAI,CAAC,iBAAiB,CAAC,EAAE,CAAC,CAAC;YAC1C,CAAC,CAAC,CAAC;SACN;;YAEG,IAAI,CAAC,iBAAiB,GAAG,EAAE,CAAC;IACpC,CAAC;IAED,IAAI,QAAQ;QACR,OAAO,IAAI,CAAC,2BAA2B,EAAE,CAAC;IAC9C,CAAC;CACJ;AAED,SAAwB,mBAAmB,CAAE,qBAAqB,EAAE,UAAU;IAC1E,OAAO,IAAI,2BAA2B,CAAC,qBAAqB,EAAE,UAAU,CAAC,CAAC;AAC9E,CAAC;AAFD,sCAEC","sourcesContent":["import { ConfigureResponseEventOptions } from 'testcafe-hammerhead';\nimport RequestHook from './hook';\nimport { parse as parseUserAgent } from 'useragent';\nimport testRunTracker from '../test-run-tracker';\nimport ReExecutablePromise from '../../utils/re-executable-promise';\nimport { APIError } from '../../errors/runtime';\nimport { RUNTIME_ERRORS } from '../../errors/types';\n\nconst DEFAULT_OPTIONS = {\n    logRequestHeaders:     false,\n    logRequestBody:        false,\n    stringifyRequestBody:  false,\n    logResponseHeaders:    false,\n    logResponseBody:       false,\n    stringifyResponseBody: false\n};\n\nclass RequestLoggerImplementation extends RequestHook {\n    constructor (requestFilterRuleInit, options) {\n        options = Object.assign({}, DEFAULT_OPTIONS, options);\n        RequestLoggerImplementation._assertLogOptions(options);\n\n        const configureResponseEventOptions = new ConfigureResponseEventOptions(options.logResponseHeaders, options.logResponseBody);\n\n        super(requestFilterRuleInit, configureResponseEventOptions);\n\n        this.options = options;\n\n        this._internalRequests = {};\n    }\n\n    static _assertLogOptions (logOptions) {\n        if (!logOptions.logRequestBody && logOptions.stringifyRequestBody)\n            throw new APIError('RequestLogger', RUNTIME_ERRORS.requestHookConfigureAPIError, 'RequestLogger', 'Cannot stringify the request body because it is not logged. Specify { logRequestBody: true } in log options.');\n\n        if (!logOptions.logResponseBody && logOptions.stringifyResponseBody)\n            throw new APIError('RequestLogger', RUNTIME_ERRORS.requestHookConfigureAPIError, 'RequestLogger', 'Cannot stringify the response body because it is not logged. Specify { logResponseBody: true } in log options.');\n    }\n\n    async onRequest (event) {\n        const userAgent = parseUserAgent(event._requestInfo.userAgent).toString();\n\n        const loggedReq = {\n            id:        event._requestInfo.requestId,\n            testRunId: event._requestInfo.sessionId,\n            userAgent,\n            request:   {\n                url:    event._requestInfo.url,\n                method: event._requestInfo.method,\n            }\n        };\n\n        if (this.options.logRequestHeaders)\n            loggedReq.request.headers = Object.assign({}, event._requestInfo.headers);\n\n        if (this.options.logRequestBody)\n            loggedReq.request.body = this.options.stringifyRequestBody ? event._requestInfo.body.toString() : event._requestInfo.body;\n\n        this._internalRequests[loggedReq.id] = loggedReq;\n    }\n\n    async onResponse (event) {\n        const loggerReq = this._internalRequests[event.requestId];\n\n        // NOTE: If the 'clear' method is called during a long running request,\n        // we should not save a response part - request part has been already removed.\n        if (!loggerReq)\n            return;\n\n        loggerReq.response            = {};\n        loggerReq.response.statusCode = event.statusCode;\n\n        if (this.options.logResponseHeaders)\n            loggerReq.response.headers = Object.assign({}, event.headers);\n\n        if (this.options.logResponseBody) {\n            loggerReq.response.body = this.options.stringifyResponseBody && event.body\n                ? event.body.toString()\n                : event.body;\n        }\n    }\n\n    _prepareInternalRequestInfo () {\n        const testRun        = testRunTracker.resolveContextTestRun();\n        let preparedRequests = Object.values(this._internalRequests);\n\n        if (testRun)\n            preparedRequests = preparedRequests.filter(r => r.testRunId === testRun.id);\n\n        return preparedRequests;\n    }\n\n    _getCompletedRequests () {\n        return this._prepareInternalRequestInfo().filter(r => r.response);\n    }\n\n    // API\n    contains (predicate) {\n        return ReExecutablePromise.fromFn(async () => {\n            return !!this._getCompletedRequests().find(predicate);\n        });\n    }\n\n    count (predicate) {\n        return ReExecutablePromise.fromFn(async () => {\n            return this._getCompletedRequests().filter(predicate).length;\n        });\n    }\n\n    clear () {\n        const testRun = testRunTracker.resolveContextTestRun();\n\n        if (testRun) {\n            Object.keys(this._internalRequests).forEach(id => {\n                if (this._internalRequests[id].testRunId === testRun.id)\n                    delete this._internalRequests[id];\n            });\n        }\n        else\n            this._internalRequests = {};\n    }\n\n    get requests () {\n        return this._prepareInternalRequestInfo();\n    }\n}\n\nexport default function createRequestLogger (requestFilterRuleInit, logOptions) {\n    return new RequestLoggerImplementation(requestFilterRuleInit, logOptions);\n}\n\n"]}