blob: ff62fb1ac4a5815613bcb0414ddc60ae6f6e1f71 [file] [log] [blame]
#!/usr/bin/env node
/**
Licensed to the Apache Software Foundation (ASF) under one
or more contributor license agreements. See the NOTICE file
distributed with this work for additional information
regarding copyright ownership. The ASF licenses this file
to you under the Apache License, Version 2.0 (the
"License"); you may not use this file except in compliance
with the License. You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing,
software distributed under the License is distributed on an
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
KIND, either express or implied. See the License for the
specific language governing permissions and limitations
under the License.
*/
const fs = require('fs');
const os = require('os');
const util = require('util');
const path = require('path');
const logger = require('cordova-common').CordovaLogger.get();
const kill = require('tree-kill');
const exec = require('./execWrapper').exec;
const execa = require('execa');
const HEADING_LINE_PATTERN = /List of devices/m;
const DEVICE_ROW_PATTERN = /(emulator|device|host)/m;
const KILL_SIGNAL = 'SIGINT';
let simulatorCollection = null;
let simulatorDataCollection = {};
function isWindows () {
return /^win/.test(os.platform());
}
function countAndroidDevices () {
const listCommand = 'adb devices';
logger.info('running:');
logger.info(' ' + listCommand);
let numDevices = 0;
const result = exec(listCommand);
result.stdout.split('\n').forEach(function (line) {
if (!HEADING_LINE_PATTERN.test(line) && DEVICE_ROW_PATTERN.test(line)) {
numDevices += 1;
}
});
return numDevices;
}
function secToMin (seconds) {
return Math.ceil(seconds / 60);
}
function getSimulatorsFolder () {
return path.join(os.homedir(), 'Library', 'Developer', 'CoreSimulator', 'Devices');
}
function getSimulatorModelId (cli, target) {
target = new RegExp(target || '^iPhone');
const args = [
'run',
'--list',
'--emulator'
].concat(module.exports.PARAMEDIC_COMMON_ARGS);
// Fetches all known simulators/emulators.
logger.info('running:');
logger.info(` ${cli} ${args.join(' ')}`);
const result = execa.sync(cli, args);
if (result.exitCode > 0) {
logger.error('Failed to find simulator we deployed to');
return;
}
// Return the individual target that is filtered from the known simulators/emulators based on provided target name. (default: ^iPhone)
const allSimulators = result.stdout
.split('\n');
const matchingSimulators = allSimulators.filter(i => i.match(target));
if (!matchingSimulators.length) {
logger.warn('Unable to find requested simulator, falling back to the first available!');
return allSimulators.filter(i => i.match(/^iPhone/)).pop().trim();
}
return matchingSimulators
.pop()
.trim();
}
function getSimulatorCollection () {
if (simulatorCollection) return simulatorCollection;
// Next, figure out the ID of the simulator we found
const instrCommand = 'instruments -s devices | grep ^iPhone';
logger.info('running:');
logger.info(' ' + instrCommand);
const instrResult = exec(instrCommand);
if (instrResult.code > 0) {
logger.error('Failed to get the list of simulators');
return false;
}
simulatorCollection = instrResult.stdout.split('\n');
return simulatorCollection;
}
function getSimulatorData (findSimResult) {
if (simulatorDataCollection[findSimResult]) return simulatorDataCollection[findSimResult];
const simulators = getSimulatorCollection();
const split = findSimResult.split(', ');
// Format of the output is "iPhone-6s-Plus, 9.1"
// Extract the device name and the version number
const device = split[0].replace(/-/g, ' ').trim();
const version = split[1].trim();
// This matches <device> (<version>) [<simulator-id>]
const simIdRegex = /^([a-zA-Z\d ]+) \(([\d.]+)\) \[([a-zA-Z\d-]*)\].*$/;
const simulatorIds = simulators
.reduce((result, line) => {
// replace ʀ in iPhone Xʀ to match ios-sim changes
if (line.indexOf('ʀ') > -1) {
line = line.replace('ʀ', 'R');
}
const simIdMatch = simIdRegex.exec(line);
if (simIdMatch && simIdMatch.length === 4 && simIdMatch[1] === device && simIdMatch[2] === version) {
result.push(encodeURIComponent(simIdMatch[3]));
}
return result;
}, []);
if (simulatorIds.length > 1) {
logger.warn('Multiple matching emulators found. Will use the first matching simulator');
}
simulatorDataCollection[findSimResult] = {
device,
version,
simId: simulatorIds[0]
};
return simulatorDataCollection[findSimResult];
}
function doesFileExist (filePath) {
let fileExists = false;
try {
fs.statSync(filePath);
fileExists = true;
} catch (e) {
fileExists = false;
}
return fileExists;
}
function mkdirSync (path) {
try {
fs.mkdirSync(path);
} catch (e) {
if (e.code !== 'EEXIST') throw e;
}
}
function getSqlite3InsertionCommand (destinationTCCFile, service, appName) {
return util.format('sqlite3 %s "insert into access' +
'(service, client, client_type, allowed, prompt_count, csreq) values(\'%s\', \'%s\', ' +
'0,1,1,NULL)"', destinationTCCFile, service, appName);
}
function contains (collection, item) {
return collection.indexOf(item) !== (-1);
}
function killProcess (pid, callback) {
kill(pid, KILL_SIGNAL, function () {
setTimeout(callback, 1000);
});
}
function getConfigPath (config) {
if (!config) return false;
// if it's absolute or relative to cwd, just return it
let configPath = path.resolve(config);
logger.normal('cordova-paramedic: looking for a config here: ' + configPath);
if (fs.existsSync(configPath)) {
return configPath;
}
// if not, search for it in the 'conf' dir
if (config.indexOf('.config.json') === -1 ||
config.indexOf('.config.json') !== config.length - 12) {
config += '.config.json';
}
configPath = path.join(__dirname, '..', '..', 'conf', config);
logger.normal('cordova-paramedic: looking for a config here: ' + configPath);
if (fs.existsSync(configPath)) {
return configPath;
}
throw new Error('Can\'t find the specified config.');
}
module.exports = {
ANDROID: 'android',
IOS: 'ios',
WINDOWS: 'windows',
BROWSER: 'browser',
PARAMEDIC_DEFAULT_APP_NAME: 'io.cordova.hellocordova',
PARAMEDIC_COMMON_CLI_ARGS: ' --no-telemetry --no-update-notifier',
PARAMEDIC_COMMON_ARGS: ['--no-telemetry', '--no-update-notifier'],
PARAMEDIC_PLUGIN_ADD_ARGS: '',
PARAMEDIC_PLATFORM_ADD_ARGS: '',
SAUCE_USER_ENV_VAR: 'SAUCE_USERNAME',
SAUCE_KEY_ENV_VAR: 'SAUCE_ACCESS_KEY',
SAUCE_TUNNEL_ID_ENV_VAR: 'TRAVIS_JOB_NUMBER',
// retry to establish a tunnel multiple times.
SAUCE_CONNECT_CONNECTION_RETRIES: 5,
// time to wait between connection retries in ms.
SAUCE_CONNECT_CONNECTION_TIMEOUT: 50000,
// retry to download the sauce connect archive multiple times.
SAUCE_CONNECT_DOWNLOAD_RETRIES: 5,
// time to wait between download retries in ms.
SAUCE_CONNECT_DOWNLOAD_TIMEOUT: 1000,
SAUCE_HOST: 'ondemand.saucelabs.com',
SAUCE_PORT: 80,
SAUCE_MAX_DURATION: 5400, // in seconds
DEFAULT_ENCODING: 'utf-8',
WD_TIMEOUT: 30 * 60 * 1000,
WD_RETRY_DELAY: 15000,
WD_RETRIES: 15,
DEFAULT_LOG_TIME: 15,
DEFAULT_LOG_TIME_ADDITIONAL: 2,
TEST_PASSED: true,
TEST_FAILED: false,
secToMin: secToMin,
isWindows: isWindows,
countAndroidDevices: countAndroidDevices,
getSimulatorsFolder: getSimulatorsFolder,
doesFileExist: doesFileExist,
getSqlite3InsertionCommand: getSqlite3InsertionCommand,
getSimulatorModelId: getSimulatorModelId,
getSimulatorData: getSimulatorData,
contains: contains,
mkdirSync: mkdirSync,
killProcess: killProcess,
getConfigPath: getConfigPath
};