Huge News!Announcing our $40M Series B led by Abstract Ventures.Learn More
Socket
Sign inDemoInstall
Socket

testcafe-browser-tools

Package Overview
Dependencies
Maintainers
7
Versions
88
Alerts
File Explorer

Advanced tools

Socket logo

Install Socket

Detect and block malicious and high-risk dependencies

Install

testcafe-browser-tools - npm Package Compare versions

Comparing version 2.0.1 to 2.0.2

lib/utils/flatten-whitespace.js

80

lib/api/get-installations.js

@@ -6,25 +6,37 @@ "use strict";

Object.defineProperty(exports, "__esModule", { value: true });
const pinkie_1 = __importDefault(require("pinkie"));
const os_family_1 = __importDefault(require("os-family"));
const which_promise_1 = __importDefault(require("which-promise"));
const lodash_1 = require("lodash");
const fs_exists_promised_1 = __importDefault(require("../utils/fs-exists-promised"));
const exec_1 = require("../utils/exec");
const unquote_1 = __importDefault(require("../utils/unquote"));
const aliases_1 = __importDefault(require("../aliases"));
const MICROSOFT_EDGE_CLASS = 'Microsoft.MicrosoftEdge';
const MICROSOFT_EDGE_KEY_GLOB = `HKCU\\Software\\Classes\\ActivatableClasses\\Package\\${MICROSOFT_EDGE_CLASS}*`;
const BROWSER_COMMANDS_KEY_GLOB = root => `${root}\\Software\\Clients\\StartMenuInternet\\*\\shell\\open\\command`;
const LINE_WRAP = '\r\n';
const DOUBLE_LINE_WRAP = LINE_WRAP + LINE_WRAP;
const REGISTRY_BROWSER_PATH_PROPERTY = '(default)';
const REGISTRY_BROWSER_NAME_PROPERTY = 'PSPath';
const REGISTRY_PROPERTIES_RE = /^(\(default\)|PSPath)\s*:\s*(.*)$/;
const MAX_OUTPUT_WIDTH = 2 ** 31 - 1;
const POWERSHELL_PIPE_SYMBOL = '|';
const GET_REGISTRY_KEY_COMMAND = key => `Get-Item 'Registry::${key}'`;
const GET_BROWSER_PATH_PROPERTY_COMMAND = `Get-ItemProperty -Name '${REGISTRY_BROWSER_PATH_PROPERTY}'`;
const FORMAT_BROWSER_INFO_COMMAND = `Format-List -Property '${REGISTRY_BROWSER_PATH_PROPERTY}','${REGISTRY_BROWSER_NAME_PROPERTY}'`;
const LIMIT_OUTPUT_WIDTH_COMMAND = `Out-String -Width ${MAX_OUTPUT_WIDTH}`;
const GET_REGISTRY_BROWSER_PROPERTIES_COMMAND = key => [
GET_REGISTRY_KEY_COMMAND(key),
GET_BROWSER_PATH_PROPERTY_COMMAND,
FORMAT_BROWSER_INFO_COMMAND,
LIMIT_OUTPUT_WIDTH_COMMAND
].join(POWERSHELL_PIPE_SYMBOL);
// Installation info cache
var installationsCache = null;
async function getRegistrySubTree(regKey) {
let script = `$cp = (chcp | Select-String '\\d+').Matches.Value;
Try
{
chcp 65001;
Get-ChildItem -Path Registry::${regKey} -Recurse;
}
Finally
{
chcp $cp;
}`;
script = script.replace(/\s+/g, ' ');
const command = `powershell.exe -NoLogo -NonInteractive -Command "${script}"`;
return exec_1.exec(command);
async function getRegistryKey(regKeyGlob) {
return exec_1.execPowershell(GET_REGISTRY_KEY_COMMAND(regKeyGlob));
}
async function getRegistryBrowserProperties(regKeyGlob) {
return exec_1.execPowershell(GET_REGISTRY_BROWSER_PROPERTIES_COMMAND(regKeyGlob));
}
// Find installations for different platforms

@@ -45,18 +57,22 @@ async function addInstallation(installations, name, instPath) {

async function detectMicrosoftEdge() {
const regKey = 'HKCU\\Software\\Classes\\ActivatableClasses';
const edgeRe = /^Microsoft\.MicrosoftEdge/m;
const subTree = await getRegistrySubTree(regKey);
return edgeRe.test(subTree) ? aliases_1.default['edge'] : null;
const registryResult = await getRegistryKey(MICROSOFT_EDGE_KEY_GLOB);
if (registryResult.stdout.includes(MICROSOFT_EDGE_CLASS))
return aliases_1.default['edge'];
return null;
}
async function searchInRegistry(registryRoot) {
const installations = {};
const text = await getRegistrySubTree(registryRoot + '\\SOFTWARE\\Clients\\StartMenuInternet');
const re = /\\SOFTWARE\\Clients\\StartMenuInternet\\([^\r\n\\]+)\\shell\\open\s+Name\s+Property[-\s]+command\s+\(default\)\s*:\s*(.+)$/gmi;
let match = re.exec(text);
while (match) {
const name = match[1].replace(/\.exe$/i, '');
const path = match[2].trim().replace(/^"(.*)"$/, '$1').replace(/\\$/, '');
await addInstallation(installations, name, path);
match = re.exec(text);
}
const registryResult = await getRegistryBrowserProperties(BROWSER_COMMANDS_KEY_GLOB(registryRoot));
const data = registryResult.stdout
.split(DOUBLE_LINE_WRAP)
.map(block => block
.split(LINE_WRAP)
.map(line => line.match(REGISTRY_PROPERTIES_RE))
.filter(match => !!match)
.map(match => ({
[match[1]]: unquote_1.default(match[2])
})))
.filter(block => block && block.length)
.map(block => lodash_1.merge(...block));
await Promise.all(data.map(record => addInstallation(installations, record[REGISTRY_BROWSER_NAME_PROPERTY], record[REGISTRY_BROWSER_PATH_PROPERTY])));
return installations;

@@ -77,3 +93,3 @@ }

var stdout = await exec_1.exec('ls "/Applications/" | grep -E "Chrome|Firefox|Opera|Safari|Chromium" | sed -E "s/ /032/"');
await pinkie_1.default.all(stdout
await Promise.all(stdout
.split('\n')

@@ -107,5 +123,5 @@ .filter(fileName => !!fileName)

});
return pinkie_1.default.all(detectionPromises);
return Promise.all(detectionPromises);
});
await pinkie_1.default.all(aliasCheckingPromises);
await Promise.all(aliasCheckingPromises);
return installations;

