UNPKG

start-server-and-test

Version:

Starts server, waits for URL, then runs test command; when the tests end, shuts down server

167 lines (145 loc) 4.36 kB
// @ts-check 'use strict' const la = require('lazy-ass') const is = require('check-more-types') const execa = require('execa') const waitOn = require('wait-on') const Promise = require('bluebird') const psTree = require('ps-tree') const debug = require('debug')('start-server-and-test') /** * Used for timeout (ms) */ const fiveMinutes = 5 * 60 * 1000 const twoSeconds = 2000 const waitOnTimeout = process.env.WAIT_ON_TIMEOUT ? Number(process.env.WAIT_ON_TIMEOUT) : fiveMinutes const waitOnInterval = process.env.WAIT_ON_INTERVAL ? Number(process.env.WAIT_ON_INTERVAL) : twoSeconds const isDebug = () => process.env.DEBUG && process.env.DEBUG.indexOf('start-server-and-test') !== -1 const isInsecure = () => process.env.START_SERVER_AND_TEST_INSECURE function waitAndRun ({ start, url, runFn, namedArguments }) { la(is.unemptyString(start), 'missing start script name', start) la(is.fn(runFn), 'missing test script name', runFn) la( is.unemptyString(url) || is.unemptyArray(url), 'missing url to wait on', url ) const isSuccessfulHttpCode = status => (status >= 200 && status < 300) || status === 304 const validateStatus = namedArguments.expect ? status => status === namedArguments.expect : isSuccessfulHttpCode debug('starting server with command "%s", verbose mode?', start, isDebug()) const server = execa(start, { shell: true, stdio: ['ignore', 'inherit', 'inherit'] }) let serverStopped function stopServer () { debug('getting child processes') if (!serverStopped) { serverStopped = true return Promise.fromNode(cb => psTree(server.pid, cb)) .then(children => { debug('stopping child processes') children.forEach(child => { try { process.kill(child.PID, 'SIGINT') } catch (e) { if (e.code === 'ESRCH') { console.log( `Child process ${child.PID} exited before trying to stop it` ) } else { throw e } } }) }) .then(() => { debug('stopping server') server.kill() }) } } const waited = new Promise((resolve, reject) => { const onClose = () => { reject(new Error('server closed unexpectedly')) } server.on('close', onClose) debug('starting waitOn %s', url) const options = { resources: Array.isArray(url) ? url : [url], interval: waitOnInterval, window: 1000, timeout: waitOnTimeout, verbose: isDebug(), strictSSL: !isInsecure(), log: isDebug(), headers: { Accept: 'text/html, application/json, text/plain, */*' }, validateStatus } debug('wait-on options %o', options) waitOn(options, err => { if (err) { debug('error waiting for url', url) debug(err.message) return reject(err) } debug('waitOn finished successfully') server.removeListener('close', onClose) resolve() }) }) return waited .tapCatch(stopServer) .then(runFn) .finally(stopServer) } const runTheTests = testCommand => () => { debug('running test script command: %s', testCommand) return execa(testCommand, { shell: true, stdio: 'inherit' }) } /** * Starts a single service and runs tests or recursively * runs a service, then goes to the next list, until it reaches 1 service and runs test. */ function startAndTest ({ services, test, namedArguments }) { if (services.length === 0) { throw new Error('Got zero services to start ...') } la( is.number(namedArguments.expect), 'expected status should be a number', namedArguments.expect ) if (services.length === 1) { const runTests = runTheTests(test) debug('single service "%s" to run and test', services[0].start) return waitAndRun({ start: services[0].start, url: services[0].url, namedArguments, runFn: runTests }) } return waitAndRun({ start: services[0].start, url: services[0].url, namedArguments, runFn: () => { debug('previous service started, now going to the next one') return startAndTest({ services: services.slice(1), test, namedArguments }) } }) } module.exports = { startAndTest }