89 lines
12 KiB
JavaScript
89 lines
12 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 events_1 = require("events");
|
|
const delay_1 = __importDefault(require("../utils/delay"));
|
|
const thennable_1 = require("../utils/thennable");
|
|
const test_run_1 = require("../errors/test-run");
|
|
const re_executable_promise_1 = __importDefault(require("../utils/re-executable-promise"));
|
|
const get_fn_1 = __importDefault(require("./get-fn"));
|
|
const marker_1 = require("../services/serialization/replicator/transforms/function-marker-transform/marker");
|
|
const marker_2 = require("../services/serialization/replicator/transforms/promise-marker-transform/marker");
|
|
const ASSERTION_DELAY = 200;
|
|
class AssertionExecutor extends events_1.EventEmitter {
|
|
constructor(command, timeout, callsite) {
|
|
super();
|
|
this.command = command;
|
|
this.timeout = timeout;
|
|
this.callsite = callsite;
|
|
this.startTime = null;
|
|
this.passed = false;
|
|
this.inRetry = false;
|
|
const fn = (0, get_fn_1.default)(this.command);
|
|
const actualCommand = this.command.actual;
|
|
if (actualCommand instanceof re_executable_promise_1.default)
|
|
this.fn = this._wrapFunction(fn);
|
|
else if (!this.command.options.allowUnawaitedPromise && this._isPromise(actualCommand))
|
|
throw new test_run_1.AssertionUnawaitedPromiseError(this.callsite);
|
|
else
|
|
this.fn = fn;
|
|
}
|
|
_isPromise(val) {
|
|
return (0, thennable_1.isThennable)(val) ||
|
|
val === Symbol.for(marker_2.PROMISE_MARKER_DESCRIPTION);
|
|
}
|
|
_getTimeLeft() {
|
|
const executionTime = new Date().getTime() - this.startTime; // eslint-disable-line @typescript-eslint/no-extra-parens
|
|
return this.timeout - executionTime;
|
|
}
|
|
_onExecutionFinished() {
|
|
if (this.inRetry)
|
|
this.emit('end-assertion-retries', this.passed);
|
|
}
|
|
_wrapFunction(fn) {
|
|
return async () => {
|
|
const resultPromise = this.command.actual;
|
|
while (!this.passed) {
|
|
this.command.actual = await resultPromise._reExecute();
|
|
try {
|
|
fn();
|
|
this.passed = true;
|
|
this._onExecutionFinished();
|
|
}
|
|
catch (err) {
|
|
if (this._getTimeLeft() <= 0) {
|
|
this._onExecutionFinished();
|
|
throw err;
|
|
}
|
|
await (0, delay_1.default)(ASSERTION_DELAY);
|
|
this.inRetry = true;
|
|
this.emit('start-assertion-retries', this._getTimeLeft());
|
|
}
|
|
}
|
|
};
|
|
}
|
|
_onBeforeRun() {
|
|
if (this.command.actual !== Symbol.for(marker_1.FUNCTION_MARKER_DESCRIPTION))
|
|
return;
|
|
this.emit('non-serializable-actual-value', this);
|
|
}
|
|
async run() {
|
|
this._onBeforeRun();
|
|
this.startTime = new Date().getTime();
|
|
try {
|
|
await this.fn();
|
|
}
|
|
catch (err) {
|
|
if (err.name === 'AssertionError' || err.constructor.name === 'AssertionError')
|
|
throw new test_run_1.ExternalAssertionLibraryError(err, this.callsite);
|
|
if (err.isTestCafeError)
|
|
err.callsite = this.callsite;
|
|
throw err;
|
|
}
|
|
}
|
|
}
|
|
exports.default = AssertionExecutor;
|
|
module.exports = exports.default;
|
|
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"executor.js","sourceRoot":"","sources":["../../src/assertions/executor.ts"],"names":[],"mappings":"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","sourcesContent":["import { EventEmitter } from 'events';\nimport delay from '../utils/delay';\nimport { isThennable } from '../utils/thennable';\nimport { ExternalAssertionLibraryError, AssertionUnawaitedPromiseError } from '../errors/test-run';\nimport ReExecutablePromise from '../utils/re-executable-promise';\nimport getFn from './get-fn';\nimport { AssertionCommand } from '../test-run/commands/assertion';\nimport { CallsiteRecord } from 'callsite-record';\nimport { FUNCTION_MARKER_DESCRIPTION } from '../services/serialization/replicator/transforms/function-marker-transform/marker';\nimport { PROMISE_MARKER_DESCRIPTION } from '../services/serialization/replicator/transforms/promise-marker-transform/marker';\n\nconst ASSERTION_DELAY = 200;\n\nexport default class AssertionExecutor extends EventEmitter {\n    public readonly command: AssertionCommand;\n    private readonly timeout: number;\n    private readonly callsite: CallsiteRecord;\n    private startTime: number | null;\n    private passed: boolean;\n    private inRetry: boolean;\n    public fn: Function;\n\n    public constructor (command: AssertionCommand, timeout: number, callsite: CallsiteRecord) {\n        super();\n\n        this.command  = command;\n        this.timeout  = timeout;\n        this.callsite = callsite;\n\n        this.startTime = null;\n        this.passed    = false;\n        this.inRetry   = false;\n\n        const fn            = getFn(this.command);\n        const actualCommand = this.command.actual;\n\n        if (actualCommand instanceof ReExecutablePromise)\n            this.fn = this._wrapFunction(fn);\n        else if (!this.command.options.allowUnawaitedPromise && this._isPromise(actualCommand))\n            throw new AssertionUnawaitedPromiseError(this.callsite);\n        else\n            this.fn = fn;\n    }\n\n    private _isPromise (val: unknown): boolean {\n        return isThennable(val) ||\n            val === Symbol.for(PROMISE_MARKER_DESCRIPTION);\n    }\n\n    private _getTimeLeft (): number {\n        const executionTime = new Date().getTime() - (this.startTime as number); // eslint-disable-line @typescript-eslint/no-extra-parens\n\n        return this.timeout - executionTime;\n    }\n\n    private _onExecutionFinished (): void {\n        if (this.inRetry)\n            this.emit('end-assertion-retries', this.passed);\n    }\n\n    private _wrapFunction (fn: Function): Function {\n        return async () => {\n            const resultPromise = this.command.actual as ReExecutablePromise;\n\n            while (!this.passed) {\n                this.command.actual = await resultPromise._reExecute();\n\n                try {\n                    fn();\n                    this.passed = true;\n                    this._onExecutionFinished();\n                }\n\n                catch (err) {\n                    if (this._getTimeLeft() <= 0) {\n                        this._onExecutionFinished();\n                        throw err;\n                    }\n\n                    await delay(ASSERTION_DELAY);\n\n                    this.inRetry = true;\n                    this.emit('start-assertion-retries', this._getTimeLeft());\n                }\n            }\n        };\n    }\n\n    private _onBeforeRun (): void {\n        if (this.command.actual !== Symbol.for(FUNCTION_MARKER_DESCRIPTION))\n            return;\n\n        this.emit('non-serializable-actual-value', this);\n    }\n\n    public async run (): Promise<void> {\n        this._onBeforeRun();\n\n        this.startTime = new Date().getTime();\n\n        try {\n            await this.fn();\n        }\n\n        catch (err: any) {\n            if (err.name === 'AssertionError' || err.constructor.name === 'AssertionError')\n                throw new ExternalAssertionLibraryError(err, this.callsite);\n\n            if (err.isTestCafeError)\n                err.callsite = this.callsite;\n\n            throw err;\n        }\n    }\n}\n"]}
|