@@ -167,2 +183,2 @@ }

exports.default = default_1;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"get-installations.js","sourceRoot":"","sources":["../../src/api/get-installations.js"],"names":[],"mappings":"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","sourcesContent":["import Promise from 'pinkie';\nimport OS from 'os-family';\nimport which from 'which-promise';\nimport exists from '../utils/fs-exists-promised';\nimport { exec } from '../utils/exec';\nimport ALIASES from '../aliases';\n\n\n// Installation info cache\nvar installationsCache = null;\n\nasync function getRegistrySubTree (regKey) {\n    let script =\n        `$cp = (chcp | Select-String '\\\\d+').Matches.Value;\n        Try\n        {\n            chcp 65001;\n            Get-ChildItem -Path Registry::${regKey} -Recurse;\n        }\n        Finally\n        {\n            chcp $cp;\n        }`;\n\n    script = script.replace(/\\s+/g, ' ');\n\n    const command = `powershell.exe -NoLogo -NonInteractive -Command \"${script}\"`;\n\n    return exec(command);\n}\n\n// Find installations for different platforms\nasync function addInstallation (installations, name, instPath) {\n    var fileExists = await exists(instPath);\n\n    if (fileExists) {\n        Object.keys(ALIASES).some(alias => {\n            var { nameRe, cmd, macOpenCmdTemplate, path } = ALIASES[alias];\n\n            if (nameRe.test(name)) {\n                installations[alias] = { path: path || instPath, cmd, macOpenCmdTemplate };\n                return true;\n            }\n\n            return false;\n        });\n    }\n}\n\nasync function detectMicrosoftEdge () {\n    const regKey  = 'HKCU\\\\Software\\\\Classes\\\\ActivatableClasses';\n    const edgeRe  = /^Microsoft\\.MicrosoftEdge/m;\n    const subTree = await getRegistrySubTree(regKey);\n\n    return edgeRe.test(subTree) ? ALIASES['edge'] : null;\n}\n\nasync function searchInRegistry (registryRoot) {\n    const installations = {};\n    const text          = await getRegistrySubTree(registryRoot + '\\\\SOFTWARE\\\\Clients\\\\StartMenuInternet');\n    const re            = /\\\\SOFTWARE\\\\Clients\\\\StartMenuInternet\\\\([^\\r\\n\\\\]+)\\\\shell\\\\open\\s+Name\\s+Property[-\\s]+command\\s+\\(default\\)\\s*:\\s*(.+)$/gmi;\n\n    let match = re.exec(text);\n\n    while (match) {\n        const name = match[1].replace(/\\.exe$/i, '');\n        const path = match[2].trim().replace(/^\"(.*)\"$/, '$1').replace(/\\\\$/, '');\n\n        await addInstallation(installations, name, path);\n\n        match = re.exec(text);\n    }\n\n    return installations;\n}\n\nasync function findWindowsBrowsers () {\n    var machineRegisteredBrowsers = await searchInRegistry('HKEY_LOCAL_MACHINE');\n    var userRegisteredBrowsers    = await searchInRegistry('HKEY_CURRENT_USER');\n    var installations             = Object.assign(machineRegisteredBrowsers, userRegisteredBrowsers);\n    var edgeAlias                 = await detectMicrosoftEdge();\n\n    if (edgeAlias)\n        installations['edge'] = edgeAlias;\n\n    return installations;\n}\n\nasync function findMacBrowsers () {\n    var installations = {};\n\n    // NOTE: replace the space symbol with code, because grep splits strings by space.\n    var stdout = await exec('ls \"/Applications/\" | grep -E \"Chrome|Firefox|Opera|Safari|Chromium\" | sed -E \"s/ /032/\"');\n\n    await Promise.all(stdout\n        .split('\\n')\n        .filter(fileName => !!fileName)\n        .map(fileName => {\n            // NOTE: restore space\n            fileName = fileName.replace(/032/g, ' ');\n\n            var name = fileName.replace(/.app$/, '');\n            var path = `/Applications/${fileName}`;\n\n            return addInstallation(installations, name, path);\n        }));\n\n    return installations;\n}\n\nasync function findLinuxBrowsers () {\n    var installations = {};\n\n    var aliasCheckingPromises = Object\n        .keys(ALIASES)\n        .map(name => {\n            var { linuxBinaries } = ALIASES[name];\n\n            if (!linuxBinaries)\n                return null;\n\n            var detectionPromises = linuxBinaries\n                .map(binary => {\n                    return which(binary)\n                        .then(path => addInstallation(installations, name, path))\n                        .catch(() => {\n                            // NOTE: binary not found, just do nothing\n                            return;\n                        });\n                });\n\n            return Promise.all(detectionPromises);\n        });\n\n    await Promise.all(aliasCheckingPromises);\n\n    return installations;\n}\n\nasync function findBrowsers () {\n    if (OS.win)\n        return await findWindowsBrowsers();\n\n    if (OS.mac)\n        return await findMacBrowsers();\n\n    if (OS.linux)\n        return await findLinuxBrowsers();\n}\n\n\n// API\n/** @typedef {Object} BrowserInfo\n * @description Object that contains information about the browser installed on the machine.\n * @property {string|undefined} path - The path to the executable file that starts the browser.\n *  Required on MacOS machines. On Windows machines, it is used when the winOpenCmdTemplate property is undefined.\n * @property {string} cmd - Additional command line parameters.\n * @property {string} macOpenCmdTemplate - A [Mustache template](https://github.com/janl/mustache.js#templates)\n *  that provides parameters for launching the browser on a MacOS machine.\n * @property {string|undefined} winOpenCmdTemplate - A [Mustache template](https://github.com/janl/mustache.js#templates)\n *  that provides parameters for launching the browser on a Windows machine.  If undefined, the path to the\n *  executable file specified by the path property is used.\n * @example\n *  {\n *       path: 'C:\\\\ProgramFiles\\\\...\\\\firefox.exe',\n *       cmd: '-new-window',\n *       macOpenCmdTemplate: 'open -a \"{{{path}}}\" {{{pageUrl}}} --args {{{cmd}}}'\n *  }\n */\n\n/**\n * Returns the list of the {@link BrowserInfo} objects that contain information about the browsers installed on the machine.\n * @function\n * @async\n * @name getInstallations\n * @returns {Object.<string, BrowserInfo>} List of the {@link BrowserInfo} objects\n *   containing information about the browsers installed on the machine.\n * @example\n * {\n *   chrome: {\n *       path: 'C:\\\\ProgramFiles\\\\...\\\\chrome.exe',\n *       cmd: '--new-window',\n *       macOpenCmdTemplate: 'open -n -a \"{{{path}}}\" --args {{{pageUrl}}} {{{cmd}}}'\n *   },\n *\n *   firefox: {\n *       path: 'C:\\\\ProgramFiles\\\\...\\\\firefox.exe',\n *       cmd: '-new-window',\n *       macOpenCmdTemplate: 'open -a \"{{{path}}}\" {{{pageUrl}}} --args {{{cmd}}}'\n *   }\n * }\n */\nexport default async function () {\n    if (!installationsCache)\n        installationsCache = await findBrowsers();\n\n    return installationsCache;\n}\n"]}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"get-installations.js","sourceRoot":"","sources":["../../src/api/get-installations.js"],"names":[],"mappings":"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","sourcesContent":["import OS from 'os-family';\nimport which from 'which-promise';\nimport { merge } from 'lodash';\nimport exists from '../utils/fs-exists-promised';\nimport { exec, execPowershell } from '../utils/exec';\nimport unquote from '../utils/unquote';\nimport ALIASES from '../aliases';\n\nconst MICROSOFT_EDGE_CLASS      = 'Microsoft.MicrosoftEdge';\nconst MICROSOFT_EDGE_KEY_GLOB   = `HKCU\\\\Software\\\\Classes\\\\ActivatableClasses\\\\Package\\\\${MICROSOFT_EDGE_CLASS}*`;\nconst BROWSER_COMMANDS_KEY_GLOB = root => `${root}\\\\Software\\\\Clients\\\\StartMenuInternet\\\\*\\\\shell\\\\open\\\\command`;\n\nconst LINE_WRAP        = '\\r\\n';\nconst DOUBLE_LINE_WRAP = LINE_WRAP + LINE_WRAP;\n\nconst REGISTRY_BROWSER_PATH_PROPERTY = '(default)';\nconst REGISTRY_BROWSER_NAME_PROPERTY = 'PSPath';\nconst REGISTRY_PROPERTIES_RE         = /^(\\(default\\)|PSPath)\\s*:\\s*(.*)$/;\n\nconst MAX_OUTPUT_WIDTH = 2 ** 31 - 1;\n\nconst POWERSHELL_PIPE_SYMBOL = '|';\n\nconst GET_REGISTRY_KEY_COMMAND          = key => `Get-Item 'Registry::${key}'`;\nconst GET_BROWSER_PATH_PROPERTY_COMMAND = `Get-ItemProperty -Name '${REGISTRY_BROWSER_PATH_PROPERTY}'`;\nconst FORMAT_BROWSER_INFO_COMMAND       = `Format-List -Property '${REGISTRY_BROWSER_PATH_PROPERTY}','${REGISTRY_BROWSER_NAME_PROPERTY}'`;\nconst LIMIT_OUTPUT_WIDTH_COMMAND        = `Out-String -Width ${MAX_OUTPUT_WIDTH}`;\n\nconst GET_REGISTRY_BROWSER_PROPERTIES_COMMAND = key => [\n    GET_REGISTRY_KEY_COMMAND(key),\n    GET_BROWSER_PATH_PROPERTY_COMMAND,\n    FORMAT_BROWSER_INFO_COMMAND,\n    LIMIT_OUTPUT_WIDTH_COMMAND\n].join(POWERSHELL_PIPE_SYMBOL);\n\n// Installation info cache\nvar installationsCache = null;\n\nasync function getRegistryKey (regKeyGlob) {\n    return execPowershell(GET_REGISTRY_KEY_COMMAND(regKeyGlob));\n}\n\nasync function getRegistryBrowserProperties (regKeyGlob) {\n    return execPowershell(GET_REGISTRY_BROWSER_PROPERTIES_COMMAND(regKeyGlob));\n}\n\n// Find installations for different platforms\nasync function addInstallation (installations, name, instPath) {\n    var fileExists = await exists(instPath);\n\n    if (fileExists) {\n        Object.keys(ALIASES).some(alias => {\n            var { nameRe, cmd, macOpenCmdTemplate, path } = ALIASES[alias];\n\n            if (nameRe.test(name)) {\n                installations[alias] = { path: path || instPath, cmd, macOpenCmdTemplate };\n                return true;\n            }\n\n            return false;\n        });\n    }\n}\n\nasync function detectMicrosoftEdge () {\n    const registryResult = await getRegistryKey(MICROSOFT_EDGE_KEY_GLOB);\n\n    if (registryResult.stdout.includes(MICROSOFT_EDGE_CLASS))\n        return ALIASES['edge'];\n\n    return null;\n}\n\nasync function searchInRegistry (registryRoot) {\n    const installations  = {};\n    const registryResult =  await getRegistryBrowserProperties(BROWSER_COMMANDS_KEY_GLOB(registryRoot));\n\n    const data = registryResult.stdout\n        .split(DOUBLE_LINE_WRAP)\n        .map(block => block\n            .split(LINE_WRAP)\n            .map(line => line.match(REGISTRY_PROPERTIES_RE))\n            .filter(match => !!match)\n            .map(match => ({\n                [match[1]]: unquote(match[2])\n            }))\n        )\n        .filter(block => block && block.length)\n        .map(block => merge(...block));\n\n    await Promise.all(data.map(record => addInstallation(installations, record[REGISTRY_BROWSER_NAME_PROPERTY], record[REGISTRY_BROWSER_PATH_PROPERTY])));\n\n    return installations;\n}\n\nasync function findWindowsBrowsers () {\n    var machineRegisteredBrowsers = await searchInRegistry('HKEY_LOCAL_MACHINE');\n    var userRegisteredBrowsers    = await searchInRegistry('HKEY_CURRENT_USER');\n    var installations             = Object.assign(machineRegisteredBrowsers, userRegisteredBrowsers);\n    var edgeAlias                 = await detectMicrosoftEdge();\n\n    if (edgeAlias)\n        installations['edge'] = edgeAlias;\n\n    return installations;\n}\n\nasync function findMacBrowsers () {\n    var installations = {};\n\n    // NOTE: replace the space symbol with code, because grep splits strings by space.\n    var stdout = await exec('ls \"/Applications/\" | grep -E \"Chrome|Firefox|Opera|Safari|Chromium\" | sed -E \"s/ /032/\"');\n\n    await Promise.all(stdout\n        .split('\\n')\n        .filter(fileName => !!fileName)\n        .map(fileName => {\n            // NOTE: restore space\n            fileName = fileName.replace(/032/g, ' ');\n\n            var name = fileName.replace(/.app$/, '');\n            var path = `/Applications/${fileName}`;\n\n            return addInstallation(installations, name, path);\n        }));\n\n    return installations;\n}\n\nasync function findLinuxBrowsers () {\n    var installations = {};\n\n    var aliasCheckingPromises = Object\n        .keys(ALIASES)\n        .map(name => {\n            var { linuxBinaries } = ALIASES[name];\n\n            if (!linuxBinaries)\n                return null;\n\n            var detectionPromises = linuxBinaries\n                .map(binary => {\n                    return which(binary)\n                        .then(path => addInstallation(installations, name, path))\n                        .catch(() => {\n                            // NOTE: binary not found, just do nothing\n                            return;\n                        });\n                });\n\n            return Promise.all(detectionPromises);\n        });\n\n    await Promise.all(aliasCheckingPromises);\n\n    return installations;\n}\n\nasync function findBrowsers () {\n    if (OS.win)\n        return await findWindowsBrowsers();\n\n    if (OS.mac)\n        return await findMacBrowsers();\n\n    if (OS.linux)\n        return await findLinuxBrowsers();\n}\n\n\n// API\n/** @typedef {Object} BrowserInfo\n * @description Object that contains information about the browser installed on the machine.\n * @property {string|undefined} path - The path to the executable file that starts the browser.\n *  Required on MacOS machines. On Windows machines, it is used when the winOpenCmdTemplate property is undefined.\n * @property {string} cmd - Additional command line parameters.\n * @property {string} macOpenCmdTemplate - A [Mustache template](https://github.com/janl/mustache.js#templates)\n *  that provides parameters for launching the browser on a MacOS machine.\n * @property {string|undefined} winOpenCmdTemplate - A [Mustache template](https://github.com/janl/mustache.js#templates)\n *  that provides parameters for launching the browser on a Windows machine.  If undefined, the path to the\n *  executable file specified by the path property is used.\n * @example\n *  {\n *       path: 'C:\\\\ProgramFiles\\\\...\\\\firefox.exe',\n *       cmd: '-new-window',\n *       macOpenCmdTemplate: 'open -a \"{{{path}}}\" {{{pageUrl}}} --args {{{cmd}}}'\n *  }\n */\n\n/**\n * Returns the list of the {@link BrowserInfo} objects that contain information about the browsers installed on the machine.\n * @function\n * @async\n * @name getInstallations\n * @returns {Object.<string, BrowserInfo>} List of the {@link BrowserInfo} objects\n *   containing information about the browsers installed on the machine.\n * @example\n * {\n *   chrome: {\n *       path: 'C:\\\\ProgramFiles\\\\...\\\\chrome.exe',\n *       cmd: '--new-window',\n *       macOpenCmdTemplate: 'open -n -a \"{{{path}}}\" --args {{{pageUrl}}} {{{cmd}}}'\n *   },\n *\n *   firefox: {\n *       path: 'C:\\\\ProgramFiles\\\\...\\\\firefox.exe',\n *       cmd: '-new-window',\n *       macOpenCmdTemplate: 'open -a \"{{{path}}}\" {{{pageUrl}}} --args {{{cmd}}}'\n *   }\n * }\n */\nexport default async function () {\n    if (!installationsCache)\n        installationsCache = await findBrowsers();\n\n    return installationsCache;\n}\n"]}

