405 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			405 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| /*
 | |
| 
 | |
| npm outdated [pkg]
 | |
| 
 | |
| Does the following:
 | |
| 
 | |
| 1. check for a new version of pkg
 | |
| 
 | |
| If no packages are specified, then run for all installed
 | |
| packages.
 | |
| 
 | |
| --parseable creates output like this:
 | |
| <fullpath>:<name@wanted>:<name@installed>:<name@latest>
 | |
| 
 | |
| */
 | |
| 
 | |
| module.exports = outdated
 | |
| 
 | |
| outdated.usage = "npm outdated [<pkg> [<pkg> ...]]"
 | |
| 
 | |
| outdated.completion = require("./utils/completion/installed-deep.js")
 | |
| 
 | |
| 
 | |
| var path = require("path")
 | |
|   , readJson = require("read-package-json")
 | |
|   , cache = require("./cache.js")
 | |
|   , asyncMap = require("slide").asyncMap
 | |
|   , npm = require("./npm.js")
 | |
|   , url = require("url")
 | |
|   , color = require("ansicolors")
 | |
|   , styles = require("ansistyles")
 | |
|   , table = require("text-table")
 | |
|   , semver = require("semver")
 | |
|   , os = require("os")
 | |
|   , mapToRegistry = require("./utils/map-to-registry.js")
 | |
|   , npa = require("npm-package-arg")
 | |
|   , readInstalled = require("read-installed")
 | |
|   , long = npm.config.get("long")
 | |
|   , log = require("npmlog")
 | |
| 
 | |
| function outdated (args, silent, cb) {
 | |
|   if (typeof cb !== "function") cb = silent, silent = false
 | |
|   var dir = path.resolve(npm.dir, "..")
 | |
| 
 | |
|   // default depth for `outdated` is 0 (cf. `ls`)
 | |
|   if (npm.config.get("depth") === Infinity) npm.config.set("depth", 0)
 | |
| 
 | |
|   outdated_(args, dir, {}, 0, function (er, list) {
 | |
|     if (!list) list = []
 | |
|     if (er || silent || list.length === 0) return cb(er, list)
 | |
|     list.sort(function(a, b) {
 | |
|       var aa = a[1].toLowerCase()
 | |
|         , bb = b[1].toLowerCase()
 | |
|       return aa === bb ? 0
 | |
|            : aa < bb ? -1 : 1
 | |
|     })
 | |
|     if (npm.config.get("json")) {
 | |
|       console.log(makeJSON(list))
 | |
|     } else if (npm.config.get("parseable")) {
 | |
|       console.log(makeParseable(list))
 | |
|     } else {
 | |
|       var outList = list.map(makePretty)
 | |
|       var outHead = [ "Package"
 | |
|                     , "Current"
 | |
|                     , "Wanted"
 | |
|                     , "Latest"
 | |
|                     , "Location"
 | |
|                     ]
 | |
|       if (long) outHead.push("Package Type")
 | |
|       var outTable = [outHead].concat(outList)
 | |
| 
 | |
|       if (npm.color) {
 | |
|         outTable[0] = outTable[0].map(function(heading) {
 | |
|           return styles.underline(heading)
 | |
|         })
 | |
|       }
 | |
| 
 | |
|       var tableOpts = { align: ["l", "r", "r", "r", "l"]
 | |
|                       , stringLength: function(s) { return ansiTrim(s).length }
 | |
|                       }
 | |
|       console.log(table(outTable, tableOpts))
 | |
|     }
 | |
|     cb(null, list)
 | |
|   })
 | |
| }
 | |
| 
 | |
| // [[ dir, dep, has, want, latest, type ]]
 | |
