node/deps/npm/lib/ls.js

250 lines
7.4 KiB
JavaScript
Raw Normal View History

// show the installed versions of packages
//
// --parseable creates output like this:
// <fullpath>:<name@ver>:<realpath>:<flags>
// Flags are a :-separated list of zero or more indicators
module.exports = exports = ls
var npm = require("./npm.js")
, readInstalled = require("./utils/read-installed.js")
, output = require("./utils/output.js")
, log = require("./utils/log.js")
, relativize = require("./utils/relativize.js")
, path = require("path")
ls.usage = "npm ls"
function ls (args, silent, cb) {
if (typeof cb !== "function") cb = silent, silent = false
if (args.length) {
log.warn("ls doesn't take positional args. Try the 'search' command")
}
var dir = path.resolve(npm.dir, "..")
readInstalled(dir, function (er, data) {
2012-02-25 10:52:17 +08:00
var lite = getLite(bfsify(data))
if (er || silent) return cb(er, data, lite)
var long = npm.config.get("long")
2012-02-25 10:52:17 +08:00
, json = npm.config.get("json")
, out
if (json) {
var seen = []
var d = long ? bfsify(data) : lite
// the raw data can be circular
out = JSON.stringify(d, function (k, o) {
if (typeof o === "object") {
if (-1 !== seen.indexOf(o)) return "[Circular]"
seen.push(o)
}
return o
}, 2)
} else {
out = makePretty(bfsify(data), long, dir).join("\n")
}
output.write(out, function (er) { cb(er, data, lite) })
})
}
2012-02-25 10:52:17 +08:00
function getLite (data, noname) {
var lite = {}
, maxDepth = npm.config.get("depth")
, url = require("url")
2012-02-25 10:52:17 +08:00
if (!noname && data.name) lite.name = data.name
if (data.version) lite.version = data.version
if (data.extraneous) {
lite.extraneous = true
lite.problems = lite.problems || []
lite.problems.push( "extraneous: "
+ data.name + "@" + data.version
+ " " + (data.path || "") )
}
if (data._from) {
var from = data._from
if (from.indexOf(data.name + "@") === 0) {
from = from.substr(data.name.length + 1)
}
var u = url.parse(from)
if (u.protocol) lite.from = from
}
2012-02-25 10:52:17 +08:00
if (data.invalid) {
lite.invalid = true
lite.problems = lite.problems || []
lite.problems.push( "invalid: "
+ data.name + "@" + data.version
+ " " + (data.path || "") )
}
if (data.dependencies) {
var deps = Object.keys(data.dependencies)
if (deps.length) lite.dependencies = deps.map(function (d) {
var dep = data.dependencies[d]
if (typeof dep === "string") {
lite.problems = lite.problems || []
var p
if (data.depth >= maxDepth) {
p = "max depth reached: "
} else {
p = "missing: "
}
p += d + "@" + dep
+ ", required by "
+ data.name + "@" + data.version
lite.problems.push(p)
return [d, { required: dep, missing: true }]
}
return [d, getLite(dep, true)]
}).reduce(function (deps, d) {
if (d[1].problems) {
lite.problems = lite.problems || []
lite.problems.push.apply(lite.problems, d[1].problems)
}
deps[d[0]] = d[1]
return deps
}, {})
}
return lite
}
function bfsify (root, current, queue, seen) {
// walk over the data, and turn it from this:
// +-- a
// | `-- b
// | `-- a (truncated)
// `--b (truncated)
// into this:
// +-- a
// `-- b
// which looks nicer
current = current || root
queue = queue || []
2012-02-09 04:36:20 +08:00
seen = seen || [root]
var deps = current.dependencies = current.dependencies || {}
Object.keys(deps).forEach(function (d) {
var dep = deps[d]
if (typeof dep !== "object") return
if (seen.indexOf(dep) !== -1) {
if (npm.config.get("parseable") || !npm.config.get("long")) {
delete deps[d]
return
} else {
dep = deps[d] = Object.create(dep)
dep.dependencies = {}
}
}
queue.push(dep)
seen.push(dep)
})
if (!queue.length) return root
return bfsify(root, queue.shift(), queue, seen)
}
function makePretty (data, long, dir, prefix, list) {
var top = !list
list = list || []
prefix = prefix || ""
list.push(format(data, long, prefix, dir))
var deps = data.dependencies || {}
, childPref = prefix.split("├─").join("│ ")
.split("└─").join(" ")
, depList = Object.keys(deps)
, depLast = depList.length - 1
, maxDepth = npm.config.get("depth")
Object.keys(deps).sort(function (a, b) {
return a > b ? 1 : -1
}).forEach(function (d, i) {
var depData = deps[d]
if (typeof depData === "string") {
if (data.depth < maxDepth) {
var p = data.link || data.path
log.warn("Unmet dependency in "+p, d+" "+deps[d])
depData = npm.config.get("parseable")
? ( npm.config.get("long")
? path.resolve(data.path, "node_modules", d)
+ ":"+d+"@"+JSON.stringify(depData)+":INVALID:MISSING"
: "" )
: "─ \033[31;40mUNMET DEPENDENCY\033[0m "+d+" "+depData
} else {
if (npm.config.get("parseable")) {
depData = path.resolve(data.path, "node_modules", d)
+ (npm.config.get("long")
? ":" + d + "@" + JSON.stringify(depData)
+ ":" // no realpath resolved
+ ":MAXDEPTH"
: "")
} else {
depData = "─ "+d+"@'"+depData +"' (max depth reached)"
}
}
}
var c = i === depLast ? "└─" : "├─"
makePretty(depData, long, dir, childPref + c, list)
})
if (top && list.length === 1 && !data._id) {
if (!npm.config.get("parseable")) {
list.push("(empty)")
} else if (npm.config.get("long")) list[0] += ":EMPTY"
}
return list.filter(function (l) { return l && l.trim() })
}
function ugly (data) {
if (typeof data === "string") {
return data
}
if (!npm.config.get("long")) return data.path
return data.path
+ ":" + (data._id || "")
+ ":" + (data.realPath !== data.path ? data.realPath : "")
+ (data.extraneous ? ":EXTRANEOUS" : "")
+ (data.invalid ? ":INVALID" : "")
}
function format (data, long, prefix, dir) {
if (npm.config.get("parseable")) return ugly(data)
if (typeof data === "string") {
return prefix + data
}
// console.log([data.path, dir], "relativize")
var depLen = Object.keys(data.dependencies).length
, space = prefix.split("├─").join("│ ")
.split("└─").join(" ")
+ (depLen ? "" : " ")
, rel = relativize(data.path || "", dir)
, l = prefix
+ (rel === "." ? "" : depLen ? "┬ " : "─ ")
+ (data._id ? data._id + " " : "")
+ (data.link ? "-> " + data.link : "") + ""
+ (rel === "." && !(long && data._id) ? dir : "")
if (data.invalid) {
if (data.realName !== data.name) l += " ("+data.realName+")"
l += " \033[31;40minvalid\033[0m"
}
if (data.extraneous && rel !== ".") {
l += " \033[32;40mextraneous\033[0m"
}
if (!long || !data._id) return l
var extras = []
if (rel !== ".") extras.push(rel)
else extras.push(dir)
if (data.description) extras.push(data.description)
if (data.repository) extras.push(data.repository.url)
if (data.homepage) extras.push(data.homepage)
extras = extras.filter(function (e) { return e })
var lastExtra = !depLen && extras.length - 1
l += extras.map(function (e, i) {
var indent = !depLen ? " " : "│ "
return "\n" + space + indent + e
}).join("")
return l
}