Files
archived-nodemon/lib/config/exec.js
Remy Sharp 9651ab88b0 Escape exec *args* only if space
This matches npm's functionality. It's also been tested against the "some\"file" test, etc. It's hard too...and late.
2015-01-14 22:45:45 +00:00

248 lines
7.1 KiB
JavaScript

'use strict';
var path = require('path'),
utils = require('../utils');
module.exports = exec;
module.exports.parseExecutable = parseExecutable;
function parseExecutable(str) {
var length = str.length;
var i = 0;
var exec = null;
var execArgs = [];
var token = '';
var inspace = false;
var escaped = false;
var unescapeNextTime = false;
var open = false;
var c = null;
for (; i < length; i++) {
c = str[i];
if (unescapeNextTime) {
escaped = false;
}
unescapeNextTime = escaped;
if (c === '\\') {
escaped = true;
// peek next character
if (str[i+1] === ' ') { // space we skip
inspace = true;
}
}
if (!escaped && (c === '"' || c === "'")) {
open = !open;
// continue;
}
if (c === ' ') {
if (!open && inspace === false) {
if (exec === null) {
exec = token;
execArgs.push(str.slice(i + 1));
token = '';
break;
} else {
execArgs.push(token);
}
// reset the token now
token = '';
continue;
} else if (inspace) {
inspace = false;
}
}
token += c;
}
// deal with the left overs
if (exec === null) {
exec = token;
} else if (token) {
execArgs.push(token);
}
var result = {
exec: exec,
execArgs: execArgs,
};
return result;
}
/**
* Reads the cwd/package.json file and looks to see if it can load a script
* and possibly an exec first from package.main, then package.start.
*
* @return {Object} exec & script if found
*/
function execFromPackage() {
// doing a try/catch because we can't use the path.exist callback pattern
// or we could, but the code would get messy, so this will do exactly
// what we're after - if the file doesn't exist, it'll throw.
try {
// note: this isn't nodemon's package, it's the user's cwd package
var pkg = require(path.join(process.cwd(), 'package.json'));
if (pkg.main !== undefined) {
// no app found to run - so give them a tip and get the feck out
return { exec: null, script: pkg.main };
} else if (pkg.scripts && pkg.scripts.start) {
var start = pkg.scripts.start.split(' ');
return { exec: start[0], script: start.slice(1).join(' ') };
}
} catch (e) {}
return null;
}
function replace(map, str) {
return str.replace(new RegExp('\{\{(' + Object.keys(map).join('|') + ')\}\}'), function (all, m) {
return map[m] || all || '';
});
}
/**
* Discovers all the options required to run the script
* and if a custom exec has been passed in, then it will
* also try to work out what extensions to monitor and
* whether there's a special way of running that script.
*
* @param {Object} nodemonOptions
* @param {Object} execMap
* @return {Object} new and updated version of nodemonOptions
*/
function exec(nodemonOptions, execMap) {
if (!execMap) {
execMap = {};
}
// if there's no exec found yet, then try to read it from the local package.json
// this logic used to sit in the cli/parse, but actually the cli should be parsed
// first, then the user options (via nodemon.json) then finally default down to
// pot shots at the directory via package.json
if (!nodemonOptions.exec) {
var found = execFromPackage();
if (found !== null) {
if (found.exec) {
nodemonOptions.exec = found.exec;
}
if (!nodemonOptions.script) {
nodemonOptions.script = found.script;
}
if (Array.isArray(nodemonOptions.args) && nodemonOptions.scriptPosition === null) {
nodemonOptions.scriptPosition = nodemonOptions.args.length;
}
}
}
var options = utils.clone(nodemonOptions || {}),
script = path.basename(options.script || ''),
scriptExt = path.extname(script).slice(1),
extension = options.ext || scriptExt || 'js',
execDefined = !!options.exec;
// strip any leading periods int he extension
if (extension.indexOf('.') === 0) {
extension = extension.slice(1);
}
// allows the user to simplify cli usage: https://github.com/remy/nodemon/issues/195
// but always give preference to the user defined argument
if (!options.exec && execMap[scriptExt] !== undefined) {
options.exec = execMap[scriptExt];
execDefined = true;
}
options.execArgs = [];
if (Array.isArray(options.exec)) {
options.execArgs = options.exec;
options.exec = options.execArgs.shift();
} else if (options.exec && options.exec.indexOf(' ') !== -1) {
options = utils.merge(parseExecutable(options.exec), options);
}
if (options.exec === undefined) {
options.exec = 'node';
} else {
// allow variable substitution for {{filename}} and {{pwd}}
var substitution = replace.bind(null, { filename: script, pwd: process.cwd() });
options.exec = substitution(options.exec);
var newExecArgs = options.execArgs.map(substitution);
if (newExecArgs.join('') !== options.execArgs.join('')) {
options.execArgs = newExecArgs;
delete options.script;
}
}
if (options.exec === 'node' && options.nodeArgs && options.nodeArgs.length) {
options.execArgs = options.execArgs.concat(options.nodeArgs);
}
// note: indexOf('coffee') handles both .coffee and .litcoffee
if (!execDefined && options.exec === 'node' && scriptExt.indexOf('coffee') !== -1) {
options.exec = 'coffee';
// ensure that we call: `coffee --nodejs ...`
if (options.execArgs === undefined) {
options.execArgs = [];
}
// if there's a leading argument to the exec that starts with `--` then
// it could be --debug or --debug-brk or something else intended for node
// so we'll add the --nodejs switch.
if ((options.args || []).join(' ').indexOf('--') === 0) {
options.execArgs.unshift('--nodejs');
}
}
if (options.exec === 'coffee') {
// don't override user specified extension tracking
if (!options.ext) {
extension = 'coffee litcoffee js';
}
// because windows can't find 'coffee', it needs the real file 'coffee.cmd'
if (utils.isWindows) {
options.exec += '.cmd';
}
}
// allow users to make a mistake on the extension to monitor
// converts js,jade => js,jade
// and 'js jade' => js,jade
// BIG NOTE: user can't do this: nodemon -e *.js
// because the terminal will automatically expand the glob against
// the file system :(
if (extension.indexOf(' ') !== -1 ||
extension.indexOf(',') !== -1 ||
extension.indexOf('*.') !== -1) {
extension = extension.replace(/\s+/g, '|') // convert spaces to pipes
.replace(/,/g, '|') // convert commas to pipes
.split('|') // split on those pipes
.map(function (item) {
return item.replace(/^[\*\.]+/, ''); // remove "*."
}).join(','); // return regexp string like: js,jade
}
options.ext = extension;
options.env = {};
// make sure it's an object (and since we don't have )
if (({}).toString.apply(nodemonOptions.env) === '[object Object]') {
options.env = utils.clone(nodemonOptions.env);
} else if (nodemonOptions.env !== undefined) {
throw new Error('nodemon env values must be an object: { PORT: 8000 }');
}
return options;
}