Spaces:
Sleeping
Sleeping
| 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; | |
| } | |