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.
		
		
		
		
		
			
		
			
				
					226 lines
				
				6.7 KiB
			
		
		
			
		
	
	
					226 lines
				
				6.7 KiB
			| 
											3 years ago
										 | 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; | ||
|  | } |