"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"]}