/* 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: ::: */ module.exports = outdated outdated.usage = "npm outdated [ [ ...]]" 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 {:"*"} // 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]] = { 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") } // 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() } } } }