| function makePretty (p) {
 | |
|   var dep = p[1]
 | |
|     , dir = path.resolve(p[0], "node_modules", dep)
 | |
|     , has = p[2]
 | |
|     , want = p[3]
 | |
|     , latest = p[4]
 | |
|     , type = p[6]
 | |
| 
 | |
|   if (!npm.config.get("global")) {
 | |
|     dir = path.relative(process.cwd(), dir)
 | |
|   }
 | |
| 
 | |
|   var columns = [ dep
 | |
|                 , has || "MISSING"
 | |
|                 , want
 | |
|                 , latest
 | |
|                 , dirToPrettyLocation(dir)
 | |
|                 ]
 | |
|   if (long) columns[5] = type
 | |
| 
 | |
|   if (npm.color) {
 | |
|     columns[0] = color[has === want ? "yellow" : "red"](columns[0]) // dep
 | |
|     columns[2] = color.green(columns[2]) // want
 | |
|     columns[3] = color.magenta(columns[3]) // latest
 | |
|     columns[4] = color.brightBlack(columns[4]) // dir
 | |
|     if (long) columns[5] = color.brightBlack(columns[5]) // type
 | |
|   }
 | |
| 
 | |
|   return columns
 | |
| }
 | |
| 
 | |
| function ansiTrim (str) {
 | |
|   var r = new RegExp("\x1b(?:\\[(?:\\d+[ABCDEFGJKSTm]|\\d+;\\d+[Hfm]|" +
 | |
|         "\\d+;\\d+;\\d+m|6n|s|u|\\?25[lh])|\\w)", "g")
 | |
|   return str.replace(r, "")
 | |
| }
 | |
| 
 | |