25

lib/utils/exec.js

@@ -11,2 +11,3 @@ "use strict";

const del_1 = __importDefault(require("del"));
const execa_1 = __importDefault(require("execa"));
const os_family_1 = __importDefault(require("os-family"));

@@ -16,2 +17,3 @@ const nanoid_1 = __importDefault(require("nanoid"));

const binaries_1 = __importDefault(require("../binaries"));
const flatten_whitespace_1 = __importDefault(require("./flatten-whitespace"));
const errors_1 = require("../errors");

@@ -21,2 +23,16 @@ const EXIT_CODE_REGEXP = /Exit code: (-?\d+)/;

const TEMP_PIPE_NAME = seed => `testcafe-browser-tools-fifo-${seed}`;
const POWERSHELL_BINARY = 'powershell.exe';
const POWERSHELL_ARGS = ['-NoLogo', '-NonInteractive', '-Command'];
const POWERSHELL_COMMAND_WRAPPER = command => flatten_whitespace_1.default `
$cp = (chcp | Select-String '\\d+').Matches.Value;
Try
{
chcp 65001;
${command};
}
Finally
{
chcp $cp;
}
`;
function getTempPipePath() {

@@ -92,2 +108,9 @@ return path_1.default.join(os_1.default.tmpdir(), TEMP_PIPE_NAME(nanoid_1.default()));

exports.exec = exec;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"exec.js","sourceRoot":"","sources":["../../src/utils/exec.js"],"names":[],"mappings":"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","sourcesContent":["import childProc from 'child_process';\nimport fs from 'fs';\nimport os from 'os';\nimport path from 'path';\nimport del from 'del';\nimport OS from 'os-family';\nimport nanoid from 'nanoid';\nimport promisify from './promisify';\nimport BINARIES from '../binaries';\nimport { NativeBinaryHasFailedError } from '../errors';\n\n\nconst EXIT_CODE_REGEXP = /Exit code: (-?\\d+)/;\n\nconst OPEN_PATH      = '/usr/bin/open';\nconst TEMP_PIPE_NAME = seed => `testcafe-browser-tools-fifo-${seed}`;\n\nfunction getTempPipePath () {\n    return path.join(os.tmpdir(), TEMP_PIPE_NAME(nanoid()));\n}\n\nvar execFilePromise = promisify(childProc.execFile);\nvar execPromise     = promisify(childProc.exec);\n\n\nfunction readPipe (pipePath) {\n    return new Promise((resolve, reject) => {\n        let data     = '';\n        const stream = fs.createReadStream(pipePath);\n\n        stream.on('data', newData => data += newData ? newData.toString() : '');\n        stream.on('end', () => resolve(data));\n        stream.on('error', reject);\n    });\n}\n\nfunction spawnApp (pipePath, binaryPath, args) {\n    return new Promise((resolve, reject) => {\n        const child = childProc.spawn(OPEN_PATH, ['-n', '-a', BINARIES.app, '--args', pipePath, binaryPath, ...args]);\n\n        let outputData = '';\n\n        child.on('error', reject);\n\n        child.on('exit', code => {\n            if (code)\n                reject(new NativeBinaryHasFailedError({ binary: binaryPath, exitCode: code, output: outputData }));\n            else\n                resolve();\n        });\n\n        child.stdout.on('data', data => outputData += String(data));\n        child.stderr.on('data', data => outputData += String(data));\n    });\n}\n\nasync function runWithMacApp (binaryPath, args) {\n    const pipePath = getTempPipePath();\n\n    await execPromise(`mkfifo ${pipePath}`);\n\n    try {\n        const [data] = await Promise.all([\n            readPipe(pipePath),\n            spawnApp(pipePath, binaryPath, args)\n        ]);\n\n        const exitCodeMatch = data.match(EXIT_CODE_REGEXP);\n\n        if (!exitCodeMatch)\n            return data;\n\n        const exitCode = Number(exitCodeMatch[1]);\n\n        if (exitCode)\n            throw new NativeBinaryHasFailedError({ binary: binaryPath, output: data, exitCode });\n\n        return data;\n    }\n    finally {\n        await del(pipePath, { force: true });\n    }\n}\n\n//API\nexport async function execFile (filePath, args) {\n    try {\n        if (OS.mac)\n            return await runWithMacApp(filePath, args);\n\n        return await execFilePromise(filePath, args);\n    }\n    catch (err) {\n        if (err instanceof NativeBinaryHasFailedError)\n            throw err;\n\n        const errorCode = err.status || err.code;\n\n        if (errorCode === void 0 || typeof errorCode === 'string')\n            throw err;\n\n        throw new NativeBinaryHasFailedError({ binary: filePath, exitCode: errorCode });\n    }\n}\n\nexport async function exec (command) {\n    return execPromise(command, { env: process.env });\n}\n"]}
async function execPowershell(command) {
const wrappedCommand = POWERSHELL_COMMAND_WRAPPER(command);
// NOTE: We have to ignore stdin due to a problem with PowerShell 2.0
// See https://stackoverflow.com/a/9157170/11818061 for details.
return execa_1.default(POWERSHELL_BINARY, [...POWERSHELL_ARGS, `"${wrappedCommand}"`], { stdin: 'ignore', shell: true });
}
exports.execPowershell = execPowershell;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"exec.js","sourceRoot":"","sources":["../../src/utils/exec.js"],"names":[],"mappings":"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","sourcesContent":["import childProc from 'child_process';\nimport fs from 'fs';\nimport os from 'os';\nimport path from 'path';\nimport del from 'del';\nimport execa from 'execa';\nimport OS from 'os-family';\nimport nanoid from 'nanoid';\nimport promisify from './promisify';\nimport BINARIES from '../binaries';\nimport flattenWhitespace from './flatten-whitespace';\nimport { NativeBinaryHasFailedError } from '../errors';\n\n\nconst EXIT_CODE_REGEXP = /Exit code: (-?\\d+)/;\n\nconst OPEN_PATH      = '/usr/bin/open';\nconst TEMP_PIPE_NAME = seed => `testcafe-browser-tools-fifo-${seed}`;\n\nconst POWERSHELL_BINARY = 'powershell.exe';\nconst POWERSHELL_ARGS   = ['-NoLogo', '-NonInteractive', '-Command'];\n\nconst POWERSHELL_COMMAND_WRAPPER = command => flattenWhitespace `\n    $cp = (chcp | Select-String '\\\\d+').Matches.Value;\n    Try\n    {\n        chcp 65001;\n        ${command};\n    }\n    Finally\n    {\n        chcp $cp;\n    }\n`;\n\nfunction getTempPipePath () {\n    return path.join(os.tmpdir(), TEMP_PIPE_NAME(nanoid()));\n}\n\nvar execFilePromise = promisify(childProc.execFile);\nvar execPromise     = promisify(childProc.exec);\n\n\nfunction readPipe (pipePath) {\n    return new Promise((resolve, reject) => {\n        let data     = '';\n        const stream = fs.createReadStream(pipePath);\n\n        stream.on('data', newData => data += newData ? newData.toString() : '');\n        stream.on('end', () => resolve(data));\n        stream.on('error', reject);\n    });\n}\n\nfunction spawnApp (pipePath, binaryPath, args) {\n    return new Promise((resolve, reject) => {\n        const child = childProc.spawn(OPEN_PATH, ['-n', '-a', BINARIES.app, '--args', pipePath, binaryPath, ...args]);\n\n        let outputData = '';\n\n        child.on('error', reject);\n\n        child.on('exit', code => {\n            if (code)\n                reject(new NativeBinaryHasFailedError({ binary: binaryPath, exitCode: code, output: outputData }));\n            else\n                resolve();\n        });\n\n        child.stdout.on('data', data => outputData += String(data));\n        child.stderr.on('data', data => outputData += String(data));\n    });\n}\n\nasync function runWithMacApp (binaryPath, args) {\n    const pipePath = getTempPipePath();\n\n    await execPromise(`mkfifo ${pipePath}`);\n\n    try {\n        const [data] = await Promise.all([\n            readPipe(pipePath),\n            spawnApp(pipePath, binaryPath, args)\n        ]);\n\n        const exitCodeMatch = data.match(EXIT_CODE_REGEXP);\n\n        if (!exitCodeMatch)\n            return data;\n\n        const exitCode = Number(exitCodeMatch[1]);\n\n        if (exitCode)\n            throw new NativeBinaryHasFailedError({ binary: binaryPath, output: data, exitCode });\n\n        return data;\n    }\n    finally {\n        await del(pipePath, { force: true });\n    }\n}\n\n//API\nexport async function execFile (filePath, args) {\n    try {\n        if (OS.mac)\n            return await runWithMacApp(filePath, args);\n\n        return await execFilePromise(filePath, args);\n    }\n    catch (err) {\n        if (err instanceof NativeBinaryHasFailedError)\n            throw err;\n\n        const errorCode = err.status || err.code;\n\n        if (errorCode === void 0 || typeof errorCode === 'string')\n            throw err;\n\n        throw new NativeBinaryHasFailedError({ binary: filePath, exitCode: errorCode });\n    }\n}\n\nexport async function exec (command) {\n    return execPromise(command, { env: process.env });\n}\n\nexport async function execPowershell (command) {\n    const wrappedCommand = POWERSHELL_COMMAND_WRAPPER(command);\n\n    // NOTE: We have to ignore stdin due to a problem with PowerShell 2.0\n    // See https://stackoverflow.com/a/9157170/11818061 for details.\n    return execa(POWERSHELL_BINARY, [...POWERSHELL_ARGS, `\"${wrappedCommand}\"`], { stdin: 'ignore', shell: true });\n}\n"]}
{
"name": "testcafe-browser-tools",
"version": "2.0.1",
"version": "2.0.2",
"description": "An utility library for performing platform-dependent actions on browsers.",

@@ -22,2 +22,3 @@ "homepage": "https://github.com/DevExpress/testcafe-browser-tools",

"graceful-fs": "^4.1.11",
"execa": "^3.3.0",
"linux-platform-info": "^0.0.3",

@@ -40,3 +41,2 @@ "mkdirp": "^0.5.1",

"eslint-plugin-babel": "^2.1.1",
"execa": "^0.9.0",
"express": "^3.21.1",

@@ -43,0 +43,0 @@ "gulp": "^4.0.2",

SocketSocket SOC 2 Logo

Product

  • Package Alerts
  • Integrations
  • Docs
  • Pricing
  • FAQ
  • Roadmap
  • Changelog

Packages

npm

Stay in touch

Get open source security insights delivered straight into your inbox.


  • Terms
  • Privacy
  • Security

Made with ⚡️ by Socket Inc