You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

348 lines
10 KiB

var fs = require('graceful-fs')
var path = require('path')
var mkdirp = require('mkdirp')
var test = require('tap').test
var rimraf = require('rimraf')
var common = require('../common-tap')
var pkg = path.resolve(__dirname, 'run-script')
var cache = path.resolve(pkg, 'cache')
var tmp = path.resolve(pkg, 'tmp')
var opts = { cwd: pkg }
var fullyPopulated = {
'name': 'runscript',
'version': '1.2.3',
'scripts': {
'start': 'node -e "console.log(process.argv[1] || \'start\')"',
'prewith-pre': 'node -e "console.log(process.argv[1] || \'pre\')"',
'with-pre': 'node -e "console.log(process.argv[1] || \'main\')"',
'with-post': 'node -e "console.log(process.argv[1] || \'main\')"',
'postwith-post': 'node -e "console.log(process.argv[1] || \'post\')"',
'prewith-both': 'node -e "console.log(process.argv[1] || \'pre\')"',
'with-both': 'node -e "console.log(process.argv[1] || \'main\')"',
'postwith-both': 'node -e "console.log(process.argv[1] || \'post\')"',
'stop': 'node -e "console.log(process.argv[1] || \'stop\')"',
'env-vars': 'node -e "console.log(process.env.run_script_foo_var)"',
'npm-env-vars': 'node -e "console.log(process.env.npm_run_script_foo_var)"',
'package-env-vars': 'node -e "console.log(process.env.run_script_foo_var)"',
'prefixed-package-env-vars': 'node -e "console.log(process.env.npm_package_run_script_foo_var)"'
},
'run_script_foo_var': 'run_script_test_foo_val'
}
var lifecycleOnly = {
name: 'scripted',
version: '1.2.3',
scripts: {
'prestart': 'echo prestart'
}
}
var directOnly = {
name: 'scripted',
version: '1.2.3',
scripts: {
'whoa': 'echo whoa'
}
}
var both = {
name: 'scripted',
version: '1.2.3',
scripts: {
'prestart': 'echo prestart',
'whoa': 'echo whoa'
}
}
var preversionOnly = {
name: 'scripted',
version: '1.2.3',
scripts: {
'preversion': 'echo preversion'
}
}
var exitCode = {
name: 'scripted',
version: '1.2.3',
scripts: {
'start': 'node -e "process.exit(7)"'
}
}
function testOutput (t, command, er, code, stdout, stderr) {
var lines
if (er) throw er
if (stderr) {
throw new Error('npm ' + command + ' stderr: ' + stderr.toString())
}
lines = stdout.trim().split('\n')
stdout = lines.filter(function (line) {
return line.trim() !== '' && line[0] !== '>'
}).join(';')
t.equal(stdout, command)
t.end()
}
function writeMetadata (object) {
fs.writeFileSync(
path.resolve(pkg, 'package.json'),
JSON.stringify(object, null, 2) + '\n'
)
}
function cleanup () {
rimraf.sync(pkg)
}
test('setup', function (t) {
cleanup()
mkdirp.sync(cache)
mkdirp.sync(tmp)
writeMetadata(fullyPopulated)
t.end()
})
test('npm run-script start', function (t) {
common.npm(['run-script', 'start'], opts, testOutput.bind(null, t, 'start'))
})
test('npm run-script with args', function (t) {
common.npm(['run-script', 'start', '--', 'stop'], opts, testOutput.bind(null, t, 'stop'))
})
test('npm run-script with args that contain spaces', function (t) {
common.npm(['run-script', 'start', '--', 'hello world'], opts, testOutput.bind(null, t, 'hello world'))
})
test('npm run-script with args that contain single quotes', function (t) {
common.npm(['run-script', 'start', '--', 'they"re awesome'], opts, testOutput.bind(null, t, 'they"re awesome'))
})
test('npm run-script with args that contain double quotes', function (t) {
common.npm(['run-script', 'start', '--', 'what"s "up"?'], opts, testOutput.bind(null, t, 'what"s "up"?'))
})
test('npm run-script with args that contain ticks', function (t) {
common.npm(['run-script', 'start', '--', 'what\'s \'up\'?'], opts, testOutput.bind(null, t, 'what\'s \'up\'?'))
})
test('npm run-script with pre script', function (t) {
common.npm(['run-script', 'with-post'], opts, testOutput.bind(null, t, 'main;post'))
})
test('npm run-script with post script', function (t) {
common.npm(['run-script', 'with-pre'], opts, testOutput.bind(null, t, 'pre;main'))
})
test('npm run-script with both pre and post script', function (t) {
common.npm(['run-script', 'with-both'], opts, testOutput.bind(null, t, 'pre;main;post'))
})
test('npm run-script with both pre and post script and with args', function (t) {
common.npm(['run-script', 'with-both', '--', 'an arg'], opts, testOutput.bind(null, t, 'pre;an arg;post'))
})
test('npm run-script explicitly call pre script with arg', function (t) {
common.npm(['run-script', 'prewith-pre', '--', 'an arg'], opts, testOutput.bind(null, t, 'an arg'))
})
test('npm run-script test', function (t) {
common.npm(['run-script', 'test'], opts, function (er, code, stdout, stderr) {
t.ifError(er, 'npm run-script test ran without issue')
t.notOk(stderr, 'should not generate errors')
t.end()
})
})
test('npm run-script env', function (t) {
common.npm(['run-script', 'env'], opts, function (er, code, stdout, stderr) {
t.ifError(er, 'using default env script')
t.notOk(stderr, 'should not generate errors')
t.ok(stdout.indexOf('npm_config_init_version') > 0, 'expected values in var list')
t.end()
})
})
test('npm run-script nonexistent-script', function (t) {
common.npm(['run-script', 'nonexistent-script'], opts, function (er, code, stdout, stderr) {
t.ifError(er, 'npm run-script nonexistent-script did not cause npm to explode')
t.ok(stderr, 'should generate errors')
t.end()
})
})
test('npm run-script restart when there isn\'t restart', function (t) {
common.npm(['run-script', 'restart'], opts, testOutput.bind(null, t, 'stop;start'))
})
test('npm run-script nonexistent-script with --if-present flag', function (t) {
common.npm(['run-script', '--if-present', 'nonexistent-script'], opts, function (er, code, stdout, stderr) {
t.ifError(er, 'npm run-script --if-present non-existent-script ran without issue')
t.notOk(stderr, 'should not generate errors')
t.end()
})
})
test('npm run-script env vars accessible', function (t) {
process.env.run_script_foo_var = 'run_script_test_foo_val'
common.npm(['run-script', 'env-vars'], {
cwd: pkg
}, function (err, code, stdout, stderr) {
t.ifError(err, 'ran run-script without crashing')
t.equal(code, 0, 'exited normally')
t.equal(stderr, '', 'no error output')
t.match(stdout,
new RegExp(process.env.run_script_foo_var),
'script had env access')
t.end()
})
})
test('npm run-script package.json vars injected', function (t) {
common.npm(['run-script', 'package-env-vars'], {
cwd: pkg
}, function (err, code, stdout, stderr) {
t.ifError(err, 'ran run-script without crashing')
t.equal(code, 0, 'exited normally')
t.equal(stderr, '', 'no error output')
t.match(stdout,
new RegExp(fullyPopulated.run_script_foo_var),
'script injected package.json value')
t.end()
})
})
test('npm run-script package.json vars injected with prefix', function (t) {
common.npm(['run-script', 'prefixed-package-env-vars'], {
cwd: pkg
}, function (err, code, stdout, stderr) {
t.ifError(err, 'ran run-script without crashing')
t.equal(code, 0, 'exited normally')
t.equal(stderr, '', 'no error output')
t.match(stdout,
new RegExp(fullyPopulated.run_script_foo_var),
'script injected npm_package-prefixed package.json value')
t.end()
})
})
test('npm run-script env vars stripped npm-prefixed', function (t) {
process.env.npm_run_script_foo_var = 'run_script_test_foo_val'
common.npm(['run-script', 'npm-env-vars'], {
cwd: pkg
}, function (err, code, stdout, stderr) {
t.ifError(err, 'ran run-script without crashing')
t.equal(code, 0, 'exited normally')
t.equal(stderr, '', 'no error output')
t.notMatch(stdout,
new RegExp(process.env.npm_run_script_foo_var),
'script stripped npm-prefixed env var')
t.end()
})
})
test('npm run-script no-params (lifecycle only)', function (t) {
var expected = [
'Lifecycle scripts included in scripted:',
' prestart',
' echo prestart',
''
].join('\n')
writeMetadata(lifecycleOnly)
common.npm(['run-script'], opts, function (err, code, stdout, stderr) {
t.ifError(err, 'ran run-script without parameters without crashing')
t.notOk(code, 'npm exited without error code')
t.notOk(stderr, 'npm printed nothing to stderr')
t.equal(stdout, expected, 'got expected output')
t.end()
})
})
test('npm run-script no-params (preversion only)', function (t) {
var expected = [
'Lifecycle scripts included in scripted:',
' preversion',
' echo preversion',
''
].join('\n')
writeMetadata(preversionOnly)
common.npm(['run-script'], opts, function (err, code, stdout, stderr) {
t.ifError(err, 'ran run-script without parameters without crashing')
t.notOk(code, 'npm exited without error code')
t.notOk(stderr, 'npm printed nothing to stderr')
t.equal(stdout, expected, 'got expected output')
t.end()
})
})
test('npm run-script no-params (direct only)', function (t) {
var expected = [
'Scripts available in scripted via `npm run-script`:',
' whoa',
' echo whoa',
''
].join('\n')
writeMetadata(directOnly)
common.npm(['run-script'], opts, function (err, code, stdout, stderr) {
t.ifError(err, 'ran run-script without parameters without crashing')
t.notOk(code, 'npm exited without error code')
t.notOk(stderr, 'npm printed nothing to stderr')
t.equal(stdout, expected, 'got expected output')
t.end()
})
})
test('npm run-script no-params (direct only)', function (t) {
var expected = [
'Lifecycle scripts included in scripted:',
' prestart',
' echo prestart',
'',
'available via `npm run-script`:',
' whoa',
' echo whoa',
''
].join('\n')
writeMetadata(both)
common.npm(['run-script'], opts, function (err, code, stdout, stderr) {
t.ifError(err, 'ran run-script without parameters without crashing')
t.notOk(code, 'npm exited without error code')
t.notOk(stderr, 'npm printed nothing to stderr')
t.equal(stdout, expected, 'got expected output')
t.end()
})
})
test('npm run-script keep non-zero exit code', function (t) {
writeMetadata(exitCode)
common.npm(['run-script', 'start'], opts, function (err, code, stdout, stderr) {
t.ifError(err, 'ran run-script without parameters without crashing')
t.equal(code, 7, 'got expected exit code')
t.ok(stderr, 'should generate errors')
t.end()
})
})
test('cleanup', function (t) {
cleanup()
t.end()
})