1 | ;
|
2 | var __importDefault = (this && this.__importDefault) || function (mod) {
|
3 | return (mod && mod.__esModule) ? mod : { "default": mod };
|
4 | };
|
5 | var __importStar = (this && this.__importStar) || function (mod) {
|
6 | if (mod && mod.__esModule) return mod;
|
7 | var result = {};
|
8 | if (mod != null) for (var k in mod) if (Object.hasOwnProperty.call(mod, k)) result[k] = mod[k];
|
9 | result["default"] = mod;
|
10 | return result;
|
11 | };
|
12 | Object.defineProperty(exports, "__esModule", { value: true });
|
13 | const lodash_1 = require("lodash");
|
14 | const read_file_relative_1 = require("read-file-relative");
|
15 | const promisify_event_1 = __importDefault(require("promisify-event"));
|
16 | const mustache_1 = __importDefault(require("mustache"));
|
17 | const async_event_emitter_1 = __importDefault(require("../utils/async-event-emitter"));
|
18 | const debug_log_1 = __importDefault(require("./debug-log"));
|
19 | const formattable_adapter_1 = __importDefault(require("../errors/test-run/formattable-adapter"));
|
20 | const error_list_1 = __importDefault(require("../errors/error-list"));
|
21 | const test_run_1 = require("../errors/test-run/");
|
22 | const phase_1 = __importDefault(require("./phase"));
|
23 | const client_messages_1 = __importDefault(require("./client-messages"));
|
24 | const type_1 = __importDefault(require("./commands/type"));
|
25 | const delay_1 = __importDefault(require("../utils/delay"));
|
26 | const marker_symbol_1 = __importDefault(require("./marker-symbol"));
|
27 | const test_run_tracker_1 = __importDefault(require("../api/test-run-tracker"));
|
28 | const phase_2 = __importDefault(require("../role/phase"));
|
29 | const plugin_host_1 = __importDefault(require("../reporter/plugin-host"));
|
30 | const browser_console_messages_1 = __importDefault(require("./browser-console-messages"));
|
31 | const unstable_network_mode_1 = require("../browser/connection/unstable-network-mode");
|
32 | const warning_log_1 = __importDefault(require("../notifications/warning-log"));
|
33 | const warning_message_1 = __importDefault(require("../notifications/warning-message"));
|
34 | const testcafe_hammerhead_1 = require("testcafe-hammerhead");
|
35 | const actions_1 = require("./commands/actions");
|
36 | const INJECTABLES = __importStar(require("../assets/injectables"));
|
37 | const utils_1 = require("../custom-client-scripts/utils");
|
38 | const get_url_1 = __importDefault(require("../custom-client-scripts/get-url"));
|
39 | const string_1 = require("../utils/string");
|
40 | const utils_2 = require("./commands/utils");
|
41 | const types_1 = require("../errors/types");
|
42 | const lazyRequire = require('import-lazy')(require);
|
43 | const SessionController = lazyRequire('./session-controller');
|
44 | const ClientFunctionBuilder = lazyRequire('../client-functions/client-function-builder');
|
45 | const BrowserManipulationQueue = lazyRequire('./browser-manipulation-queue');
|
46 | const TestRunBookmark = lazyRequire('./bookmark');
|
47 | const AssertionExecutor = lazyRequire('../assertions/executor');
|
48 | const actionCommands = lazyRequire('./commands/actions');
|
49 | const browserManipulationCommands = lazyRequire('./commands/browser-manipulation');
|
50 | const serviceCommands = lazyRequire('./commands/service');
|
51 | const { executeJsExpression, executeAsyncJsExpression } = lazyRequire('./execute-js-expression');
|
52 | const TEST_RUN_TEMPLATE = read_file_relative_1.readSync('../client/test-run/index.js.mustache');
|
53 | const IFRAME_TEST_RUN_TEMPLATE = read_file_relative_1.readSync('../client/test-run/iframe.js.mustache');
|
54 | const TEST_DONE_CONFIRMATION_RESPONSE = 'test-done-confirmation';
|
55 | const MAX_RESPONSE_DELAY = 3000;
|
56 | const CHILD_WINDOW_READY_TIMEOUT = 30 * 1000;
|
57 | const ALL_DRIVER_TASKS_ADDED_TO_QUEUE_EVENT = 'all-driver-tasks-added-to-queue';
|
58 | class TestRun extends async_event_emitter_1.default {
|
59 | constructor(test, browserConnection, screenshotCapturer, globalWarningLog, opts) {
|
60 | super();
|
61 | this[marker_symbol_1.default] = true;
|
62 | this.warningLog = new warning_log_1.default(globalWarningLog);
|
63 | this.opts = opts;
|
64 | this.test = test;
|
65 | this.browserConnection = browserConnection;
|
66 | this.phase = phase_1.default.initial;
|
67 | this.driverTaskQueue = [];
|
68 | this.testDoneCommandQueued = false;
|
69 | this.activeDialogHandler = null;
|
70 | this.activeIframeSelector = null;
|
71 | this.speed = this.opts.speed;
|
72 | this.pageLoadTimeout = this.opts.pageLoadTimeout;
|
73 | this.disablePageReloads = test.disablePageReloads || opts.disablePageReloads && test.disablePageReloads !==
|
74 | false;
|
75 | this.disablePageCaching = test.disablePageCaching || opts.disablePageCaching;
|
76 | this.allowMultipleWindows = opts.allowMultipleWindows;
|
77 | this.session = SessionController.getSession(this);
|
78 | this.consoleMessages = new browser_console_messages_1.default();
|
79 | this.pendingRequest = null;
|
80 | this.pendingPageError = null;
|
81 | this.controller = null;
|
82 | this.ctx = Object.create(null);
|
83 | this.fixtureCtx = null;
|
84 | this.currentRoleId = null;
|
85 | this.usedRoleStates = Object.create(null);
|
86 | this.errs = [];
|
87 | this.lastDriverStatusId = null;
|
88 | this.lastDriverStatusResponse = null;
|
89 | this.fileDownloadingHandled = false;
|
90 | this.resolveWaitForFileDownloadingPromise = null;
|
91 | this.addingDriverTasksCount = 0;
|
92 | this.debugging = this.opts.debugMode;
|
93 | this.debugOnFail = this.opts.debugOnFail;
|
94 | this.disableDebugBreakpoints = false;
|
95 | this.debugReporterPluginHost = new plugin_host_1.default({ noColors: false });
|
96 | this.browserManipulationQueue = new BrowserManipulationQueue(browserConnection, screenshotCapturer, this.warningLog);
|
97 | this.debugLog = new debug_log_1.default(this.browserConnection.userAgent);
|
98 | this.quarantine = null;
|
99 | this.debugLogger = this.opts.debugLogger;
|
100 | this._addInjectables();
|
101 | this._initRequestHooks();
|
102 | }
|
103 | _addClientScriptContentWarningsIfNecessary() {
|
104 | const { empty, duplicatedContent } = utils_1.findProblematicScripts(this.test.clientScripts);
|
105 | if (empty.length)
|
106 | this.warningLog.addWarning(warning_message_1.default.clientScriptsWithEmptyContent);
|
107 | if (duplicatedContent.length) {
|
108 | const suffix = string_1.getPluralSuffix(duplicatedContent);
|
109 | const duplicatedContentClientScriptsStr = string_1.getConcatenatedValuesString(duplicatedContent, ',\n ');
|
110 | this.warningLog.addWarning(warning_message_1.default.clientScriptsWithDuplicatedContent, suffix, duplicatedContentClientScriptsStr);
|
111 | }
|
112 | }
|
113 | _addInjectables() {
|
114 | this._addClientScriptContentWarningsIfNecessary();
|
115 | this.injectable.scripts.push(...INJECTABLES.SCRIPTS);
|
116 | this.injectable.userScripts.push(...this.test.clientScripts.map(script => {
|
117 | return {
|
118 | url: get_url_1.default(script),
|
119 | page: script.page
|
120 | };
|
121 | }));
|
122 | this.injectable.styles.push(INJECTABLES.TESTCAFE_UI_STYLES);
|
123 | }
|
124 | get id() {
|
125 | return this.session.id;
|
126 | }
|
127 | get injectable() {
|
128 | return this.session.injectable;
|
129 | }
|
130 | addQuarantineInfo(quarantine) {
|
131 | this.quarantine = quarantine;
|
132 | }
|
133 | addRequestHook(hook) {
|
134 | if (this.requestHooks.indexOf(hook) !== -1)
|
135 | return;
|
136 | this.requestHooks.push(hook);
|
137 | this._initRequestHook(hook);
|
138 | }
|
139 | removeRequestHook(hook) {
|
140 | if (this.requestHooks.indexOf(hook) === -1)
|
141 | return;
|
142 | lodash_1.pull(this.requestHooks, hook);
|
143 | this._disposeRequestHook(hook);
|
144 | }
|
145 | _initRequestHook(hook) {
|
146 | hook.warningLog = this.warningLog;
|
147 | hook._instantiateRequestFilterRules();
|
148 | hook._instantiatedRequestFilterRules.forEach(rule => {
|
149 | this.session.addRequestEventListeners(rule, {
|
150 | onRequest: hook.onRequest.bind(hook),
|
151 | onConfigureResponse: hook._onConfigureResponse.bind(hook),
|
152 | onResponse: hook.onResponse.bind(hook)
|
153 | }, err => this._onRequestHookMethodError(err, hook));
|
154 | });
|
155 | }
|
156 | _onRequestHookMethodError(event, hook) {
|
157 | let err = event.error;
|
158 | const isRequestHookNotImplementedMethodError = err instanceof test_run_1.RequestHookNotImplementedMethodError;
|
159 | if (!isRequestHookNotImplementedMethodError) {
|
160 | const hookClassName = hook.constructor.name;
|
161 | err = new test_run_1.RequestHookUnhandledError(err, hookClassName, event.methodName);
|
162 | }
|
163 | this.addError(err);
|
164 | }
|
165 | _disposeRequestHook(hook) {
|
166 | hook.warningLog = null;
|
167 | hook._instantiatedRequestFilterRules.forEach(rule => {
|
168 | this.session.removeRequestEventListeners(rule);
|
169 | });
|
170 | }
|
171 | _initRequestHooks() {
|
172 | this.requestHooks = Array.from(this.test.requestHooks);
|
173 | this.requestHooks.forEach(hook => this._initRequestHook(hook));
|
174 | }
|
175 | // Hammerhead payload
|
176 | _getPayloadScript() {
|
177 | this.fileDownloadingHandled = false;
|
178 | this.resolveWaitForFileDownloadingPromise = null;
|
179 | return mustache_1.default.render(TEST_RUN_TEMPLATE, {
|
180 | testRunId: JSON.stringify(this.session.id),
|
181 | browserId: JSON.stringify(this.browserConnection.id),
|
182 | browserHeartbeatRelativeUrl: JSON.stringify(this.browserConnection.heartbeatRelativeUrl),
|
183 | browserStatusRelativeUrl: JSON.stringify(this.browserConnection.statusRelativeUrl),
|
184 | browserStatusDoneRelativeUrl: JSON.stringify(this.browserConnection.statusDoneRelativeUrl),
|
185 | browserActiveWindowIdUrl: JSON.stringify(this.browserConnection.activeWindowIdUrl),
|
186 | userAgent: JSON.stringify(this.browserConnection.userAgent),
|
187 | testName: JSON.stringify(this.test.name),
|
188 | fixtureName: JSON.stringify(this.test.fixture.name),
|
189 | selectorTimeout: this.opts.selectorTimeout,
|
190 | pageLoadTimeout: this.pageLoadTimeout,
|
191 | childWindowReadyTimeout: CHILD_WINDOW_READY_TIMEOUT,
|
192 | skipJsErrors: this.opts.skipJsErrors,
|
193 | retryTestPages: this.opts.retryTestPages,
|
194 | speed: this.speed,
|
195 | dialogHandler: JSON.stringify(this.activeDialogHandler)
|
196 | });
|
197 | }
|
198 | _getIframePayloadScript() {
|
199 | return mustache_1.default.render(IFRAME_TEST_RUN_TEMPLATE, {
|
200 | testRunId: JSON.stringify(this.session.id),
|
201 | selectorTimeout: this.opts.selectorTimeout,
|
202 | pageLoadTimeout: this.pageLoadTimeout,
|
203 | retryTestPages: !!this.opts.retryTestPages,
|
204 | speed: this.speed,
|
205 | dialogHandler: JSON.stringify(this.activeDialogHandler)
|
206 | });
|
207 | }
|
208 | // Hammerhead handlers
|
209 | getAuthCredentials() {
|
210 | return this.test.authCredentials;
|
211 | }
|
212 | handleFileDownload() {
|
213 | if (this.resolveWaitForFileDownloadingPromise) {
|
214 | this.resolveWaitForFileDownloadingPromise(true);
|
215 | this.resolveWaitForFileDownloadingPromise = null;
|
216 | }
|
217 | else
|
218 | this.fileDownloadingHandled = true;
|
219 | }
|
220 | handlePageError(ctx, err) {
|
221 | if (ctx.req.headers[unstable_network_mode_1.UNSTABLE_NETWORK_MODE_HEADER]) {
|
222 | ctx.closeWithError(500, err.toString());
|
223 | return;
|
224 | }
|
225 | this.pendingPageError = new test_run_1.PageLoadError(err, ctx.reqOpts.url);
|
226 | ctx.redirect(ctx.toProxyUrl(testcafe_hammerhead_1.SPECIAL_ERROR_PAGE));
|
227 | }
|
228 | // Test function execution
|
229 | async _executeTestFn(phase, fn) {
|
230 | this.phase = phase;
|
231 | try {
|
232 | await fn(this);
|
233 | }
|
234 | catch (err) {
|
235 | let screenshotPath = null;
|
236 | const { screenshots } = this.opts;
|
237 | if (screenshots && screenshots.takeOnFails)
|
238 | screenshotPath = await this.executeCommand(new browserManipulationCommands.TakeScreenshotOnFailCommand());
|
239 | this.addError(err, screenshotPath);
|
240 | return false;
|
241 | }
|
242 | return !this._addPendingPageErrorIfAny();
|
243 | }
|
244 | async _runBeforeHook() {
|
245 | if (this.test.beforeFn)
|
246 | return await this._executeTestFn(phase_1.default.inTestBeforeHook, this.test.beforeFn);
|
247 | if (this.test.fixture.beforeEachFn)
|
248 | return await this._executeTestFn(phase_1.default.inFixtureBeforeEachHook, this.test.fixture.beforeEachFn);
|
249 | return true;
|
250 | }
|
251 | async _runAfterHook() {
|
252 | if (this.test.afterFn)
|
253 | return await this._executeTestFn(phase_1.default.inTestAfterHook, this.test.afterFn);
|
254 | if (this.test.fixture.afterEachFn)
|
255 | return await this._executeTestFn(phase_1.default.inFixtureAfterEachHook, this.test.fixture.afterEachFn);
|
256 | return true;
|
257 | }
|
258 | async start() {
|
259 | test_run_tracker_1.default.activeTestRuns[this.session.id] = this;
|
260 | await this.emit('start');
|
261 | const onDisconnected = err => this._disconnect(err);
|
262 | this.browserConnection.once('disconnected', onDisconnected);
|
263 | await this.once('connected');
|
264 | await this.emit('ready');
|
265 | if (await this._runBeforeHook()) {
|
266 | await this._executeTestFn(phase_1.default.inTest, this.test.fn);
|
267 | await this._runAfterHook();
|
268 | }
|
269 | if (this.disconnected)
|
270 | return;
|
271 | this.browserConnection.removeListener('disconnected', onDisconnected);
|
272 | if (this.errs.length && this.debugOnFail)
|
273 | await this._enqueueSetBreakpointCommand(null, this.debugReporterPluginHost.formatError(this.errs[0]));
|
274 | await this.emit('before-done');
|
275 | await this.executeCommand(new serviceCommands.TestDoneCommand());
|
276 | this._addPendingPageErrorIfAny();
|
277 | this.session.clearRequestEventListeners();
|
278 | this.normalizeRequestHookErrors();
|
279 | delete test_run_tracker_1.default.activeTestRuns[this.session.id];
|
280 | await this.emit('done');
|
281 | }
|
282 | // Errors
|
283 | _addPendingPageErrorIfAny() {
|
284 | if (this.pendingPageError) {
|
285 | this.addError(this.pendingPageError);
|
286 | this.pendingPageError = null;
|
287 | return true;
|
288 | }
|
289 | return false;
|
290 | }
|
291 | _createErrorAdapter(err, screenshotPath) {
|
292 | return new formattable_adapter_1.default(err, {
|
293 | userAgent: this.browserConnection.userAgent,
|
294 | screenshotPath: screenshotPath || '',
|
295 | testRunPhase: this.phase
|
296 | });
|
297 | }
|
298 | addError(err, screenshotPath) {
|
299 | const errList = err instanceof error_list_1.default ? err.items : [err];
|
300 | errList.forEach(item => {
|
301 | const adapter = this._createErrorAdapter(item, screenshotPath);
|
302 | this.errs.push(adapter);
|
303 | });
|
304 | }
|
305 | normalizeRequestHookErrors() {
|
306 | const requestHookErrors = lodash_1.remove(this.errs, e => e.code === types_1.TEST_RUN_ERRORS.requestHookNotImplementedError ||
|
307 | e.code === types_1.TEST_RUN_ERRORS.requestHookUnhandledError);
|
308 | if (!requestHookErrors.length)
|
309 | return;
|
310 | const uniqRequestHookErrors = lodash_1.chain(requestHookErrors)
|
311 | .uniqBy(e => e.hookClassName + e.methodName)
|
312 | .sortBy(['hookClassName', 'methodName'])
|
313 | .value();
|
314 | this.errs = this.errs.concat(uniqRequestHookErrors);
|
315 | }
|
316 | // Task queue
|
317 | _enqueueCommand(command, callsite) {
|
318 | if (this.pendingRequest)
|
319 | this._resolvePendingRequest(command);
|
320 | return new Promise(async (resolve, reject) => {
|
321 | this.addingDriverTasksCount--;
|
322 | this.driverTaskQueue.push({ command, resolve, reject, callsite });
|
323 | if (!this.addingDriverTasksCount)
|
324 | await this.emit(ALL_DRIVER_TASKS_ADDED_TO_QUEUE_EVENT, this.driverTaskQueue.length);
|
325 | });
|
326 | }
|
327 | get driverTaskQueueLength() {
|
328 | return this.addingDriverTasksCount ? promisify_event_1.default(this, ALL_DRIVER_TASKS_ADDED_TO_QUEUE_EVENT) : Promise.resolve(this.driverTaskQueue.length);
|
329 | }
|
330 | async _enqueueBrowserConsoleMessagesCommand(command, callsite) {
|
331 | await this._enqueueCommand(command, callsite);
|
332 | const consoleMessageCopy = this.consoleMessages.getCopy();
|
333 | return consoleMessageCopy[this.browserConnection.activeWindowId];
|
334 | }
|
335 | async _enqueueSetBreakpointCommand(callsite, error) {
|
336 | if (this.browserConnection.isHeadlessBrowser()) {
|
337 | this.warningLog.addWarning(warning_message_1.default.debugInHeadlessError);
|
338 | return;
|
339 | }
|
340 | if (this.debugLogger)
|
341 | this.debugLogger.showBreakpoint(this.session.id, this.browserConnection.userAgent, callsite, error);
|
342 | this.debugging = await this.executeCommand(new serviceCommands.SetBreakpointCommand(!!error), callsite);
|
343 | }
|
344 | _removeAllNonServiceTasks() {
|
345 | this.driverTaskQueue = this.driverTaskQueue.filter(driverTask => utils_2.isServiceCommand(driverTask.command));
|
346 | this.browserManipulationQueue.removeAllNonServiceManipulations();
|
347 | }
|
348 | // Current driver task
|
349 | get currentDriverTask() {
|
350 | return this.driverTaskQueue[0];
|
351 | }
|
352 | _resolveCurrentDriverTask(result) {
|
353 | this.currentDriverTask.resolve(result);
|
354 | this.driverTaskQueue.shift();
|
355 | if (this.testDoneCommandQueued)
|
356 | this._removeAllNonServiceTasks();
|
357 | }
|
358 | _rejectCurrentDriverTask(err) {
|
359 | err.callsite = err.callsite || this.currentDriverTask.callsite;
|
360 | this.currentDriverTask.reject(err);
|
361 | this._removeAllNonServiceTasks();
|
362 | }
|
363 | // Pending request
|
364 | _clearPendingRequest() {
|
365 | if (this.pendingRequest) {
|
366 | clearTimeout(this.pendingRequest.responseTimeout);
|
367 | this.pendingRequest = null;
|
368 | }
|
369 | }
|
370 | _resolvePendingRequest(command) {
|
371 | this.lastDriverStatusResponse = command;
|
372 | this.pendingRequest.resolve(command);
|
373 | this._clearPendingRequest();
|
374 | }
|
375 | // Handle driver request
|
376 | _fulfillCurrentDriverTask(driverStatus) {
|
377 | if (!this.currentDriverTask)
|
378 | return;
|
379 | if (driverStatus.executionError)
|
380 | this._rejectCurrentDriverTask(driverStatus.executionError);
|
381 | else
|
382 | this._resolveCurrentDriverTask(driverStatus.result);
|
383 | }
|
384 | _handlePageErrorStatus(pageError) {
|
385 | if (this.currentDriverTask && utils_2.isCommandRejectableByPageError(this.currentDriverTask.command)) {
|
386 | this._rejectCurrentDriverTask(pageError);
|
387 | this.pendingPageError = null;
|
388 | return true;
|
389 | }
|
390 | this.pendingPageError = this.pendingPageError || pageError;
|
391 | return false;
|
392 | }
|
393 | _handleDriverRequest(driverStatus) {
|
394 | const isTestDone = this.currentDriverTask && this.currentDriverTask.command.type ===
|
395 | type_1.default.testDone;
|
396 | const pageError = this.pendingPageError || driverStatus.pageError;
|
397 | const currentTaskRejectedByError = pageError && this._handlePageErrorStatus(pageError);
|
398 | if (this.disconnected)
|
399 | return new Promise((_, reject) => reject());
|
400 | this.consoleMessages.concat(driverStatus.consoleMessages);
|
401 | if (!currentTaskRejectedByError && driverStatus.isCommandResult) {
|
402 | if (isTestDone) {
|
403 | this._resolveCurrentDriverTask();
|
404 | return TEST_DONE_CONFIRMATION_RESPONSE;
|
405 | }
|
406 | this._fulfillCurrentDriverTask(driverStatus);
|
407 | if (driverStatus.isPendingWindowSwitching)
|
408 | return null;
|
409 | }
|
410 | return this._getCurrentDriverTaskCommand();
|
411 | }
|
412 | _getCurrentDriverTaskCommand() {
|
413 | if (!this.currentDriverTask)
|
414 | return null;
|
415 | const command = this.currentDriverTask.command;
|
416 | if (command.type === type_1.default.navigateTo && command.stateSnapshot)
|
417 | this.session.useStateSnapshot(JSON.parse(command.stateSnapshot));
|
418 | return command;
|
419 | }
|
420 | // Execute command
|
421 | _executeJsExpression(command) {
|
422 | const resultVariableName = command.resultVariableName;
|
423 | let expression = command.expression;
|
424 | if (resultVariableName)
|
425 | expression = `${resultVariableName} = ${expression}, ${resultVariableName}`;
|
426 | return executeJsExpression(expression, this, { skipVisibilityCheck: false });
|
427 | }
|
428 | async _executeAssertion(command, callsite) {
|
429 | const assertionTimeout = command.options.timeout ===
|
430 | void 0 ? this.opts.assertionTimeout : command.options.timeout;
|
431 | const executor = new AssertionExecutor(command, assertionTimeout, callsite);
|
432 | executor.once('start-assertion-retries', timeout => this.executeCommand(new serviceCommands.ShowAssertionRetriesStatusCommand(timeout)));
|
433 | executor.once('end-assertion-retries', success => this.executeCommand(new serviceCommands.HideAssertionRetriesStatusCommand(success)));
|
434 | const executeFn = this.decoratePreventEmitActionEvents(() => executor.run(), { prevent: true });
|
435 | return await executeFn();
|
436 | }
|
437 | _adjustConfigurationWithCommand(command) {
|
438 | if (command.type === type_1.default.testDone) {
|
439 | this.testDoneCommandQueued = true;
|
440 | if (this.debugLogger)
|
441 | this.debugLogger.hideBreakpoint(this.session.id);
|
442 | }
|
443 | else if (command.type === type_1.default.setNativeDialogHandler)
|
444 | this.activeDialogHandler = command.dialogHandler;
|
445 | else if (command.type === type_1.default.switchToIframe)
|
446 | this.activeIframeSelector = command.selector;
|
447 | else if (command.type === type_1.default.switchToMainWindow)
|
448 | this.activeIframeSelector = null;
|
449 | else if (command.type === type_1.default.setTestSpeed)
|
450 | this.speed = command.speed;
|
451 | else if (command.type === type_1.default.setPageLoadTimeout)
|
452 | this.pageLoadTimeout = command.duration;
|
453 | else if (command.type === type_1.default.debug)
|
454 | this.debugging = true;
|
455 | }
|
456 | async _adjustScreenshotCommand(command) {
|
457 | const browserId = this.browserConnection.id;
|
458 | const { hasChromelessScreenshots } = await this.browserConnection.provider.hasCustomActionForBrowser(browserId);
|
459 | if (!hasChromelessScreenshots)
|
460 | command.generateScreenshotMark();
|
461 | }
|
462 | async _setBreakpointIfNecessary(command, callsite) {
|
463 | if (!this.disableDebugBreakpoints && this.debugging && utils_2.canSetDebuggerBreakpointBeforeCommand(command))
|
464 | await this._enqueueSetBreakpointCommand(callsite);
|
465 | }
|
466 | async executeAction(actionName, command, callsite) {
|
467 | let error = null;
|
468 | let result = null;
|
469 | await this.emitActionStart(actionName, command);
|
470 | try {
|
471 | result = await this.executeCommand(command, callsite);
|
472 | }
|
473 | catch (err) {
|
474 | error = err;
|
475 | }
|
476 | await this.emitActionDone(actionName, command, error);
|
477 | if (error)
|
478 | throw error;
|
479 | return result;
|
480 | }
|
481 | async executeCommand(command, callsite) {
|
482 | this.debugLog.command(command);
|
483 | if (this.pendingPageError && utils_2.isCommandRejectableByPageError(command))
|
484 | return this._rejectCommandWithPageError(callsite);
|
485 | if (utils_2.isExecutableOnClientCommand(command))
|
486 | this.addingDriverTasksCount++;
|
487 | this._adjustConfigurationWithCommand(command);
|
488 | await this._setBreakpointIfNecessary(command, callsite);
|
489 | if (utils_2.isScreenshotCommand(command)) {
|
490 | if (this.opts.disableScreenshots) {
|
491 | this.warningLog.addWarning(warning_message_1.default.screenshotsDisabled);
|
492 | return null;
|
493 | }
|
494 | await this._adjustScreenshotCommand(command);
|
495 | }
|
496 | if (utils_2.isBrowserManipulationCommand(command)) {
|
497 | this.browserManipulationQueue.push(command);
|
498 | if (utils_2.isResizeWindowCommand(command) && this.opts.videoPath)
|
499 | this.warningLog.addWarning(warning_message_1.default.videoBrowserResizing, this.test.name);
|
500 | }
|
501 | if (command.type === type_1.default.wait)
|
502 | return delay_1.default(command.timeout);
|
503 | if (command.type === type_1.default.setPageLoadTimeout)
|
504 | return null;
|
505 | if (command.type === type_1.default.debug)
|
506 | return await this._enqueueSetBreakpointCommand(callsite);
|
507 | if (command.type === type_1.default.useRole) {
|
508 | let fn = () => this._useRole(command.role, callsite);
|
509 | fn = this.decoratePreventEmitActionEvents(fn, { prevent: true });
|
510 | fn = this.decorateDisableDebugBreakpoints(fn, { disable: true });
|
511 | return await fn();
|
512 | }
|
513 | if (command.type === type_1.default.assertion)
|
514 | return this._executeAssertion(command, callsite);
|
515 | if (command.type === type_1.default.executeExpression)
|
516 | return await this._executeJsExpression(command, callsite);
|
517 | if (command.type === type_1.default.executeAsyncExpression)
|
518 | return await executeAsyncJsExpression(command.expression, this, callsite);
|
519 | if (command.type === type_1.default.getBrowserConsoleMessages)
|
520 | return await this._enqueueBrowserConsoleMessagesCommand(command, callsite);
|
521 | return this._enqueueCommand(command, callsite);
|
522 | }
|
523 | _rejectCommandWithPageError(callsite) {
|
524 | const err = this.pendingPageError;
|
525 | err.callsite = callsite;
|
526 | this.pendingPageError = null;
|
527 | return Promise.reject(err);
|
528 | }
|
529 | _decorateWithFlag(fn, flagName, value) {
|
530 | return async () => {
|
531 | this[flagName] = value;
|
532 | try {
|
533 | return await fn();
|
534 | }
|
535 | catch (err) {
|
536 | throw err;
|
537 | }
|
538 | finally {
|
539 | this[flagName] = !value;
|
540 | }
|
541 | };
|
542 | }
|
543 | decoratePreventEmitActionEvents(fn, { prevent }) {
|
544 | return this._decorateWithFlag(fn, 'preventEmitActionEvents', prevent);
|
545 | }
|
546 | decorateDisableDebugBreakpoints(fn, { disable }) {
|
547 | return this._decorateWithFlag(fn, 'disableDebugBreakpoints', disable);
|
548 | }
|
549 | // Role management
|
550 | async getStateSnapshot() {
|
551 | const state = this.session.getStateSnapshot();
|
552 | state.storages = await this.executeCommand(new serviceCommands.BackupStoragesCommand());
|
553 | return state;
|
554 | }
|
555 | async switchToCleanRun(url) {
|
556 | this.ctx = Object.create(null);
|
557 | this.fixtureCtx = Object.create(null);
|
558 | this.consoleMessages = new browser_console_messages_1.default();
|
559 | this.session.useStateSnapshot(testcafe_hammerhead_1.StateSnapshot.empty());
|
560 | if (this.speed !== this.opts.speed) {
|
561 | const setSpeedCommand = new actionCommands.SetTestSpeedCommand({ speed: this.opts.speed });
|
562 | await this.executeCommand(setSpeedCommand);
|
563 | }
|
564 | if (this.pageLoadTimeout !== this.opts.pageLoadTimeout) {
|
565 | const setPageLoadTimeoutCommand = new actionCommands.SetPageLoadTimeoutCommand({ duration: this.opts.pageLoadTimeout });
|
566 | await this.executeCommand(setPageLoadTimeoutCommand);
|
567 | }
|
568 | await this.navigateToUrl(url, true);
|
569 | if (this.activeDialogHandler) {
|
570 | const removeDialogHandlerCommand = new actionCommands.SetNativeDialogHandlerCommand({ dialogHandler: { fn: null } });
|
571 | await this.executeCommand(removeDialogHandlerCommand);
|
572 | }
|
573 | }
|
574 | async navigateToUrl(url, forceReload, stateSnapshot) {
|
575 | const navigateCommand = new actions_1.NavigateToCommand({ url, forceReload, stateSnapshot });
|
576 | await this.executeCommand(navigateCommand);
|
577 | }
|
578 | async _getStateSnapshotFromRole(role) {
|
579 | const prevPhase = this.phase;
|
580 | this.phase = phase_1.default.inRoleInitializer;
|
581 | if (role.phase === phase_2.default.uninitialized)
|
582 | await role.initialize(this);
|
583 | else if (role.phase === phase_2.default.pendingInitialization)
|
584 | await promisify_event_1.default(role, 'initialized');
|
585 | if (role.initErr)
|
586 | throw role.initErr;
|
587 | this.phase = prevPhase;
|
588 | return role.stateSnapshot;
|
589 | }
|
590 | async _useRole(role, callsite) {
|
591 | if (this.phase === phase_1.default.inRoleInitializer)
|
592 | throw new test_run_1.RoleSwitchInRoleInitializerError(callsite);
|
593 | const bookmark = new TestRunBookmark(this, role);
|
594 | await bookmark.init();
|
595 | if (this.currentRoleId)
|
596 | this.usedRoleStates[this.currentRoleId] = await this.getStateSnapshot();
|
597 | const stateSnapshot = this.usedRoleStates[role.id] || await this._getStateSnapshotFromRole(role);
|
598 | this.session.useStateSnapshot(stateSnapshot);
|
599 | this.currentRoleId = role.id;
|
600 | await bookmark.restore(callsite, stateSnapshot);
|
601 | }
|
602 | // Get current URL
|
603 | async getCurrentUrl() {
|
604 | const builder = new ClientFunctionBuilder(() => {
|
605 | /* eslint-disable no-undef */
|
606 | return window.location.href;
|
607 | /* eslint-enable no-undef */
|
608 | }, { boundTestRun: this });
|
609 | const getLocation = builder.getFunction();
|
610 | return await getLocation();
|
611 | }
|
612 | _disconnect(err) {
|
613 | this.disconnected = true;
|
614 | if (this.currentDriverTask)
|
615 | this._rejectCurrentDriverTask(err);
|
616 | this.emit('disconnected', err);
|
617 | delete test_run_tracker_1.default.activeTestRuns[this.session.id];
|
618 | }
|
619 | async emitActionStart(apiActionName, command) {
|
620 | if (!this.preventEmitActionEvents)
|
621 | await this.emit('action-start', { command, apiActionName });
|
622 | }
|
623 | async emitActionDone(apiActionName, command, errors) {
|
624 | if (!this.preventEmitActionEvents)
|
625 | await this.emit('action-done', { command, apiActionName, errors });
|
626 | }
|
627 | }
|
628 | exports.default = TestRun;
|
629 | // Service message handlers
|
630 | const ServiceMessages = TestRun.prototype;
|
631 | // NOTE: this function is time-critical and must return ASAP to avoid client disconnection
|
632 | ServiceMessages[client_messages_1.default.ready] = function (msg) {
|
633 | this.debugLog.driverMessage(msg);
|
634 | this.emit('connected');
|
635 | this._clearPendingRequest();
|
636 | // NOTE: the driver sends the status for the second time if it didn't get a response at the
|
637 | // first try. This is possible when the page was unloaded after the driver sent the status.
|
638 | if (msg.status.id === this.lastDriverStatusId)
|
639 | return this.lastDriverStatusResponse;
|
640 | this.lastDriverStatusId = msg.status.id;
|
641 | this.lastDriverStatusResponse = this._handleDriverRequest(msg.status);
|
642 | if (this.lastDriverStatusResponse || msg.status.isPendingWindowSwitching)
|
643 | return this.lastDriverStatusResponse;
|
644 | // NOTE: we send an empty response after the MAX_RESPONSE_DELAY timeout is exceeded to keep connection
|
645 | // with the client and prevent the response timeout exception on the client side
|
646 | const responseTimeout = setTimeout(() => this._resolvePendingRequest(null), MAX_RESPONSE_DELAY);
|
647 | return new Promise((resolve, reject) => {
|
648 | this.pendingRequest = { resolve, reject, responseTimeout };
|
649 | });
|
650 | };
|
651 | ServiceMessages[client_messages_1.default.readyForBrowserManipulation] = async function (msg) {
|
652 | this.debugLog.driverMessage(msg);
|
653 | let result = null;
|
654 | let error = null;
|
655 | try {
|
656 | result = await this.browserManipulationQueue.executePendingManipulation(msg);
|
657 | }
|
658 | catch (err) {
|
659 | error = err;
|
660 | }
|
661 | return { result, error };
|
662 | };
|
663 | ServiceMessages[client_messages_1.default.waitForFileDownload] = function (msg) {
|
664 | this.debugLog.driverMessage(msg);
|
665 | return new Promise(resolve => {
|
666 | if (this.fileDownloadingHandled) {
|
667 | this.fileDownloadingHandled = false;
|
668 | resolve(true);
|
669 | }
|
670 | else
|
671 | this.resolveWaitForFileDownloadingPromise = resolve;
|
672 | });
|
673 | };
|
674 | module.exports = exports.default;
|
675 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/test-run/index.js"],"names":[],"mappings":"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","sourcesContent":["import { pull, remove, chain } from 'lodash';\nimport { readSync as read } from 'read-file-relative';\nimport promisifyEvent from 'promisify-event';\nimport Mustache from 'mustache';\nimport AsyncEventEmitter from '../utils/async-event-emitter';\nimport TestRunDebugLog from './debug-log';\nimport TestRunErrorFormattableAdapter from '../errors/test-run/formattable-adapter';\nimport TestCafeErrorList from '../errors/error-list';\nimport {\n    RequestHookUnhandledError,\n    PageLoadError,\n    RequestHookNotImplementedMethodError,\n    RoleSwitchInRoleInitializerError\n} from '../errors/test-run/';\nimport PHASE from './phase';\nimport CLIENT_MESSAGES from './client-messages';\nimport COMMAND_TYPE from './commands/type';\nimport delay from '../utils/delay';\nimport testRunMarker from './marker-symbol';\nimport testRunTracker from '../api/test-run-tracker';\nimport ROLE_PHASE from '../role/phase';\nimport ReporterPluginHost from '../reporter/plugin-host';\nimport BrowserConsoleMessages from './browser-console-messages';\nimport { UNSTABLE_NETWORK_MODE_HEADER } from '../browser/connection/unstable-network-mode';\nimport WarningLog from '../notifications/warning-log';\nimport WARNING_MESSAGE from '../notifications/warning-message';\nimport { StateSnapshot, SPECIAL_ERROR_PAGE } from 'testcafe-hammerhead';\nimport { NavigateToCommand } from './commands/actions';\nimport * as INJECTABLES from '../assets/injectables';\nimport { findProblematicScripts } from '../custom-client-scripts/utils';\nimport getCustomClientScriptUrl from '../custom-client-scripts/get-url';\nimport { getPluralSuffix, getConcatenatedValuesString } from '../utils/string';\n\nimport {\n    isCommandRejectableByPageError,\n    isBrowserManipulationCommand,\n    isScreenshotCommand,\n    isServiceCommand,\n    canSetDebuggerBreakpointBeforeCommand,\n    isExecutableOnClientCommand,\n    isResizeWindowCommand\n} from './commands/utils';\n\nimport { TEST_RUN_ERRORS } from '../errors/types';\n\nconst lazyRequire                 = require('import-lazy')(require);\nconst SessionController           = lazyRequire('./session-controller');\nconst ClientFunctionBuilder       = lazyRequire('../client-functions/client-function-builder');\nconst BrowserManipulationQueue    = lazyRequire('./browser-manipulation-queue');\nconst TestRunBookmark             = lazyRequire('./bookmark');\nconst AssertionExecutor           = lazyRequire('../assertions/executor');\nconst actionCommands              = lazyRequire('./commands/actions');\nconst browserManipulationCommands = lazyRequire('./commands/browser-manipulation');\nconst serviceCommands             = lazyRequire('./commands/service');\n\nconst { executeJsExpression, executeAsyncJsExpression } = lazyRequire('./execute-js-expression');\n\nconst TEST_RUN_TEMPLATE               = read('../client/test-run/index.js.mustache');\nconst IFRAME_TEST_RUN_TEMPLATE        = read('../client/test-run/iframe.js.mustache');\nconst TEST_DONE_CONFIRMATION_RESPONSE = 'test-done-confirmation';\nconst MAX_RESPONSE_DELAY              = 3000;\nconst CHILD_WINDOW_READY_TIMEOUT      = 30 * 1000;\n\nconst ALL_DRIVER_TASKS_ADDED_TO_QUEUE_EVENT = 'all-driver-tasks-added-to-queue';\n\nexport default class TestRun extends AsyncEventEmitter {\n    constructor (test, browserConnection, screenshotCapturer, globalWarningLog, opts) {\n        super();\n\n        this[testRunMarker] = true;\n\n        this.warningLog = new WarningLog(globalWarningLog);\n\n        this.opts              = opts;\n        this.test              = test;\n        this.browserConnection = browserConnection;\n\n        this.phase = PHASE.initial;\n\n        this.driverTaskQueue       = [];\n        this.testDoneCommandQueued = false;\n\n        this.activeDialogHandler  = null;\n        this.activeIframeSelector = null;\n        this.speed                = this.opts.speed;\n        this.pageLoadTimeout      = this.opts.pageLoadTimeout;\n\n        this.disablePageReloads   = test.disablePageReloads || opts.disablePageReloads && test.disablePageReloads !==\n                                    false;\n        this.disablePageCaching   = test.disablePageCaching || opts.disablePageCaching;\n        this.allowMultipleWindows = opts.allowMultipleWindows;\n\n        this.session = SessionController.getSession(this);\n\n        this.consoleMessages = new BrowserConsoleMessages();\n\n        this.pendingRequest   = null;\n        this.pendingPageError = null;\n\n        this.controller = null;\n        this.ctx        = Object.create(null);\n        this.fixtureCtx = null;\n\n        this.currentRoleId  = null;\n        this.usedRoleStates = Object.create(null);\n\n        this.errs = [];\n\n        this.lastDriverStatusId       = null;\n        this.lastDriverStatusResponse = null;\n\n        this.fileDownloadingHandled               = false;\n        this.resolveWaitForFileDownloadingPromise = null;\n\n        this.addingDriverTasksCount = 0;\n\n        this.debugging               = this.opts.debugMode;\n        this.debugOnFail             = this.opts.debugOnFail;\n        this.disableDebugBreakpoints = false;\n        this.debugReporterPluginHost = new ReporterPluginHost({ noColors: false });\n\n        this.browserManipulationQueue = new BrowserManipulationQueue(browserConnection, screenshotCapturer, this.warningLog);\n\n        this.debugLog = new TestRunDebugLog(this.browserConnection.userAgent);\n\n        this.quarantine  = null;\n\n        this.debugLogger = this.opts.debugLogger;\n\n        this._addInjectables();\n        this._initRequestHooks();\n    }\n\n    _addClientScriptContentWarningsIfNecessary () {\n        const { empty, duplicatedContent } = findProblematicScripts(this.test.clientScripts);\n\n        if (empty.length)\n            this.warningLog.addWarning(WARNING_MESSAGE.clientScriptsWithEmptyContent);\n\n        if (duplicatedContent.length) {\n            const suffix                            = getPluralSuffix(duplicatedContent);\n            const duplicatedContentClientScriptsStr = getConcatenatedValuesString(duplicatedContent, ',\\n ');\n\n            this.warningLog.addWarning(WARNING_MESSAGE.clientScriptsWithDuplicatedContent, suffix, duplicatedContentClientScriptsStr);\n        }\n    }\n\n    _addInjectables () {\n        this._addClientScriptContentWarningsIfNecessary();\n        this.injectable.scripts.push(...INJECTABLES.SCRIPTS);\n        this.injectable.userScripts.push(...this.test.clientScripts.map(script => {\n            return {\n                url:  getCustomClientScriptUrl(script),\n                page: script.page\n            };\n        }));\n        this.injectable.styles.push(INJECTABLES.TESTCAFE_UI_STYLES);\n    }\n\n    get id () {\n        return this.session.id;\n    }\n\n    get injectable () {\n        return this.session.injectable;\n    }\n\n    addQuarantineInfo (quarantine) {\n        this.quarantine = quarantine;\n    }\n\n    addRequestHook (hook) {\n        if (this.requestHooks.indexOf(hook) !== -1)\n            return;\n\n        this.requestHooks.push(hook);\n        this._initRequestHook(hook);\n    }\n\n    removeRequestHook (hook) {\n        if (this.requestHooks.indexOf(hook) === -1)\n            return;\n\n        pull(this.requestHooks, hook);\n        this._disposeRequestHook(hook);\n    }\n\n    _initRequestHook (hook) {\n        hook.warningLog = this.warningLog;\n\n        hook._instantiateRequestFilterRules();\n        hook._instantiatedRequestFilterRules.forEach(rule => {\n            this.session.addRequestEventListeners(rule, {\n                onRequest:           hook.onRequest.bind(hook),\n                onConfigureResponse: hook._onConfigureResponse.bind(hook),\n                onResponse:          hook.onResponse.bind(hook)\n            }, err => this._onRequestHookMethodError(err, hook));\n        });\n    }\n\n    _onRequestHookMethodError (event, hook) {\n        let err                                      = event.error;\n        const isRequestHookNotImplementedMethodError = err instanceof RequestHookNotImplementedMethodError;\n\n        if (!isRequestHookNotImplementedMethodError) {\n            const hookClassName = hook.constructor.name;\n\n            err = new RequestHookUnhandledError(err, hookClassName, event.methodName);\n        }\n\n        this.addError(err);\n    }\n\n    _disposeRequestHook (hook) {\n        hook.warningLog = null;\n\n        hook._instantiatedRequestFilterRules.forEach(rule => {\n            this.session.removeRequestEventListeners(rule);\n        });\n    }\n\n    _initRequestHooks () {\n        this.requestHooks = Array.from(this.test.requestHooks);\n\n        this.requestHooks.forEach(hook => this._initRequestHook(hook));\n    }\n\n    // Hammerhead payload\n    _getPayloadScript () {\n        this.fileDownloadingHandled               = false;\n        this.resolveWaitForFileDownloadingPromise = null;\n\n        return Mustache.render(TEST_RUN_TEMPLATE, {\n            testRunId:                    JSON.stringify(this.session.id),\n            browserId:                    JSON.stringify(this.browserConnection.id),\n            browserHeartbeatRelativeUrl:  JSON.stringify(this.browserConnection.heartbeatRelativeUrl),\n            browserStatusRelativeUrl:     JSON.stringify(this.browserConnection.statusRelativeUrl),\n            browserStatusDoneRelativeUrl: JSON.stringify(this.browserConnection.statusDoneRelativeUrl),\n            browserActiveWindowIdUrl:     JSON.stringify(this.browserConnection.activeWindowIdUrl),\n            userAgent:                    JSON.stringify(this.browserConnection.userAgent),\n            testName:                     JSON.stringify(this.test.name),\n            fixtureName:                  JSON.stringify(this.test.fixture.name),\n            selectorTimeout:              this.opts.selectorTimeout,\n            pageLoadTimeout:              this.pageLoadTimeout,\n            childWindowReadyTimeout:      CHILD_WINDOW_READY_TIMEOUT,\n            skipJsErrors:                 this.opts.skipJsErrors,\n            retryTestPages:               this.opts.retryTestPages,\n            speed:                        this.speed,\n            dialogHandler:                JSON.stringify(this.activeDialogHandler)\n        });\n    }\n\n    _getIframePayloadScript () {\n        return Mustache.render(IFRAME_TEST_RUN_TEMPLATE, {\n            testRunId:       JSON.stringify(this.session.id),\n            selectorTimeout: this.opts.selectorTimeout,\n            pageLoadTimeout: this.pageLoadTimeout,\n            retryTestPages:  !!this.opts.retryTestPages,\n            speed:           this.speed,\n            dialogHandler:   JSON.stringify(this.activeDialogHandler)\n        });\n    }\n\n    // Hammerhead handlers\n    getAuthCredentials () {\n        return this.test.authCredentials;\n    }\n\n    handleFileDownload () {\n        if (this.resolveWaitForFileDownloadingPromise) {\n            this.resolveWaitForFileDownloadingPromise(true);\n            this.resolveWaitForFileDownloadingPromise = null;\n        }\n        else\n            this.fileDownloadingHandled = true;\n    }\n\n    handlePageError (ctx, err) {\n        if (ctx.req.headers[UNSTABLE_NETWORK_MODE_HEADER]) {\n            ctx.closeWithError(500, err.toString());\n            return;\n        }\n\n        this.pendingPageError = new PageLoadError(err, ctx.reqOpts.url);\n\n        ctx.redirect(ctx.toProxyUrl(SPECIAL_ERROR_PAGE));\n    }\n\n    // Test function execution\n    async _executeTestFn (phase, fn) {\n        this.phase = phase;\n\n        try {\n            await fn(this);\n        }\n        catch (err) {\n            let screenshotPath = null;\n\n            const { screenshots } = this.opts;\n\n            if (screenshots && screenshots.takeOnFails)\n                screenshotPath = await this.executeCommand(new browserManipulationCommands.TakeScreenshotOnFailCommand());\n\n            this.addError(err, screenshotPath);\n            return false;\n        }\n\n        return !this._addPendingPageErrorIfAny();\n    }\n\n    async _runBeforeHook () {\n        if (this.test.beforeFn)\n            return await this._executeTestFn(PHASE.inTestBeforeHook, this.test.beforeFn);\n\n        if (this.test.fixture.beforeEachFn)\n            return await this._executeTestFn(PHASE.inFixtureBeforeEachHook, this.test.fixture.beforeEachFn);\n\n        return true;\n    }\n\n    async _runAfterHook () {\n        if (this.test.afterFn)\n            return await this._executeTestFn(PHASE.inTestAfterHook, this.test.afterFn);\n\n        if (this.test.fixture.afterEachFn)\n            return await this._executeTestFn(PHASE.inFixtureAfterEachHook, this.test.fixture.afterEachFn);\n\n        return true;\n    }\n\n    async start () {\n        testRunTracker.activeTestRuns[this.session.id] = this;\n\n        await this.emit('start');\n\n        const onDisconnected = err => this._disconnect(err);\n\n        this.browserConnection.once('disconnected', onDisconnected);\n\n        await this.once('connected');\n\n        await this.emit('ready');\n\n        if (await this._runBeforeHook()) {\n            await this._executeTestFn(PHASE.inTest, this.test.fn);\n            await this._runAfterHook();\n        }\n\n        if (this.disconnected)\n            return;\n\n        this.browserConnection.removeListener('disconnected', onDisconnected);\n\n        if (this.errs.length && this.debugOnFail)\n            await this._enqueueSetBreakpointCommand(null, this.debugReporterPluginHost.formatError(this.errs[0]));\n\n        await this.emit('before-done');\n\n        await this.executeCommand(new serviceCommands.TestDoneCommand());\n\n        this._addPendingPageErrorIfAny();\n        this.session.clearRequestEventListeners();\n        this.normalizeRequestHookErrors();\n\n        delete testRunTracker.activeTestRuns[this.session.id];\n\n        await this.emit('done');\n    }\n\n    // Errors\n    _addPendingPageErrorIfAny () {\n        if (this.pendingPageError) {\n            this.addError(this.pendingPageError);\n            this.pendingPageError = null;\n            return true;\n        }\n\n        return false;\n    }\n\n    _createErrorAdapter (err, screenshotPath) {\n        return new TestRunErrorFormattableAdapter(err, {\n            userAgent:      this.browserConnection.userAgent,\n            screenshotPath: screenshotPath || '',\n            testRunPhase:   this.phase\n        });\n    }\n\n    addError (err, screenshotPath) {\n        const errList = err instanceof TestCafeErrorList ? err.items : [err];\n\n        errList.forEach(item => {\n            const adapter = this._createErrorAdapter(item, screenshotPath);\n\n            this.errs.push(adapter);\n        });\n    }\n\n    normalizeRequestHookErrors () {\n        const requestHookErrors = remove(this.errs, e =>\n            e.code === TEST_RUN_ERRORS.requestHookNotImplementedError ||\n            e.code === TEST_RUN_ERRORS.requestHookUnhandledError);\n\n        if (!requestHookErrors.length)\n            return;\n\n        const uniqRequestHookErrors = chain(requestHookErrors)\n            .uniqBy(e => e.hookClassName + e.methodName)\n            .sortBy(['hookClassName', 'methodName'])\n            .value();\n\n        this.errs = this.errs.concat(uniqRequestHookErrors);\n    }\n\n    // Task queue\n    _enqueueCommand (command, callsite) {\n        if (this.pendingRequest)\n            this._resolvePendingRequest(command);\n\n        return new Promise(async (resolve, reject) => {\n            this.addingDriverTasksCount--;\n            this.driverTaskQueue.push({ command, resolve, reject, callsite });\n\n            if (!this.addingDriverTasksCount)\n                await this.emit(ALL_DRIVER_TASKS_ADDED_TO_QUEUE_EVENT, this.driverTaskQueue.length);\n        });\n    }\n\n    get driverTaskQueueLength () {\n        return this.addingDriverTasksCount ? promisifyEvent(this, ALL_DRIVER_TASKS_ADDED_TO_QUEUE_EVENT) : Promise.resolve(this.driverTaskQueue.length);\n    }\n\n    async _enqueueBrowserConsoleMessagesCommand (command, callsite) {\n        await this._enqueueCommand(command, callsite);\n\n        const consoleMessageCopy = this.consoleMessages.getCopy();\n\n        return consoleMessageCopy[this.browserConnection.activeWindowId];\n    }\n\n    async _enqueueSetBreakpointCommand (callsite, error) {\n        if (this.browserConnection.isHeadlessBrowser()) {\n            this.warningLog.addWarning(WARNING_MESSAGE.debugInHeadlessError);\n            return;\n        }\n\n        if (this.debugLogger)\n            this.debugLogger.showBreakpoint(this.session.id, this.browserConnection.userAgent, callsite, error);\n\n        this.debugging = await this.executeCommand(new serviceCommands.SetBreakpointCommand(!!error), callsite);\n    }\n\n    _removeAllNonServiceTasks () {\n        this.driverTaskQueue = this.driverTaskQueue.filter(driverTask => isServiceCommand(driverTask.command));\n\n        this.browserManipulationQueue.removeAllNonServiceManipulations();\n    }\n\n    // Current driver task\n    get currentDriverTask () {\n        return this.driverTaskQueue[0];\n    }\n\n    _resolveCurrentDriverTask (result) {\n        this.currentDriverTask.resolve(result);\n        this.driverTaskQueue.shift();\n\n        if (this.testDoneCommandQueued)\n            this._removeAllNonServiceTasks();\n    }\n\n    _rejectCurrentDriverTask (err) {\n        err.callsite = err.callsite || this.currentDriverTask.callsite;\n\n        this.currentDriverTask.reject(err);\n        this._removeAllNonServiceTasks();\n    }\n\n    // Pending request\n    _clearPendingRequest () {\n        if (this.pendingRequest) {\n            clearTimeout(this.pendingRequest.responseTimeout);\n            this.pendingRequest = null;\n        }\n    }\n\n    _resolvePendingRequest (command) {\n        this.lastDriverStatusResponse = command;\n        this.pendingRequest.resolve(command);\n        this._clearPendingRequest();\n    }\n\n    // Handle driver request\n    _fulfillCurrentDriverTask (driverStatus) {\n        if (!this.currentDriverTask)\n            return;\n\n        if (driverStatus.executionError)\n            this._rejectCurrentDriverTask(driverStatus.executionError);\n        else\n            this._resolveCurrentDriverTask(driverStatus.result);\n    }\n\n    _handlePageErrorStatus (pageError) {\n        if (this.currentDriverTask && isCommandRejectableByPageError(this.currentDriverTask.command)) {\n            this._rejectCurrentDriverTask(pageError);\n            this.pendingPageError = null;\n\n            return true;\n        }\n\n        this.pendingPageError = this.pendingPageError || pageError;\n\n        return false;\n    }\n\n    _handleDriverRequest (driverStatus) {\n        const isTestDone                 = this.currentDriverTask && this.currentDriverTask.command.type ===\n                                           COMMAND_TYPE.testDone;\n        const pageError                  = this.pendingPageError || driverStatus.pageError;\n        const currentTaskRejectedByError = pageError && this._handlePageErrorStatus(pageError);\n\n        if (this.disconnected)\n            return new Promise((_, reject) => reject());\n\n        this.consoleMessages.concat(driverStatus.consoleMessages);\n\n        if (!currentTaskRejectedByError && driverStatus.isCommandResult) {\n            if (isTestDone) {\n                this._resolveCurrentDriverTask();\n\n                return TEST_DONE_CONFIRMATION_RESPONSE;\n            }\n\n            this._fulfillCurrentDriverTask(driverStatus);\n\n            if (driverStatus.isPendingWindowSwitching)\n                return null;\n        }\n\n        return this._getCurrentDriverTaskCommand();\n    }\n\n    _getCurrentDriverTaskCommand () {\n        if (!this.currentDriverTask)\n            return null;\n\n        const command = this.currentDriverTask.command;\n\n        if (command.type === COMMAND_TYPE.navigateTo && command.stateSnapshot)\n            this.session.useStateSnapshot(JSON.parse(command.stateSnapshot));\n\n        return command;\n    }\n\n    // Execute command\n    _executeJsExpression (command) {\n        const resultVariableName = command.resultVariableName;\n        let expression           = command.expression;\n\n        if (resultVariableName)\n            expression = `${resultVariableName} = ${expression}, ${resultVariableName}`;\n\n        return executeJsExpression(expression, this, { skipVisibilityCheck: false });\n    }\n\n    async _executeAssertion (command, callsite) {\n        const assertionTimeout = command.options.timeout ===\n                                 void 0 ? this.opts.assertionTimeout : command.options.timeout;\n        const executor         = new AssertionExecutor(command, assertionTimeout, callsite);\n\n        executor.once('start-assertion-retries', timeout => this.executeCommand(new serviceCommands.ShowAssertionRetriesStatusCommand(timeout)));\n        executor.once('end-assertion-retries', success => this.executeCommand(new serviceCommands.HideAssertionRetriesStatusCommand(success)));\n\n        const executeFn = this.decoratePreventEmitActionEvents(() => executor.run(), { prevent: true });\n\n        return await executeFn();\n    }\n\n    _adjustConfigurationWithCommand (command) {\n        if (command.type === COMMAND_TYPE.testDone) {\n            this.testDoneCommandQueued = true;\n            if (this.debugLogger)\n                this.debugLogger.hideBreakpoint(this.session.id);\n        }\n\n        else if (command.type === COMMAND_TYPE.setNativeDialogHandler)\n            this.activeDialogHandler = command.dialogHandler;\n\n        else if (command.type === COMMAND_TYPE.switchToIframe)\n            this.activeIframeSelector = command.selector;\n\n        else if (command.type === COMMAND_TYPE.switchToMainWindow)\n            this.activeIframeSelector = null;\n\n        else if (command.type === COMMAND_TYPE.setTestSpeed)\n            this.speed = command.speed;\n\n        else if (command.type === COMMAND_TYPE.setPageLoadTimeout)\n            this.pageLoadTimeout = command.duration;\n\n        else if (command.type === COMMAND_TYPE.debug)\n            this.debugging = true;\n    }\n\n    async _adjustScreenshotCommand (command) {\n        const browserId                    = this.browserConnection.id;\n        const { hasChromelessScreenshots } = await this.browserConnection.provider.hasCustomActionForBrowser(browserId);\n\n        if (!hasChromelessScreenshots)\n            command.generateScreenshotMark();\n    }\n\n    async _setBreakpointIfNecessary (command, callsite) {\n        if (!this.disableDebugBreakpoints && this.debugging && canSetDebuggerBreakpointBeforeCommand(command))\n            await this._enqueueSetBreakpointCommand(callsite);\n    }\n\n    async executeAction (actionName, command, callsite) {\n        let error  = null;\n        let result = null;\n\n        await this.emitActionStart(actionName, command);\n\n        try {\n            result = await this.executeCommand(command, callsite);\n        }\n        catch (err) {\n            error = err;\n        }\n\n        await this.emitActionDone(actionName, command, error);\n\n        if (error)\n            throw error;\n\n        return result;\n    }\n\n    async executeCommand (command, callsite) {\n        this.debugLog.command(command);\n\n        if (this.pendingPageError && isCommandRejectableByPageError(command))\n            return this._rejectCommandWithPageError(callsite);\n\n        if (isExecutableOnClientCommand(command))\n            this.addingDriverTasksCount++;\n\n        this._adjustConfigurationWithCommand(command);\n\n        await this._setBreakpointIfNecessary(command, callsite);\n\n        if (isScreenshotCommand(command)) {\n            if (this.opts.disableScreenshots) {\n                this.warningLog.addWarning(WARNING_MESSAGE.screenshotsDisabled);\n\n                return null;\n            }\n\n            await this._adjustScreenshotCommand(command);\n        }\n\n        if (isBrowserManipulationCommand(command)) {\n            this.browserManipulationQueue.push(command);\n\n            if (isResizeWindowCommand(command) && this.opts.videoPath)\n                this.warningLog.addWarning(WARNING_MESSAGE.videoBrowserResizing, this.test.name);\n        }\n\n        if (command.type === COMMAND_TYPE.wait)\n            return delay(command.timeout);\n\n        if (command.type === COMMAND_TYPE.setPageLoadTimeout)\n            return null;\n\n        if (command.type === COMMAND_TYPE.debug)\n            return await this._enqueueSetBreakpointCommand(callsite);\n\n        if (command.type === COMMAND_TYPE.useRole) {\n            let fn = () => this._useRole(command.role, callsite);\n\n            fn = this.decoratePreventEmitActionEvents(fn, { prevent: true });\n            fn = this.decorateDisableDebugBreakpoints(fn, { disable: true });\n\n            return await fn();\n        }\n\n        if (command.type === COMMAND_TYPE.assertion)\n            return this._executeAssertion(command, callsite);\n\n        if (command.type === COMMAND_TYPE.executeExpression)\n            return await this._executeJsExpression(command, callsite);\n\n        if (command.type === COMMAND_TYPE.executeAsyncExpression)\n            return await executeAsyncJsExpression(command.expression, this, callsite);\n\n        if (command.type === COMMAND_TYPE.getBrowserConsoleMessages)\n            return await this._enqueueBrowserConsoleMessagesCommand(command, callsite);\n\n        return this._enqueueCommand(command, callsite);\n    }\n\n    _rejectCommandWithPageError (callsite) {\n        const err = this.pendingPageError;\n\n        err.callsite          = callsite;\n        this.pendingPageError = null;\n\n        return Promise.reject(err);\n    }\n\n    _decorateWithFlag (fn, flagName, value) {\n        return async () => {\n            this[flagName] = value;\n\n            try {\n                return await fn();\n            }\n            catch (err) {\n                throw err;\n            }\n            finally {\n                this[flagName] = !value;\n            }\n        };\n    }\n\n    decoratePreventEmitActionEvents (fn, { prevent }) {\n        return this._decorateWithFlag(fn, 'preventEmitActionEvents', prevent);\n    }\n\n    decorateDisableDebugBreakpoints (fn, { disable }) {\n        return this._decorateWithFlag(fn, 'disableDebugBreakpoints', disable);\n    }\n\n    // Role management\n    async getStateSnapshot () {\n        const state = this.session.getStateSnapshot();\n\n        state.storages = await this.executeCommand(new serviceCommands.BackupStoragesCommand());\n\n        return state;\n    }\n\n    async switchToCleanRun (url) {\n        this.ctx             = Object.create(null);\n        this.fixtureCtx      = Object.create(null);\n        this.consoleMessages = new BrowserConsoleMessages();\n\n        this.session.useStateSnapshot(StateSnapshot.empty());\n\n        if (this.speed !== this.opts.speed) {\n            const setSpeedCommand = new actionCommands.SetTestSpeedCommand({ speed: this.opts.speed });\n\n            await this.executeCommand(setSpeedCommand);\n        }\n\n        if (this.pageLoadTimeout !== this.opts.pageLoadTimeout) {\n            const setPageLoadTimeoutCommand = new actionCommands.SetPageLoadTimeoutCommand({ duration: this.opts.pageLoadTimeout });\n\n            await this.executeCommand(setPageLoadTimeoutCommand);\n        }\n\n        await this.navigateToUrl(url, true);\n\n        if (this.activeDialogHandler) {\n            const removeDialogHandlerCommand = new actionCommands.SetNativeDialogHandlerCommand({ dialogHandler: { fn: null } });\n\n            await this.executeCommand(removeDialogHandlerCommand);\n        }\n    }\n\n    async navigateToUrl (url, forceReload, stateSnapshot) {\n        const navigateCommand = new NavigateToCommand({ url, forceReload, stateSnapshot });\n\n        await this.executeCommand(navigateCommand);\n    }\n\n    async _getStateSnapshotFromRole (role) {\n        const prevPhase = this.phase;\n\n        this.phase = PHASE.inRoleInitializer;\n\n        if (role.phase === ROLE_PHASE.uninitialized)\n            await role.initialize(this);\n\n        else if (role.phase === ROLE_PHASE.pendingInitialization)\n            await promisifyEvent(role, 'initialized');\n\n        if (role.initErr)\n            throw role.initErr;\n\n        this.phase = prevPhase;\n\n        return role.stateSnapshot;\n    }\n\n    async _useRole (role, callsite) {\n        if (this.phase === PHASE.inRoleInitializer)\n            throw new RoleSwitchInRoleInitializerError(callsite);\n\n        const bookmark = new TestRunBookmark(this, role);\n\n        await bookmark.init();\n\n        if (this.currentRoleId)\n            this.usedRoleStates[this.currentRoleId] = await this.getStateSnapshot();\n\n        const stateSnapshot = this.usedRoleStates[role.id] || await this._getStateSnapshotFromRole(role);\n\n        this.session.useStateSnapshot(stateSnapshot);\n\n        this.currentRoleId = role.id;\n\n        await bookmark.restore(callsite, stateSnapshot);\n    }\n\n    // Get current URL\n    async getCurrentUrl () {\n        const builder = new ClientFunctionBuilder(() => {\n            /* eslint-disable no-undef */\n            return window.location.href;\n            /* eslint-enable no-undef */\n        }, { boundTestRun: this });\n\n        const getLocation = builder.getFunction();\n\n        return await getLocation();\n    }\n\n    _disconnect (err) {\n        this.disconnected = true;\n\n        if (this.currentDriverTask)\n            this._rejectCurrentDriverTask(err);\n\n        this.emit('disconnected', err);\n\n        delete testRunTracker.activeTestRuns[this.session.id];\n    }\n\n    async emitActionStart (apiActionName, command) {\n        if (!this.preventEmitActionEvents)\n            await this.emit('action-start', { command, apiActionName });\n    }\n\n    async emitActionDone (apiActionName, command, errors) {\n        if (!this.preventEmitActionEvents)\n            await this.emit('action-done', { command, apiActionName, errors });\n    }\n}\n\n// Service message handlers\nconst ServiceMessages = TestRun.prototype;\n\n// NOTE: this function is time-critical and must return ASAP to avoid client disconnection\nServiceMessages[CLIENT_MESSAGES.ready] = function (msg) {\n    this.debugLog.driverMessage(msg);\n\n    this.emit('connected');\n\n    this._clearPendingRequest();\n\n    // NOTE: the driver sends the status for the second time if it didn't get a response at the\n    // first try. This is possible when the page was unloaded after the driver sent the status.\n    if (msg.status.id === this.lastDriverStatusId)\n        return this.lastDriverStatusResponse;\n\n    this.lastDriverStatusId       = msg.status.id;\n    this.lastDriverStatusResponse = this._handleDriverRequest(msg.status);\n\n    if (this.lastDriverStatusResponse || msg.status.isPendingWindowSwitching)\n        return this.lastDriverStatusResponse;\n\n    // NOTE: we send an empty response after the MAX_RESPONSE_DELAY timeout is exceeded to keep connection\n    // with the client and prevent the response timeout exception on the client side\n    const responseTimeout = setTimeout(() => this._resolvePendingRequest(null), MAX_RESPONSE_DELAY);\n\n    return new Promise((resolve, reject) => {\n        this.pendingRequest = { resolve, reject, responseTimeout };\n    });\n};\n\nServiceMessages[CLIENT_MESSAGES.readyForBrowserManipulation] = async function (msg) {\n    this.debugLog.driverMessage(msg);\n\n    let result = null;\n    let error  = null;\n\n    try {\n        result = await this.browserManipulationQueue.executePendingManipulation(msg);\n    }\n    catch (err) {\n        error = err;\n    }\n\n    return { result, error };\n};\n\nServiceMessages[CLIENT_MESSAGES.waitForFileDownload] = function (msg) {\n    this.debugLog.driverMessage(msg);\n\n    return new Promise(resolve => {\n        if (this.fileDownloadingHandled) {\n            this.fileDownloadingHandled = false;\n            resolve(true);\n        }\n        else\n            this.resolveWaitForFileDownloadingPromise = resolve;\n    });\n};\n"]} |
\ | No newline at end of file |