114 lines
19 KiB
JavaScript
114 lines
19 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 hook_1 = __importDefault(require("./hook"));
|
|
const parse_user_agent_1 = require("../../utils/parse-user-agent");
|
|
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,
|
|
};
|
|
const REQUEST_LOGGER_CLASS_NAME = 'RequestLogger';
|
|
class RequestLoggerImplementation extends hook_1.default {
|
|
constructor(requestFilterRuleInit, options) {
|
|
const effectiveOptions = Object.assign({}, DEFAULT_OPTIONS, options);
|
|
RequestLoggerImplementation._assertLogOptions(effectiveOptions);
|
|
const configureResponseEventOptions = new testcafe_hammerhead_1.ConfigureResponseEventOptions(effectiveOptions.logResponseHeaders, effectiveOptions.logResponseBody);
|
|
super(requestFilterRuleInit, configureResponseEventOptions);
|
|
this._className = REQUEST_LOGGER_CLASS_NAME;
|
|
this._options = effectiveOptions;
|
|
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 loggedReq = {
|
|
id: event._requestInfo.requestId,
|
|
testRunId: event._requestInfo.sessionId,
|
|
userAgent: (0, parse_user_agent_1.parseUserAgent)(event._requestInfo.userAgent).prettyUserAgent,
|
|
request: {
|
|
timestamp: Date.now(),
|
|
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 loggedReq = 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 (!loggedReq)
|
|
return;
|
|
loggedReq.response = {
|
|
statusCode: event.statusCode,
|
|
timestamp: Date.now(),
|
|
};
|
|
if (this._options.logResponseHeaders)
|
|
loggedReq.response.headers = Object.assign({}, event.headers);
|
|
if (this._options.logResponseBody) {
|
|
loggedReq.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.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n    ConfigureResponseEventOptions,\n    RequestEvent,\n    ResponseEvent,\n    RequestFilterRuleInit,\n} from 'testcafe-hammerhead';\n\nimport RequestHook from './hook';\nimport { parseUserAgent } from '../../utils/parse-user-agent';\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\nimport {\n    RequestHookLogOptionsInit,\n    RequestHookLogOptions,\n} from './interfaces';\n\nimport { Dictionary } from '../../configuration/interfaces';\n\n\nconst DEFAULT_OPTIONS: RequestHookLogOptions = {\n    logRequestHeaders:     false,\n    logRequestBody:        false,\n    stringifyRequestBody:  false,\n    logResponseHeaders:    false,\n    logResponseBody:       false,\n    stringifyResponseBody: false,\n};\n\ninterface OptionallyLoggedPart {\n    headers?: any;\n    body?: Buffer | string;\n}\n\ninterface LoggedRequestPart extends OptionallyLoggedPart {\n    timestamp: number;\n    url: string;\n    method: string;\n}\ninterface LoggedResponsePart extends OptionallyLoggedPart {\n    statusCode: number;\n    timestamp: number;\n}\n\ninterface LoggedRequest {\n    id: string;\n    testRunId: string;\n    userAgent: string;\n    request: LoggedRequestPart;\n    response?: LoggedResponsePart;\n}\n\nconst REQUEST_LOGGER_CLASS_NAME = 'RequestLogger';\n\nclass RequestLoggerImplementation extends RequestHook {\n    private readonly _options: RequestHookLogOptions;\n    private _internalRequests: Dictionary<LoggedRequest>;\n\n    public constructor (requestFilterRuleInit?: RequestFilterRuleInit | RequestFilterRuleInit[], options?: RequestHookLogOptionsInit) {\n        const effectiveOptions = Object.assign({}, DEFAULT_OPTIONS, options) as RequestHookLogOptions;\n\n        RequestLoggerImplementation._assertLogOptions(effectiveOptions);\n\n        const configureResponseEventOptions = new ConfigureResponseEventOptions(effectiveOptions.logResponseHeaders, effectiveOptions.logResponseBody);\n\n        super(requestFilterRuleInit, configureResponseEventOptions);\n\n        this._className        = REQUEST_LOGGER_CLASS_NAME;\n        this._options          = effectiveOptions;\n        this._internalRequests = {};\n    }\n\n    private static _assertLogOptions (logOptions: RequestHookLogOptions): void {\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    public async onRequest (event: RequestEvent): Promise<void> {\n        const loggedReq: LoggedRequest = {\n            id:        event._requestInfo.requestId,\n            testRunId: event._requestInfo.sessionId,\n            userAgent: parseUserAgent(event._requestInfo.userAgent).prettyUserAgent,\n            request:   {\n                timestamp: Date.now(),\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    public async onResponse (event: ResponseEvent): Promise<void> {\n        const loggedReq = 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 (!loggedReq)\n            return;\n\n        loggedReq.response = {\n            statusCode: event.statusCode,\n            timestamp:  Date.now(),\n        };\n\n        if (this._options.logResponseHeaders)\n            loggedReq.response.headers = Object.assign({}, event.headers);\n\n        if (this._options.logResponseBody) {\n            loggedReq.response.body = this._options.stringifyResponseBody && event.body\n                ? event.body.toString()\n                : event.body;\n        }\n    }\n\n    private _prepareInternalRequestInfo (): LoggedRequest[] {\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    private _getCompletedRequests (): LoggedRequest[] {\n        return this._prepareInternalRequestInfo().filter(r => r.response);\n    }\n\n    // API\n    public contains (predicate: (request: LoggedRequest) => boolean): ReExecutablePromise {\n        return ReExecutablePromise.fromFn(async () => {\n            return !!this._getCompletedRequests().find(predicate);\n        });\n    }\n\n    public count (predicate: (request: LoggedRequest) => boolean): ReExecutablePromise {\n        return ReExecutablePromise.fromFn(async () => {\n            return this._getCompletedRequests().filter(predicate).length;\n        });\n    }\n\n    public clear (): void {\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    public get requests (): LoggedRequest[] {\n        return this._prepareInternalRequestInfo();\n    }\n}\n\nexport default function createRequestLogger (requestFilterRuleInit: RequestFilterRuleInit | RequestFilterRuleInit[] | undefined, logOptions: RequestHookLogOptionsInit): RequestLoggerImplementation {\n    return new RequestLoggerImplementation(requestFilterRuleInit, logOptions);\n}\n"]}
|