testcafe
Version:
Automated browser testing for the modern web development stack.
151 lines • 23.8 kB
JavaScript
"use strict";
var __importDefault = (this && this.__importDefault) || function (mod) {
return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
const debug_1 = __importDefault(require("debug"));
const path_1 = require("path");
const fs_1 = __importDefault(require("fs"));
const child_process_1 = require("child_process");
const make_dir_1 = __importDefault(require("make-dir"));
const temp_directory_1 = __importDefault(require("../utils/temp-directory"));
const path_pattern_1 = __importDefault(require("../utils/path-pattern"));
const warning_message_1 = __importDefault(require("../notifications/warning-message"));
const string_1 = require("../utils/string");
const test_run_video_recorder_1 = __importDefault(require("./test-run-video-recorder"));
const events_1 = require("events");
const create_safe_listener_1 = __importDefault(require("../utils/create-safe-listener"));
const DEBUG_LOGGER = (0, debug_1.default)('testcafe:video-recorder');
const VIDEO_EXTENSION = 'mp4';
const TEMP_DIR_PREFIX = 'video';
class VideoRecorder extends events_1.EventEmitter {
constructor(browserJob, basePath, opts, encodingOpts, warningLog) {
super();
this.browserJob = browserJob;
this.basePath = basePath;
this.failedOnly = opts.failedOnly;
this.singleFile = opts.singleFile;
this.ffmpegPath = opts.ffmpegPath;
this.customPathPattern = opts.pathPattern;
this.timeStamp = opts.timeStamp;
this.encodingOptions = encodingOpts;
this.warningLog = warningLog;
this.tempDirectory = new temp_directory_1.default(TEMP_DIR_PREFIX);
this.firstFile = true;
this.testRunVideoRecorders = {};
this._assignEventHandlers(browserJob);
}
_createSafeListener(listener) {
return (0, create_safe_listener_1.default)(this, listener, DEBUG_LOGGER);
}
_assignEventHandlers(browserJob) {
browserJob.once('start', this._createSafeListener(() => {
this.tempDirectoryInitializedPromise = this._onBrowserJobStart();
return this.tempDirectoryInitializedPromise;
}));
browserJob.once('done', this._createSafeListener(this._onBrowserJobDone));
browserJob.on('test-run-create', this._createSafeListener(this._onTestRunCreate));
browserJob.on('test-run-ready', this._createSafeListener(this._onTestRunReady));
browserJob.on('test-run-before-done', this._createSafeListener(this._onTestRunBeforeDone));
browserJob.on('test-run-restart', this._createSafeListener(this._onTestRunRestart));
}
_addProblematicPlaceholdersWarning(placeholders) {
const problematicPlaceholderListStr = (0, string_1.getConcatenatedValuesString)(placeholders);
const suffix = (0, string_1.getPluralSuffix)(placeholders);
const verb = (0, string_1.getToBeInPastTense)(placeholders);
this.warningLog.addWarning(warning_message_1.default.problematicPathPatternPlaceholderForVideoRecording, suffix, problematicPlaceholderListStr, suffix, verb);
}
_getTargetVideoPath(testRunRecorder) {
const data = Object.assign(testRunRecorder.testRunInfo, { now: this.timeStamp });
if (this.singleFile) {
data.testIndex = null;
data.fixture = null;
data.test = null;
}
const pathPattern = new path_pattern_1.default(this.customPathPattern, VIDEO_EXTENSION, data);
pathPattern.on('problematic-placeholders-found', ({ placeholders }) => this._addProblematicPlaceholdersWarning(placeholders));
return (0, path_1.join)(this.basePath, pathPattern.getPath());
}
_concatVideo(targetVideoPath, { tempVideoPath, tempMergeConfigPath, tmpMergeName }) {
if (this.firstFile) {
this.firstFile = false;
return;
}
fs_1.default.writeFileSync(tempMergeConfigPath, `
file '${targetVideoPath}'
file '${tempVideoPath}'
`);
(0, child_process_1.spawnSync)(this.ffmpegPath, ['-y', '-f', 'concat', '-safe', '0', '-i', tempMergeConfigPath, '-c', 'copy', tmpMergeName], { stdio: 'ignore' });
fs_1.default.copyFileSync(tmpMergeName, tempVideoPath);
}
async _onBrowserJobStart() {
await this.tempDirectory.init();
}
async _onBrowserJobDone() {
await this.tempDirectory.dispose();
}
async _onTestRunCreate(testRunInfo) {
if (testRunInfo.legacy)
return;
await this.tempDirectoryInitializedPromise;
const recordingOptions = {
path: this.tempDirectory.path,
ffmpegPath: this.ffmpegPath,
encodingOptions: this.encodingOptions,
};
const testRunVideoRecorder = this._createTestRunVideoRecorder(testRunInfo, recordingOptions);
const isVideoSupported = await testRunVideoRecorder.isVideoSupported();
if (!isVideoSupported) {
this.warningLog.addWarning(warning_message_1.default.videoNotSupportedByBrowser, testRunVideoRecorder.testRunInfo.alias);
return;
}
const isVideoEnabled = await testRunVideoRecorder.isVideoEnabled();
if (!isVideoEnabled)
return;
await testRunVideoRecorder.init();
this.testRunVideoRecorders[testRunVideoRecorder.index] = testRunVideoRecorder;
}
_createTestRunVideoRecorder(testRunInfo, recordingOptions) {
return new test_run_video_recorder_1.default(testRunInfo, recordingOptions, this.warningLog);
}
async _onTestRunReady({ index }) {
const testRunRecorder = this.testRunVideoRecorders[index];
if (!testRunRecorder)
return;
await testRunRecorder.startCapturing();
}
async _onTestRunRestart({ index }) {
const testRunRecorder = this.testRunVideoRecorders[index];
if (!testRunRecorder)
return;
testRunRecorder.onTestRunRestart();
}
async _onTestRunBeforeDone({ index }) {
const testRunRecorder = this.testRunVideoRecorders[index];
if (!testRunRecorder)
return;
delete this.testRunVideoRecorders[index];
await testRunRecorder.finishCapturing();
if (this.failedOnly && !testRunRecorder.hasErrors)
return;
const videoPath = this._getTargetVideoPath(testRunRecorder);
await this._saveFiles(testRunRecorder, videoPath);
const testRunVideoSavedEventArgs = {
testRun: testRunRecorder.testRun,
videoPath,
singleFile: !!this.singleFile,
};
if (!this.singleFile)
testRunVideoSavedEventArgs.timecodes = testRunRecorder.testRunInfo.timecodes;
this.emit('test-run-video-saved', testRunVideoSavedEventArgs);
}
async _saveFiles(testRunRecorder, videoPath) {
await (0, make_dir_1.default)((0, path_1.dirname)(videoPath));
if (this.singleFile)
this._concatVideo(videoPath, testRunRecorder.tempFiles);
fs_1.default.copyFileSync(testRunRecorder.tempFiles.tempVideoPath, videoPath);
}
}
exports.default = VideoRecorder;
module.exports = exports.default;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"recorder.js","sourceRoot":"","sources":["../../src/video-recorder/recorder.js"],"names":[],"mappings":"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","sourcesContent":["import debug from 'debug';\nimport { join, dirname } from 'path';\nimport fs from 'fs';\nimport { spawnSync } from 'child_process';\nimport makeDir from 'make-dir';\nimport TempDirectory from '../utils/temp-directory';\nimport PathPattern from '../utils/path-pattern';\nimport WARNING_MESSAGES from '../notifications/warning-message';\nimport {\n    getPluralSuffix,\n    getConcatenatedValuesString,\n    getToBeInPastTense,\n} from '../utils/string';\n\nimport TestRunVideoRecorder from './test-run-video-recorder';\nimport { EventEmitter } from 'events';\nimport createSafeListener from '../utils/create-safe-listener';\n\nconst DEBUG_LOGGER = debug('testcafe:video-recorder');\n\nconst VIDEO_EXTENSION = 'mp4';\nconst TEMP_DIR_PREFIX = 'video';\n\nexport default class VideoRecorder extends EventEmitter {\n    constructor (browserJob, basePath, opts, encodingOpts, warningLog) {\n        super();\n\n        this.browserJob        = browserJob;\n        this.basePath          = basePath;\n        this.failedOnly        = opts.failedOnly;\n        this.singleFile        = opts.singleFile;\n        this.ffmpegPath        = opts.ffmpegPath;\n        this.customPathPattern = opts.pathPattern;\n        this.timeStamp         = opts.timeStamp;\n        this.encodingOptions   = encodingOpts;\n\n        this.warningLog = warningLog;\n\n        this.tempDirectory = new TempDirectory(TEMP_DIR_PREFIX);\n\n        this.firstFile = true;\n\n        this.testRunVideoRecorders = {};\n\n        this._assignEventHandlers(browserJob);\n    }\n\n    _createSafeListener (listener) {\n        return createSafeListener(this, listener, DEBUG_LOGGER);\n    }\n\n    _assignEventHandlers (browserJob) {\n        browserJob.once('start', this._createSafeListener(() => {\n            this.tempDirectoryInitializedPromise = this._onBrowserJobStart();\n\n            return this.tempDirectoryInitializedPromise;\n        }));\n\n        browserJob.once('done', this._createSafeListener(this._onBrowserJobDone));\n        browserJob.on('test-run-create', this._createSafeListener(this._onTestRunCreate));\n        browserJob.on('test-run-ready', this._createSafeListener(this._onTestRunReady));\n        browserJob.on('test-run-before-done', this._createSafeListener(this._onTestRunBeforeDone));\n        browserJob.on('test-run-restart', this._createSafeListener(this._onTestRunRestart));\n    }\n\n    _addProblematicPlaceholdersWarning (placeholders) {\n        const problematicPlaceholderListStr = getConcatenatedValuesString(placeholders);\n        const suffix                        = getPluralSuffix(placeholders);\n        const verb                          = getToBeInPastTense(placeholders);\n\n        this.warningLog.addWarning(WARNING_MESSAGES.problematicPathPatternPlaceholderForVideoRecording, suffix, problematicPlaceholderListStr, suffix, verb);\n    }\n\n    _getTargetVideoPath (testRunRecorder) {\n        const data = Object.assign(testRunRecorder.testRunInfo, { now: this.timeStamp });\n\n        if (this.singleFile) {\n            data.testIndex = null;\n            data.fixture = null;\n            data.test = null;\n        }\n\n        const pathPattern = new PathPattern(this.customPathPattern, VIDEO_EXTENSION, data);\n\n        pathPattern.on('problematic-placeholders-found', ({ placeholders }) => this._addProblematicPlaceholdersWarning(placeholders));\n\n        return join(this.basePath, pathPattern.getPath());\n    }\n\n    _concatVideo (targetVideoPath, { tempVideoPath, tempMergeConfigPath, tmpMergeName }) {\n        if (this.firstFile) {\n            this.firstFile = false;\n            return;\n        }\n\n        fs.writeFileSync(tempMergeConfigPath, `\n            file '${targetVideoPath}'\n            file '${tempVideoPath}'\n        `);\n\n        spawnSync(this.ffmpegPath, ['-y', '-f', 'concat', '-safe', '0', '-i', tempMergeConfigPath, '-c', 'copy', tmpMergeName], { stdio: 'ignore' });\n        fs.copyFileSync(tmpMergeName, tempVideoPath);\n    }\n\n    async _onBrowserJobStart () {\n        await this.tempDirectory.init();\n    }\n\n    async _onBrowserJobDone () {\n        await this.tempDirectory.dispose();\n    }\n\n    async _onTestRunCreate (testRunInfo) {\n        if (testRunInfo.legacy)\n            return;\n\n        await this.tempDirectoryInitializedPromise;\n\n        const recordingOptions = {\n            path:            this.tempDirectory.path,\n            ffmpegPath:      this.ffmpegPath,\n            encodingOptions: this.encodingOptions,\n        };\n\n        const testRunVideoRecorder = this._createTestRunVideoRecorder(testRunInfo, recordingOptions);\n        const isVideoSupported     = await testRunVideoRecorder.isVideoSupported();\n\n        if (!isVideoSupported) {\n            this.warningLog.addWarning(WARNING_MESSAGES.videoNotSupportedByBrowser, testRunVideoRecorder.testRunInfo.alias);\n            return;\n        }\n\n        const isVideoEnabled = await testRunVideoRecorder.isVideoEnabled();\n\n        if (!isVideoEnabled)\n            return;\n\n        await testRunVideoRecorder.init();\n\n        this.testRunVideoRecorders[testRunVideoRecorder.index] = testRunVideoRecorder;\n    }\n\n    _createTestRunVideoRecorder (testRunInfo, recordingOptions) {\n        return new TestRunVideoRecorder(testRunInfo, recordingOptions, this.warningLog);\n    }\n\n    async _onTestRunReady ({ index }) {\n        const testRunRecorder = this.testRunVideoRecorders[index];\n\n        if (!testRunRecorder)\n            return;\n\n        await testRunRecorder.startCapturing();\n    }\n\n    async _onTestRunRestart ({ index }) {\n        const testRunRecorder = this.testRunVideoRecorders[index];\n\n        if (!testRunRecorder)\n            return;\n\n        testRunRecorder.onTestRunRestart();\n    }\n\n    async _onTestRunBeforeDone ({ index }) {\n        const testRunRecorder = this.testRunVideoRecorders[index];\n\n        if (!testRunRecorder)\n            return;\n\n        delete this.testRunVideoRecorders[index];\n\n        await testRunRecorder.finishCapturing();\n\n        if (this.failedOnly && !testRunRecorder.hasErrors)\n            return;\n\n        const videoPath = this._getTargetVideoPath(testRunRecorder);\n\n        await this._saveFiles(testRunRecorder, videoPath);\n\n        const testRunVideoSavedEventArgs = {\n            testRun:    testRunRecorder.testRun,\n            videoPath,\n            singleFile: !!this.singleFile,\n        };\n\n        if (!this.singleFile)\n            testRunVideoSavedEventArgs.timecodes = testRunRecorder.testRunInfo.timecodes;\n\n        this.emit('test-run-video-saved', testRunVideoSavedEventArgs);\n    }\n\n    async _saveFiles (testRunRecorder, videoPath) {\n        await makeDir(dirname(videoPath));\n\n        if (this.singleFile)\n            this._concatVideo(videoPath, testRunRecorder.tempFiles);\n\n        fs.copyFileSync(testRunRecorder.tempFiles.tempVideoPath, videoPath);\n    }\n}\n"]}