UNPKG

testcafe

Version:

Automated browser testing for the modern web development stack.

153 lines 19.3 kB
"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 child_process_1 = require("child_process"); const lodash_1 = require("lodash"); const async_event_emitter_1 = __importDefault(require("../utils/async-event-emitter")); const delay_1 = __importDefault(require("../utils/delay")); const DEBUG_LOGGER_PREFIX = 'testcafe:video-recorder:process:'; const DEFAULT_OPTIONS = { // NOTE: use to force stdin and stdout formats 'f': 'image2pipe', // NOTE: don't ask confirmation for rewriting the output file 'y': true, // NOTE: use the time when a frame is read from the source as its timestamp // IMPORTANT: must be specified before configuring the source 'use_wallclock_as_timestamps': 1, // NOTE: use stdin as a source 'i': 'pipe:0', // NOTE: use the H.264 video codec 'c:v': 'libx264', // NOTE: use the 'ultrafast' compression preset 'preset': 'ultrafast', // NOTE: use the yuv420p pixel format (the most widely supported) 'pix_fmt': 'yuv420p', // NOTE: scale input frames to make the frame height divisible by 2 (yuv420p's requirement) 'vf': 'scale=trunc(iw/2)*2:trunc(ih/2)*2', // NOTE: set the frame rate to 30 in the output video (the most widely supported) 'r': 30, }; const FFMPEG_START_DELAY = 500; const DELAY_AFTER_EMPTY_FRAME = 20; class VideoRecorder extends async_event_emitter_1.default { constructor(basePath, ffmpegPath, connection, customOptions) { super(); this.debugLogger = (0, debug_1.default)(DEBUG_LOGGER_PREFIX + connection.id); this.customOptions = customOptions; this.videoPath = basePath; this.connection = connection; this.ffmpegPath = ffmpegPath; this.ffmpegProcess = null; this.ffmpegStdoutBuf = ''; this.ffmpegStderrBuf = ''; this.ffmpegClosingPromise = null; this.disposed = false; this.closed = false; this.optionsList = this._getOptionsList(); this.capturingPromise = null; } static _filterOption([key, value]) { if (value === true) return ['-' + key]; return ['-' + key, value]; } _setupFFMPEGBuffers() { this.ffmpegProcess.stdout.on('data', data => { this.ffmpegStdoutBuf += String(data); }); this.ffmpegProcess.stderr.on('data', data => { this.ffmpegStderrBuf += String(data); }); } _getChildProcessPromise() { return new Promise((resolve, reject) => { this.ffmpegProcess.on('exit', resolve); this.ffmpegProcess.on('error', reject); }); } _getOptionsList() { const optionsObject = Object.assign({}, DEFAULT_OPTIONS, this.customOptions); const optionsList = (0, lodash_1.flatten)(Object.entries(optionsObject).map(VideoRecorder._filterOption)); optionsList.push(this.videoPath); return optionsList; } get active() { return !this.closed && !this.disposed; } async _addFrame(frameData) { const writingFinished = this.ffmpegProcess.stdin.write(frameData); if (!writingFinished) await new Promise(r => this.ffmpegProcess.stdin.once('drain', r)); } async _capture() { while (this.active) { try { const frame = await this.connection.provider.getVideoFrameData(this.connection.id); if (frame) { await this.emit('frame'); await this._addFrame(frame); } else await (0, delay_1.default)(DELAY_AFTER_EMPTY_FRAME); } catch (error) { this.debugLogger(error); } } } async _startCapturing() { await this.connection.provider.startCapturingVideo(this.connection.id); } async _stopCapturing() { await this.connection.provider.stopCapturingVideo(this.connection.id); } async init() { this.ffmpegProcess = (0, child_process_1.spawn)(this.ffmpegPath, this.optionsList, { stdio: 'pipe' }); this._setupFFMPEGBuffers(); this.ffmpegClosingPromise = this ._getChildProcessPromise() .then(code => { this.closed = true; this.disposed = true; if (code) { this.debugLogger(code); this.debugLogger(this.ffmpegStdoutBuf); this.debugLogger(this.ffmpegStderrBuf); } }) .catch(error => { this.closed = true; this.disposed = true; this.debugLogger(error); this.debugLogger(this.ffmpegStdoutBuf); this.debugLogger(this.ffmpegStderrBuf); }); await (0, delay_1.default)(FFMPEG_START_DELAY); } async dispose() { if (this.disposed) return; this.disposed = true; this.ffmpegProcess.stdin.end(); await this.ffmpegClosingPromise; } async startCapturing() { await this._startCapturing(); this.capturingPromise = this._capture(); await this.once('frame'); } async finishCapturing() { if (this.closed) return; this.closed = true; await this._stopCapturing(); await this.capturingPromise; await this.dispose(); } } exports.default = VideoRecorder; module.exports = exports.default; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"process.js","sourceRoot":"","sources":["../../src/video-recorder/process.js"],"names":[],"mappings":"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","sourcesContent":["import debug from 'debug';\nimport { spawn } from 'child_process';\nimport { flatten } from 'lodash';\nimport AsyncEmitter from '../utils/async-event-emitter';\nimport delay from '../utils/delay';\n\n\nconst DEBUG_LOGGER_PREFIX = 'testcafe:video-recorder:process:';\n\nconst DEFAULT_OPTIONS = {\n\n    // NOTE: use to force stdin and stdout formats\n    'f': 'image2pipe',\n\n    // NOTE: don't ask confirmation for rewriting the output file\n    'y': true,\n\n    // NOTE: use the time when a frame is read from the source as its timestamp\n    // IMPORTANT: must be specified before configuring the source\n    'use_wallclock_as_timestamps': 1,\n\n    // NOTE: use stdin as a source\n    'i': 'pipe:0',\n\n    // NOTE: use the H.264 video codec\n    'c:v': 'libx264',\n\n    // NOTE: use the 'ultrafast' compression preset\n    'preset': 'ultrafast',\n\n    // NOTE: use the yuv420p pixel format (the most widely supported)\n    'pix_fmt': 'yuv420p',\n\n    // NOTE: scale input frames to make the frame height divisible by 2 (yuv420p's requirement)\n    'vf': 'scale=trunc(iw/2)*2:trunc(ih/2)*2',\n\n    // NOTE: set the frame rate to 30 in the output video (the most widely supported)\n    'r': 30,\n};\n\nconst FFMPEG_START_DELAY = 500;\n\nconst DELAY_AFTER_EMPTY_FRAME = 20;\n\nexport default class VideoRecorder extends AsyncEmitter {\n    constructor (basePath, ffmpegPath, connection, customOptions) {\n        super();\n\n        this.debugLogger = debug(DEBUG_LOGGER_PREFIX + connection.id);\n\n        this.customOptions = customOptions;\n        this.videoPath     = basePath;\n        this.connection    = connection;\n        this.ffmpegPath    = ffmpegPath;\n        this.ffmpegProcess = null;\n\n        this.ffmpegStdoutBuf = '';\n        this.ffmpegStderrBuf = '';\n\n        this.ffmpegClosingPromise = null;\n\n        this.disposed = false;\n        this.closed = false;\n\n        this.optionsList = this._getOptionsList();\n\n        this.capturingPromise = null;\n    }\n\n    static _filterOption ([key, value]) {\n        if (value === true)\n            return ['-' + key];\n\n        return ['-' + key, value];\n    }\n\n    _setupFFMPEGBuffers () {\n        this.ffmpegProcess.stdout.on('data', data => {\n            this.ffmpegStdoutBuf += String(data);\n        });\n\n        this.ffmpegProcess.stderr.on('data', data => {\n            this.ffmpegStderrBuf += String(data);\n        });\n    }\n\n    _getChildProcessPromise () {\n        return new Promise((resolve, reject) => {\n            this.ffmpegProcess.on('exit', resolve);\n            this.ffmpegProcess.on('error', reject);\n        });\n    }\n\n    _getOptionsList () {\n        const optionsObject = Object.assign({}, DEFAULT_OPTIONS, this.customOptions);\n\n        const optionsList = flatten(Object.entries(optionsObject).map(VideoRecorder._filterOption));\n\n        optionsList.push(this.videoPath);\n\n        return optionsList;\n    }\n\n    get active () {\n        return !this.closed && !this.disposed;\n    }\n\n    async _addFrame (frameData) {\n        const writingFinished = this.ffmpegProcess.stdin.write(frameData);\n\n        if (!writingFinished)\n            await new Promise(r => this.ffmpegProcess.stdin.once('drain', r));\n    }\n\n    async _capture () {\n        while (this.active) {\n            try {\n                const frame = await this.connection.provider.getVideoFrameData(this.connection.id);\n\n                if (frame) {\n                    await this.emit('frame');\n                    await this._addFrame(frame);\n                }\n                else\n                    await delay(DELAY_AFTER_EMPTY_FRAME);\n            }\n            catch (error) {\n                this.debugLogger(error);\n            }\n        }\n    }\n\n    async _startCapturing () {\n        await this.connection.provider.startCapturingVideo(this.connection.id);\n    }\n\n    async _stopCapturing () {\n        await this.connection.provider.stopCapturingVideo(this.connection.id);\n    }\n\n    async init () {\n        this.ffmpegProcess = spawn(this.ffmpegPath, this.optionsList, { stdio: 'pipe' });\n\n        this._setupFFMPEGBuffers();\n\n        this.ffmpegClosingPromise = this\n            ._getChildProcessPromise()\n            .then(code => {\n                this.closed = true;\n                this.disposed = true;\n\n                if (code) {\n                    this.debugLogger(code);\n                    this.debugLogger(this.ffmpegStdoutBuf);\n                    this.debugLogger(this.ffmpegStderrBuf);\n                }\n            })\n            .catch(error => {\n                this.closed = true;\n                this.disposed = true;\n\n                this.debugLogger(error);\n                this.debugLogger(this.ffmpegStdoutBuf);\n                this.debugLogger(this.ffmpegStderrBuf);\n            });\n\n        await delay(FFMPEG_START_DELAY);\n    }\n\n    async dispose () {\n        if (this.disposed)\n            return;\n\n        this.disposed = true;\n        this.ffmpegProcess.stdin.end();\n\n        await this.ffmpegClosingPromise;\n    }\n\n    async startCapturing () {\n        await this._startCapturing();\n\n        this.capturingPromise = this._capture();\n\n        await this.once('frame');\n    }\n\n    async finishCapturing () {\n        if (this.closed)\n            return;\n\n        this.closed = true;\n\n        await this._stopCapturing();\n        await this.capturingPromise;\n        await this.dispose();\n    }\n}\n"]}