pulsar/script/test

246 lines
8.5 KiB
Plaintext
Raw Normal View History

2016-07-29 18:42:56 +03:00
#!/usr/bin/env node
'use strict'
require('colors')
const argv = require('yargs')
.option('core-main', {
describe: 'Run core main process tests',
boolean: true,
default: false
})
.option('skip-main', {
describe: 'Skip main process tests if they would otherwise run on your platform',
boolean: true,
default: false,
conflicts: 'core-main'
})
.option('core-renderer', {
describe: 'Run core renderer process tests',
boolean: true,
default: false
})
.option('core-benchmark', {
describe: 'Run core benchmarks',
boolean: true,
default: false
})
.option('package', {
describe: 'Run bundled package specs',
boolean: true,
default: false
})
.help()
.argv
2016-09-09 12:05:51 +03:00
const assert = require('assert')
2016-07-29 21:23:28 +03:00
const async = require('async')
2016-07-29 18:42:56 +03:00
const childProcess = require('child_process')
const fs = require('fs-extra')
2016-09-09 12:05:51 +03:00
const glob = require('glob')
2016-07-29 21:23:28 +03:00
const path = require('path')
const temp = require('temp').track()
2016-07-29 21:23:28 +03:00
2016-07-29 18:42:56 +03:00
const CONFIG = require('./config')
2017-05-17 16:51:19 +03:00
const backupNodeModules = require('./lib/backup-node-modules')
const runApmInstall = require('./lib/run-apm-install')
2016-07-29 18:42:56 +03:00
const resourcePath = CONFIG.repositoryRootPath
2016-09-09 12:05:51 +03:00
let executablePath
if (process.platform === 'darwin') {
const executablePaths = glob.sync(path.join(CONFIG.buildOutputPath, '*.app'))
assert(executablePaths.length === 1, `More than one application to run tests against was found. ${executablePaths.join(',')}`)
executablePath = path.join(executablePaths[0], 'Contents', 'MacOS', path.basename(executablePaths[0], '.app'))
} else if (process.platform === 'linux') {
2018-05-07 19:53:31 +03:00
const executablePaths = glob.sync(path.join(CONFIG.buildOutputPath, 'atom-*', 'atom'))
2016-09-09 12:05:51 +03:00
assert(executablePaths.length === 1, `More than one application to run tests against was found. ${executablePaths.join(',')}`)
executablePath = executablePaths[0]
2016-09-09 13:07:03 +03:00
} else if (process.platform === 'win32') {
const executablePaths = glob.sync(path.join(CONFIG.buildOutputPath, '**', 'atom*.exe'))
2016-09-09 13:07:03 +03:00
assert(executablePaths.length === 1, `More than one application to run tests against was found. ${executablePaths.join(',')}`)
executablePath = executablePaths[0]
} else {
throw new Error('Running tests on this platform is not supported.')
2016-09-09 12:05:51 +03:00
}
2016-07-29 21:10:05 +03:00
function prepareEnv (suiteName) {
const atomHomeDirPath = temp.mkdirSync(suiteName)
const env = Object.assign({}, process.env, {ATOM_HOME: atomHomeDirPath})
if (process.env.TEST_JUNIT_XML_ROOT) {
// Tell Jasmine to output this suite's results as a JUnit XML file to a subdirectory of the root, so that a
// CI system can interpret it.
2019-02-16 00:09:27 +03:00
const fileName = suiteName + '.xml'
const outputPath = path.join(process.env.TEST_JUNIT_XML_ROOT, fileName)
env.TEST_JUNIT_XML_PATH = outputPath
}
return env
}
2016-07-29 21:10:05 +03:00
function runCoreMainProcessTests (callback) {
const testPath = path.join(CONFIG.repositoryRootPath, 'spec', 'main-process')
const testArguments = [
'--resource-path', resourcePath,
'--test', '--main-process', testPath
]
const testEnv = Object.assign({}, prepareEnv('core-main-process'), {ATOM_GITHUB_INLINE_GIT_EXEC: 'true'})
2016-08-08 16:18:21 +03:00
console.log('Executing core main process tests'.bold.green)
const cp = childProcess.spawn(executablePath, testArguments, {stdio: 'inherit', env: testEnv})
2016-07-29 21:10:05 +03:00
cp.on('error', error => { callback(error) })
cp.on('close', exitCode => { callback(null, {exitCode, step: 'core-main-process'}) })
2016-07-29 21:10:05 +03:00
}
function runCoreRenderProcessTests (callback) {
const testPath = path.join(CONFIG.repositoryRootPath, 'spec')
const testArguments = [
'--resource-path', resourcePath,
'--test', testPath
]
const testEnv = prepareEnv('core-render-process')
2016-07-29 21:10:05 +03:00
2016-08-08 16:18:21 +03:00
console.log('Executing core render process tests'.bold.green)
const cp = childProcess.spawn(executablePath, testArguments, {stdio: 'inherit', env: testEnv})
cp.on('error', error => { callback(error) })
cp.on('close', exitCode => { callback(null, {exitCode, step: 'core-render-process'}) })
}
// Build an array of functions, each running tests for a different bundled package
2016-07-29 21:23:28 +03:00
const packageTestSuites = []
for (let packageName in CONFIG.appMetadata.packageDependencies) {
if (process.env.ATOM_PACKAGES_TO_TEST) {
const packagesToTest = process.env.ATOM_PACKAGES_TO_TEST.split(',').map(pkg => pkg.trim())
if (!packagesToTest.includes(packageName)) continue
}
const repositoryPackagePath = path.join(CONFIG.repositoryRootPath, 'node_modules', packageName)
const testSubdir = ['spec', 'test'].find(subdir => fs.existsSync(path.join(repositoryPackagePath, subdir)))
if (!testSubdir) {
console.log(`No test folder found for package: ${packageName}`.yellow)
continue
}
const testFolder = path.join(repositoryPackagePath, testSubdir)
2016-07-29 21:23:28 +03:00
packageTestSuites.push(function (callback) {
const testArguments = [
'--resource-path', resourcePath,
'--test', testFolder
2016-07-29 21:23:28 +03:00
]
const testEnv = prepareEnv(`bundled-package-${packageName}`)
2016-07-29 21:23:28 +03:00
const pkgJsonPath = path.join(repositoryPackagePath, 'package.json')
const nodeModulesPath = path.join(repositoryPackagePath, 'node_modules')
let finalize = () => null
if (require(pkgJsonPath).atomTestRunner) {
console.log(`Installing test runner dependencies for ${packageName}`.bold.green)
if (fs.existsSync(nodeModulesPath)) {
const backup = backupNodeModules(repositoryPackagePath)
finalize = backup.restore
} else {
finalize = () => fs.removeSync(nodeModulesPath)
}
runApmInstall(repositoryPackagePath)
console.log(`Executing ${packageName} tests`.green)
} else {
console.log(`Executing ${packageName} tests`.bold.green)
}
const cp = childProcess.spawn(executablePath, testArguments, {env: testEnv})
let stderrOutput = ''
2017-03-24 18:38:11 +03:00
cp.stderr.on('data', data => { stderrOutput += data })
cp.stdout.on('data', data => { stderrOutput += data })
cp.on('error', error => {
finalize()
callback(error)
})
cp.on('close', exitCode => {
if (exitCode !== 0) {
console.log(`Package tests failed for ${packageName}:`.red)
console.log(stderrOutput)
}
finalize()
callback(null, {exitCode, step: `package-${packageName}`})
})
2016-07-29 21:23:28 +03:00
})
}
function runBenchmarkTests (callback) {
const benchmarksPath = path.join(CONFIG.repositoryRootPath, 'benchmarks')
const testArguments = ['--benchmark-test', benchmarksPath]
const testEnv = prepareEnv('benchmark')
console.log('Executing benchmark tests'.bold.green)
const cp = childProcess.spawn(executablePath, testArguments, {stdio: 'inherit', env: testEnv})
cp.on('error', error => { callback(error) })
cp.on('close', exitCode => { callback(null, {exitCode, step: 'core-benchmarks'}) })
}
let testSuitesToRun = requestedTestSuites() || testSuitesForPlatform(process.platform)
function requestedTestSuites () {
const suites = []
if (argv.coreMain) {
suites.push(runCoreMainProcessTests)
}
if (argv.coreRenderer) {
suites.push(runCoreRenderProcessTests)
}
if (argv.coreBenchmark) {
suites.push(runBenchmarkTests)
}
if (argv.package) {
suites.push(...packageTestSuites)
}
return suites.length > 0 ? suites : null
}
2016-11-23 02:18:47 +03:00
2017-08-02 15:18:25 +03:00
function testSuitesForPlatform (platform) {
let suites = []
2017-08-02 15:18:25 +03:00
switch (platform) {
case 'darwin':
const PACKAGES_TO_TEST_IN_PARALLEL = 23
if (process.env.ATOM_RUN_CORE_TESTS === 'true') {
suites = [runCoreMainProcessTests, runCoreRenderProcessTests]
2019-06-05 11:05:19 +03:00
} else if (process.env.ATOM_RUN_PACKAGE_TESTS === '1') {
suites = packageTestSuites.slice(0, PACKAGES_TO_TEST_IN_PARALLEL)
2019-06-05 11:05:19 +03:00
} else if (process.env.ATOM_RUN_PACKAGE_TESTS === '2') {
suites = packageTestSuites.slice(PACKAGES_TO_TEST_IN_PARALLEL)
} else {
suites = [runCoreMainProcessTests, runCoreRenderProcessTests].concat(packageTestSuites)
}
break
2017-08-02 15:18:25 +03:00
case 'win32':
suites = (process.arch === 'x64') ? [runCoreMainProcessTests, runCoreRenderProcessTests] : [runCoreMainProcessTests]
break
2017-08-02 15:18:25 +03:00
case 'linux':
suites = [runCoreMainProcessTests]
break
2017-08-02 15:18:25 +03:00
default:
console.log(`Unrecognized platform: ${platform}`)
2016-11-23 02:18:47 +03:00
}
2017-09-09 01:56:48 +03:00
if (argv.skipMainProcessTests) {
suites = suites.filter(suite => suite !== runCoreMainProcessTests)
}
2019-02-16 00:09:27 +03:00
return suites
2016-09-09 12:05:51 +03:00
}
2016-07-29 21:10:05 +03:00
async.series(testSuitesToRun, function (err, results) {
if (err) {
console.error(err)
process.exit(1)
} else {
const failedSteps = results.filter(({exitCode}) => exitCode !== 0)
for (const {step} of failedSteps) {
console.error(`Error! The '${step}' test step finished with a non-zero exit code`)
}
process.exit(failedSteps.length === 0 ? 0 : 1)
}
})