|
|
|
// emit JSON describing versions of all packages currently installed (for later
|
|
|
|
// use with shrinkwrap install)
|
|
|
|
|
|
|
|
module.exports = exports = shrinkwrap
|
|
|
|
|
|
|
|
var path = require('path')
|
|
|
|
var log = require('npmlog')
|
|
|
|
var writeFileAtomic = require('write-file-atomic')
|
|
|
|
var iferr = require('iferr')
|
|
|
|
var readPackageJson = require('read-package-json')
|
|
|
|
var readPackageTree = require('read-package-tree')
|
|
|
|
var validate = require('aproba')
|
|
|
|
var chain = require('slide').chain
|
|
|
|
var npm = require('./npm.js')
|
|
|
|
var recalculateMetadata = require('./install/deps.js').recalculateMetadata
|
|
|
|
var validatePeerDeps = require('./install/deps.js').validatePeerDeps
|
|
|
|
var isExtraneous = require('./install/is-extraneous.js')
|
|
|
|
var isOnlyDev = require('./install/is-dev.js').isOnlyDev
|
|
|
|
var packageId = require('./utils/package-id.js')
|
|
|
|
var moduleName = require('./utils/module-name.js')
|
|
|
|
var output = require('./utils/output.js')
|
|
|
|
var lifecycle = require('./utils/lifecycle.js')
|
|
|
|
|
|
|
|
shrinkwrap.usage = 'npm shrinkwrap'
|
|
|
|
|
|
|
|
function shrinkwrap (args, silent, cb) {
|
|
|
|
if (typeof cb !== 'function') {
|
|
|
|
cb = silent
|
|
|
|
silent = false
|
|
|
|
}
|
|
|
|
|
|
|
|
if (args.length) {
|
|
|
|
log.warn('shrinkwrap', "doesn't take positional args")
|
|
|
|
}
|
|
|
|
|
|
|
|
var dir = path.resolve(npm.dir, '..')
|
|
|
|
var packagePath = path.join(npm.localPrefix, 'package.json')
|
|
|
|
npm.config.set('production', true)
|
|
|
|
|
|
|
|
readPackageJson(packagePath, iferr(cb, function (data) {
|
|
|
|
lifecycle(data, 'preshrinkwrap', function () {
|
|
|
|
readPackageTree(dir, andRecalculateMetadata(iferr(cb, function (tree) {
|
|
|
|
var pkginfo = treeToShrinkwrap(tree, !!npm.config.get('dev') || /^dev(elopment)?$/.test(npm.config.get('also')))
|
|
|
|
|
|
|
|
chain([
|
|
|
|
[lifecycle, tree.package, 'shrinkwrap'],
|
|
|
|
[shrinkwrap_, pkginfo, silent],
|
|
|
|
[lifecycle, tree.package, 'postshrinkwrap']
|
|
|
|
], iferr(cb, function (data) {
|
|
|
|
cb(null, data[0])
|
|
|
|
}))
|
|
|
|
})))
|
|
|
|
})
|
|
|
|
}))
|
|
|
|
}
|
|
|
|
|
|
|
|
function andRecalculateMetadata (next) {
|
|
|
|
validate('F', arguments)
|
|
|
|
return function (er, tree) {
|
|
|
|
validate('EO', arguments)
|
|
|
|
if (er) return next(er)
|
|
|
|
recalculateMetadata(tree, log, next)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function treeToShrinkwrap (tree, dev) {
|
|
|
|
validate('OB', arguments)
|
|
|
|
var pkginfo = {}
|
|
|
|
if (tree.package.name) pkginfo.name = tree.package.name
|
|
|
|
if (tree.package.version) pkginfo.version = tree.package.version
|
|
|
|
var problems = []
|
|
|
|
if (tree.children.length) {
|
|
|
|
shrinkwrapDeps(dev, problems, pkginfo.dependencies = {}, tree)
|
|
|
|
}
|
|
|
|
if (problems.length) pkginfo.problems = problems
|
|
|
|
return pkginfo
|
|
|
|
}
|
|
|
|
|
|
|
|
function shrinkwrapDeps (dev, problems, deps, tree, seen) {
|
|
|
|
validate('BAOO', [dev, problems, deps, tree])
|
|
|
|
if (!seen) seen = {}
|
|
|
|
if (seen[tree.path]) return
|
|
|
|
seen[tree.path] = true
|
|
|
|
Object.keys(tree.missingDeps).forEach(function (name) {
|
|
|
|
var invalid = tree.children.filter(function (dep) { return moduleName(dep) === name })[0]
|
|
|
|
if (invalid) {
|
|
|
|
problems.push('invalid: have ' + invalid.package._id + ' (expected: ' + tree.missingDeps[name] + ') ' + invalid.path)
|
|
|
|
} else if (!tree.package.optionalDependencies || !tree.package.optionalDependencies[name]) {
|
|
|
|
var topname = packageId(tree)
|
|
|
|
problems.push('missing: ' + name + '@' + tree.package.dependencies[name] +
|
|
|
|
(topname ? ', required by ' + topname : ''))
|
|
|
|
}
|
|
|
|
})
|
|
|
|
tree.children.sort(function (aa, bb) { return moduleName(aa).localeCompare(moduleName(bb)) }).forEach(function (child) {
|
|
|
|
if (!dev && isOnlyDev(child)) {
|
|
|
|
log.warn('shrinkwrap', 'Excluding devDependency: %s', packageId(child), child.parent.package.dependencies)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
var pkginfo = deps[moduleName(child)] = {}
|
|
|
|
pkginfo.version = child.package.version
|
|
|
|
pkginfo.from = child.package._from
|
|
|
|
pkginfo.resolved = child.package._resolved
|
|
|
|
if (isExtraneous(child)) {
|
|
|
|
problems.push('extraneous: ' + child.package._id + ' ' + child.path)
|
|
|
|
}
|
|
|
|
validatePeerDeps(child, function (tree, pkgname, version) {
|
|
|
|
problems.push('peer invalid: ' + pkgname + '@' + version +
|
|
|
|
', required by ' + child.package._id)
|
|
|
|
})
|
|
|
|
if (child.children.length) {
|
|
|
|
shrinkwrapDeps(dev, problems, pkginfo.dependencies = {}, child, seen)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
function shrinkwrap_ (pkginfo, silent, cb) {
|
|
|
|
if (pkginfo.problems) {
|
|
|
|
return cb(new Error('Problems were encountered\n' +
|
|
|
|
'Please correct and try again.\n' +
|
|
|
|
pkginfo.problems.join('\n')))
|
|
|
|
}
|
|
|
|
|
|
|
|
save(pkginfo, silent, cb)
|
|
|
|
}
|
|
|
|
|
|
|
|
function save (pkginfo, silent, cb) {
|
|
|
|
// copy the keys over in a well defined order
|
|
|
|
// because javascript objects serialize arbitrarily
|
|
|
|
var swdata
|
|
|
|
try {
|
|
|
|
swdata = JSON.stringify(pkginfo, null, 2) + '\n'
|
|
|
|
} catch (er) {
|
|
|
|
log.error('shrinkwrap', 'Error converting package info to json')
|
|
|
|
return cb(er)
|
|
|
|
}
|
|
|
|
|
|
|
|
var file = path.resolve(npm.prefix, 'npm-shrinkwrap.json')
|
|
|
|
|
|
|
|
writeFileAtomic(file, swdata, function (er) {
|
|
|
|
if (er) return cb(er)
|
|
|
|
if (silent) return cb(null, pkginfo)
|
|
|
|
output('wrote npm-shrinkwrap.json')
|
|
|
|
cb(null, pkginfo)
|
|
|
|
})
|
|
|
|
}
|