| function dirToPrettyLocation (dir) {
 | |
|   return dir.replace(/^node_modules[/\\]/, "")
 | |
|             .replace(/[[/\\]node_modules[/\\]/g, " > ")
 | |
| }
 | |
| 
 | |
| function makeParseable (list) {
 | |
|   return list.map(function (p) {
 | |
| 
 | |
|     var dep = p[1]
 | |
|       , dir = path.resolve(p[0], "node_modules", dep)
 | |
|       , has = p[2]
 | |
|       , want = p[3]
 | |
|       , latest = p[4]
 | |
|       , type = p[6]
 | |
| 
 | |
|     var out = [ dir
 | |
|            , dep + "@" + want
 | |
|            , (has ? (dep + "@" + has) : "MISSING")
 | |
|            , dep + "@" + latest
 | |
|            ]
 | |
|    if (long) out.push(type)
 | |
| 
 | |
|    return out.join(":")
 | |
|   }).join(os.EOL)
 | |
| }
 | |
| 
 | |
| function makeJSON (list) {
 | |
|   var out = {}
 | |
|   list.forEach(function (p) {
 | |
|     var dir = path.resolve(p[0], "node_modules", p[1])
 | |
|     if (!npm.config.get("global")) {
 | |
|       dir = path.relative(process.cwd(), dir)
 | |
|     }
 | |
|     out[p[1]] = { current: p[2]
 | |
|                 , wanted: p[3]
 | |
|                 , latest: p[4]
 | |
|                 , location: dir
 | |
|                 }
 | |
|     if (long) out[p[1]].type = p[6]
 | |
|   })
 | |
|   return JSON.stringify(out, null, 2)
 | |
| }
 | |
| 
 | |
| function outdated_ (args, dir, parentHas, depth, cb) {
 | |
|   // get the deps from package.json, or {<dir/node_modules/*>:"*"}
 | |
|   // asyncMap over deps:
 | |
|   //   shouldHave = cache.add(dep, req).version
 | |
|   //   if has === shouldHave then
 | |
|   //     return outdated(args, dir/node_modules/dep, parentHas + has)
 | |
|   //   else if dep in args or args is empty
 | |
|   //     return [dir, dep, has, shouldHave]
 | |
| 
 | |
|   if (depth > npm.config.get("depth")) {
 | |
|     return cb(null, [])
 | |
|   }
 | |
|   var deps = null
 | |
|   var types = {}
 | |
|   readJson(path.resolve(dir, "package.json"), function (er, d) {
 | |
|     d = d || {}
 | |
|     if (er && er.code !== "ENOENT" && er.code !== "ENOTDIR") return cb(er)
 | |
|     deps = (er) ? true : (d.dependencies || {})
 | |
|     if (!er) {
 | |
|       Object.keys(deps).forEach(function (k) {
 | |
|         types[k] = "dependencies"
 | |
|       })
 | |
|     }
 | |
| 
 | |
|     if (npm.config.get("save-dev")) {
 | |
|       deps = d.devDependencies || {}
 | |
|       Object.keys(deps).forEach(function (k) {
 | |
|         types[k] = "devDependencies"
 | |
|       })
 | |
| 
 | |
|       return next()
 | |
|     }
 | |
| 
 | |
|     if (npm.config.get("save")) {
 | |
|       // remove optional dependencies from dependencies during --save.
 | |
|       Object.keys(d.optionalDependencies || {}).forEach(function (k) {
 | |
|         delete deps[k]
 | |
|       })
 | |
|       return next()
 | |
|     }
 | |
| 
 | |
|     if (npm.config.get("save-optional")) {
 | |
|       deps = d.optionalDependencies || {}
 | |
|       Object.keys(deps).forEach(function (k) {
 | |
|         types[k] = "optionalDependencies"
 | |
|       })
 | |
|       return next()
 | |
|     }
 | |
| 
 | |
|     var doUpdate = npm.config.get("dev") ||
 | |
|                     (!npm.config.get("production") &&
 | |
|                     !Object.keys(parentHas).length &&
 | |
|                     !npm.config.get("global"))
 | |
| 
 | |
|     if (!er && d && doUpdate) {
 | |
|       Object.keys(d.devDependencies || {}).forEach(function (k) {
 | |
|         if (!(k in parentHas)) {
 | |
|           deps[k] = d.devDependencies[k]
 | |
|           types[k] = "devDependencies"
 | |
|         }
 | |
|       })
 | |
|     }
 | |
|     return next()
 | |
|   })
 | |
| 
 | |
|   var has = null
 | |
|   readInstalled(path.resolve(dir), { dev : true }, function (er, data) {
 | |
|     if (er) {
 | |
|       has = Object.create(parentHas)
 | |
|       return next()
 | |
|     }
 | |
|     var pkgs = Object.keys(data.dependencies)
 | |
|     pkgs = pkgs.filter(function (p) {
 | |
|       return !p.match(/^[\._-]/)
 | |
|     })
 | |
|     asyncMap(pkgs, function (pkg, cb) {
 | |
|       var jsonFile = path.resolve(dir, "node_modules", pkg, "package.json")
 | |
|       readJson(jsonFile, function (er, d) {
 | |
|         if (er && er.code !== "ENOENT" && er.code !== "ENOTDIR") return cb(er)
 | |
|         if (d && d.name && d.private) delete deps[d.name]
 | |
|         cb(null, er ? [] : [[d.name, d.version, d._from]])
 | |
|       })
 | |
|     }, function (er, pvs) {
 | |
|       if (er) return cb(er)
 | |
|       has = Object.create(parentHas)
 | |
|       pvs.forEach(function (pv) {
 | |
|         has[pv[0]] = {
 | |
|           link: data.dependencies[pv[0]].link,
 | |
|           version: pv[1],
 | |
|           from: pv[2]
 | |
|         }
 | |
|       })
 | |
| 
 | |
|       next()
 | |
|     })
 | |
|   })
 | |
| 
 | |
|   function next () {
 | |
|     if (!has || !deps) return
 | |
|     if (deps === true) {
 | |
|       deps = Object.keys(has).reduce(function (l, r) {
 | |
|         l[r] = "latest"
 | |
|         return l
 | |
|       }, {})
 | |
|     }
 | |
| 
 | |
|     // now get what we should have, based on the dep.
 | |
|     // if has[dep] !== shouldHave[dep], then cb with the data
 | |
|     // otherwise dive into the folder
 | |
|     asyncMap(Object.keys(deps), function (dep, cb) {
 | |
|       if (!long) return shouldUpdate(args, dir, dep, has, deps[dep], depth, cb)
 | |
| 
 | |
|       shouldUpdate(args, dir, dep, has, deps[dep], depth, cb, types[dep])
 | |
|     }, cb)
 | |
|   }
 | |
| }
 | |
| 
 | |
| function shouldUpdate (args, dir, dep, has, req, depth, cb, type) {
 | |
|   // look up the most recent version.
 | |
|   // if that's what we already have, or if it's not on the args list,
 | |
|   // then dive into it.  Otherwise, cb() with the data.
 | |
| 
 | |
|   // { version: , from: }
 | |
|   var curr = has[dep]
 | |
| 
 | |
|   function skip (er) {
 | |
|     // show user that no viable version can be found
 | |
|     if (er) return cb(er)
 | |
|     outdated_( args
 | |
|              , path.resolve(dir, "node_modules", dep)
 | |
|              , has
 | |
|              , depth + 1
 | |
|              , cb )
 | |
|   }
 | |
| 
 | |
|   function doIt (wanted, latest) {
 | |
|     if (!long) {
 | |
|       return cb(null, [[ dir, dep, curr && curr.version, wanted, latest, req]])
 | |
|     }
 | |
|     cb(null, [[ dir, dep, curr && curr.version, wanted, latest, req, type]])
 | |
|   }
 | |
| 
 | |
|   if (args.length && args.indexOf(dep) === -1) return skip()
 | |
|   var parsed = npa(dep + '@' + req)
 | |
|   if (parsed.type === "git" || (parsed.hosted && parsed.hosted.type === "github")) {
 | |
|     return doIt("git", "git")
 | |
|   }
 | |
|   if (curr && curr.link) {
 | |
|     return doIt("linked", "linked")
 | |
|   }
 | |
| 
 | |
|   // search for the latest package
 | |
|   mapToRegistry(dep, npm.config, function (er, uri, auth) {
 | |
|     if (er) return cb(er)
 | |
| 
 | |
|     npm.registry.get(uri, { auth : auth }, updateDeps)
 | |
|   })
 | |
| 
 | |
|   function updateLocalDeps (latestRegistryVersion) {
 | |
|     readJson(path.resolve(parsed.spec, 'package.json'), function (er, localDependency) {
 | |
|       if (er) return cb()
 | |
| 
 | |
|       var wanted = localDependency.version
 | |
|       var latest = localDependency.version
 | |
| 
 | |
|       if (latestRegistryVersion) {
 | |
|         latest = latestRegistryVersion
 | |
|         if (semver.lt(wanted, latestRegistryVersion)) {
 | |
|           wanted = latestRegistryVersion
 | |
|           req = dep + '@' + latest
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       if (curr.version !== wanted) {
 | |
|         doIt(wanted, latest)
 | |
|       } else {
 | |
|         skip()
 | |
|       }
 | |
|     })
 | |
|   }
 | |
| 
 | |
|   function updateDeps (er, d) {
 | |
|     if (er) {
 | |
|       if (parsed.type !== 'local') return cb(er)
 | |
|       return updateLocalDeps()
 | |
|     }
 | |
| 
 | |
|     if (!d || !d["dist-tags"] || !d.versions) return cb()
 | |
|     var l = d.versions[d["dist-tags"].latest]
 | |
|     if (!l) return cb()
 | |
| 
 | |
|     var r = req
 | |
|     if (d["dist-tags"][req])
 | |
|       r = d["dist-tags"][req]
 | |
| 
 | |
|     if (semver.validRange(r, true)) {
 | |
|       // some kind of semver range.
 | |
|       // see if it's in the doc.
 | |
|       var vers = Object.keys(d.versions)
 | |
|       var v = semver.maxSatisfying(vers, r, true)
 | |
|       if (v) {
 | |
|         return onCacheAdd(null, d.versions[v])
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     // We didn't find the version in the doc.  See if cache can find it.
 | |
|     cache.add(dep, req, null, false, onCacheAdd)
 | |
| 
 | |
|     function onCacheAdd(er, d) {
 | |
|       // if this fails, then it means we can't update this thing.
 | |
|       // it's probably a thing that isn't published.
 | |
|       if (er) {
 | |
|         if (er.code && er.code === "ETARGET") {
 | |
|           // no viable version found
 | |
|           return skip(er)
 | |
|         }
 | |
|         return skip()
 | |
|       }
 | |
| 
 | |
|       // check that the url origin hasn't changed (#1727) and that
 | |
|       // there is no newer version available
 | |
|       var dFromUrl = d._from && url.parse(d._from).protocol
 | |
|       var cFromUrl = curr && curr.from && url.parse(curr.from).protocol
 | |
| 
 | |
|       if (!curr || dFromUrl && cFromUrl && d._from !== curr.from
 | |
|           || d.version !== curr.version
 | |
|           || d.version !== l.version) {
 | |
|         if (parsed.type === 'local') return updateLocalDeps(l.version)
 | |
| 
 | |
|         doIt(d.version, l.version)
 | |
|       }
 | |
|       else {
 | |
|         skip()
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| }
 |