226 lines
6.7 KiB
JavaScript
226 lines
6.7 KiB
JavaScript
|
const path = require('path');
|
||
|
const fs = require('fs');
|
||
|
const existsSync = fs.existsSync;
|
||
|
const utils = require('../utils');
|
||
|
|
||
|
module.exports = exec;
|
||
|
module.exports.expandScript = expandScript;
|
||
|
|
||
|
/**
|
||
|
* 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 };
|
||
|
}
|
||
|
|
||
|
if (pkg.scripts && pkg.scripts.start) {
|
||
|
return { exec: pkg.scripts.start };
|
||
|
}
|
||
|
} catch (e) { }
|
||
|
|
||
|
return null;
|
||
|
}
|
||
|
|
||
|
function replace(map, str) {
|
||
|
var re = new RegExp('{{(' + Object.keys(map).join('|') + ')}}', 'g');
|
||
|
return str.replace(re, function (all, m) {
|
||
|
return map[m] || all || '';
|
||
|
});
|
||
|
}
|
||
|
|
||
|
function expandScript(script, ext) {
|
||
|
if (!ext) {
|
||
|
ext = '.js';
|
||
|
}
|
||
|
if (script.indexOf(ext) !== -1) {
|
||
|
return script;
|
||
|
}
|
||
|
|
||
|
if (existsSync(path.resolve(script))) {
|
||
|
return script;
|
||
|
}
|
||
|
|
||
|
if (existsSync(path.resolve(script + ext))) {
|
||
|
return script + ext;
|
||
|
}
|
||
|
|
||
|
return script;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 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 = {};
|
||
|
}
|
||
|
|
||
|
var options = utils.clone(nodemonOptions || {});
|
||
|
var script;
|
||
|
|
||
|
// if there's no script passed, try to get it from the first argument
|
||
|
if (!options.script && (options.args || []).length) {
|
||
|
script = expandScript(options.args[0],
|
||
|
options.ext && ('.' + (options.ext || 'js').split(',')[0]));
|
||
|
|
||
|
// if the script was found, shift it off our args
|
||
|
if (script !== options.args[0]) {
|
||
|
options.script = script;
|
||
|
options.args.shift();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// 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 (!options.exec && !options.script) {
|
||
|
var found = execFromPackage();
|
||
|
if (found !== null) {
|
||
|
if (found.exec) {
|
||
|
options.exec = found.exec;
|
||
|
}
|
||
|
if (!options.script) {
|
||
|
options.script = found.script;
|
||
|
}
|
||
|
if (Array.isArray(options.args) &&
|
||
|
options.scriptPosition === null) {
|
||
|
options.scriptPosition = options.args.length;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// var options = utils.clone(nodemonOptions || {});
|
||
|
script = path.basename(options.script || '');
|
||
|
|
||
|
var scriptExt = path.extname(script).slice(1);
|
||
|
|
||
|
var extension = options.ext;
|
||
|
if (extension === undefined) {
|
||
|
var isJS = scriptExt === 'js' || scriptExt === 'mjs';
|
||
|
extension = (isJS || !scriptExt) ? 'js,mjs' : scriptExt;
|
||
|
extension += ',json'; // Always watch JSON files
|
||
|
}
|
||
|
|
||
|
var execDefined = !!options.exec;
|
||
|
|
||
|
// 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 = nodemonOptions.execArgs || [];
|
||
|
|
||
|
if (Array.isArray(options.exec)) {
|
||
|
options.execArgs = options.exec;
|
||
|
options.exec = options.execArgs.shift();
|
||
|
}
|
||
|
|
||
|
if (options.exec === undefined) {
|
||
|
options.exec = 'node';
|
||
|
} else {
|
||
|
// allow variable substitution for {{filename}} and {{pwd}}
|
||
|
var substitution = replace.bind(null, {
|
||
|
filename: options.script,
|
||
|
pwd: process.cwd(),
|
||
|
});
|
||
|
|
||
|
var newExec = substitution(options.exec);
|
||
|
if (newExec !== options.exec &&
|
||
|
options.exec.indexOf('{{filename}}') !== -1) {
|
||
|
options.script = null;
|
||
|
}
|
||
|
options.exec = newExec;
|
||
|
|
||
|
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';
|
||
|
|
||
|
// we need to get execArgs set before the script
|
||
|
// for example, in `nodemon --debug my-script.coffee --my-flag`, debug is an
|
||
|
// execArg, while my-flag is a script arg
|
||
|
var leadingArgs = (options.args || []).splice(0, options.scriptPosition);
|
||
|
options.execArgs = options.execArgs.concat(leadingArgs);
|
||
|
options.scriptPosition = 0;
|
||
|
|
||
|
if (options.execArgs.length > 0) {
|
||
|
// because this is the coffee executable, we need to combine the exec args
|
||
|
// into a single argument after the nodejs flag
|
||
|
options.execArgs = ['--nodejs', options.execArgs.join(' ')];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (options.exec === 'coffee') {
|
||
|
// don't override user specified extension tracking
|
||
|
if (options.ext === undefined) {
|
||
|
if (extension) { extension += ','; }
|
||
|
extension += 'coffee,litcoffee';
|
||
|
}
|
||
|
|
||
|
// 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, pug => js,pug
|
||
|
// BIG NOTE: user can't do this: nodemon -e *.js
|
||
|
// because the terminal will automatically expand the glob against
|
||
|
// the file system :(
|
||
|
extension = (extension.match(/[^,*\s]+/g) || [])
|
||
|
.map(ext => ext.replace(/^\./, ''))
|
||
|
.join(',');
|
||
|
|
||
|
options.ext = extension;
|
||
|
|
||
|
if (options.script) {
|
||
|
options.script = expandScript(options.script,
|
||
|
extension && ('.' + extension.split(',')[0]));
|
||
|
}
|
||
|
|
||
|
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;
|
||
|
}
|