123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197 |
- // link with no args: symlink the folder to the global location
- // link with package arg: symlink the global to the local
- var npm = require('./npm.js')
- var symlink = require('./utils/link.js')
- var fs = require('graceful-fs')
- var log = require('npmlog')
- var asyncMap = require('slide').asyncMap
- var chain = require('slide').chain
- var path = require('path')
- var build = require('./build.js')
- var npa = require('npm-package-arg')
- var usage = require('./utils/usage')
- var output = require('./utils/output.js')
- module.exports = link
- link.usage = usage(
- 'link',
- 'npm link (in package dir)' +
- '\nnpm link [<@scope>/]<pkg>[@<version>]'
- )
- link.completion = function (opts, cb) {
- var dir = npm.globalDir
- fs.readdir(dir, function (er, files) {
- cb(er, files.filter(function (f) {
- return !f.match(/^[._-]/)
- }))
- })
- }
- function link (args, cb) {
- if (process.platform === 'win32') {
- var semver = require('semver')
- if (!semver.gte(process.version, '0.7.9')) {
- var msg = 'npm link not supported on windows prior to node 0.7.9'
- var e = new Error(msg)
- e.code = 'ENOTSUP'
- e.errno = require('constants').ENOTSUP // eslint-disable-line node/no-deprecated-api
- return cb(e)
- }
- }
- if (npm.config.get('global')) {
- return cb(new Error(
- 'link should never be --global.\n' +
- 'Please re-run this command with --local'
- ))
- }
- if (args.length === 1 && args[0] === '.') args = []
- if (args.length) return linkInstall(args, cb)
- linkPkg(npm.prefix, cb)
- }
- function parentFolder (id, folder) {
- if (id[0] === '@') {
- return path.resolve(folder, '..', '..')
- } else {
- return path.resolve(folder, '..')
- }
- }
- function linkInstall (pkgs, cb) {
- asyncMap(pkgs, function (pkg, cb) {
- var t = path.resolve(npm.globalDir, '..')
- var pp = path.resolve(npm.globalDir, pkg)
- var rp = null
- var target = path.resolve(npm.dir, pkg)
- function n (er, data) {
- if (er) return cb(er, data)
- // we want the ONE thing that was installed into the global dir
- var installed = data.filter(function (info) {
- var id = info[0]
- var folder = info[1]
- return parentFolder(id, folder) === npm.globalDir
- })
- var id = installed[0][0]
- pp = installed[0][1]
- var what = npa(id)
- pkg = what.name
- target = path.resolve(npm.dir, pkg)
- next()
- }
- // if it's a folder, a random not-installed thing, or not a scoped package,
- // then link or install it first
- if (pkg[0] !== '@' && (pkg.indexOf('/') !== -1 || pkg.indexOf('\\') !== -1)) {
- return fs.lstat(path.resolve(pkg), function (er, st) {
- if (er || !st.isDirectory()) {
- npm.commands.install(t, pkg, n)
- } else {
- rp = path.resolve(pkg)
- linkPkg(rp, n)
- }
- })
- }
- fs.lstat(pp, function (er, st) {
- if (er) {
- rp = pp
- return npm.commands.install(t, [pkg], n)
- } else if (!st.isSymbolicLink()) {
- rp = pp
- next()
- } else {
- return fs.realpath(pp, function (er, real) {
- if (er) log.warn('invalid symbolic link', pkg)
- else rp = real
- next()
- })
- }
- })
- function next () {
- if (npm.config.get('dry-run')) return resultPrinter(pkg, pp, target, rp, cb)
- chain(
- [
- [ function (cb) {
- log.verbose('link', 'symlinking %s to %s', pp, target)
- cb()
- } ],
- [symlink, pp, target, false, false],
- // do not run any scripts
- rp && [build, [target], npm.config.get('global'), build._noLC, true],
- [resultPrinter, pkg, pp, target, rp]
- ],
- cb
- )
- }
- }, cb)
- }
- function linkPkg (folder, cb_) {
- var me = folder || npm.prefix
- var readJson = require('read-package-json')
- log.verbose('linkPkg', folder)
- readJson(path.resolve(me, 'package.json'), function (er, d) {
- function cb (er) {
- return cb_(er, [[d && d._id, target, null, null]])
- }
- if (er) return cb(er)
- if (!d.name) {
- er = new Error('Package must have a name field to be linked')
- return cb(er)
- }
- var target = path.resolve(npm.globalDir, d.name)
- if (npm.config.get('dry-run')) return resultPrinter(path.basename(me), me, target, cb)
- symlink(me, target, false, true, function (er) {
- if (er) return cb(er)
- log.verbose('link', 'build target', target)
- // also install missing dependencies.
- npm.commands.install(me, [], function (er) {
- if (er) return cb(er)
- // build the global stuff. Don't run *any* scripts, because
- // install command already will have done that.
- build([target], true, build._noLC, true, function (er) {
- if (er) return cb(er)
- resultPrinter(path.basename(me), me, target, cb)
- })
- })
- })
- })
- }
- function resultPrinter (pkg, src, dest, rp, cb) {
- if (typeof cb !== 'function') {
- cb = rp
- rp = null
- }
- var where = dest
- rp = (rp || '').trim()
- src = (src || '').trim()
- // XXX If --json is set, then look up the data from the package.json
- if (npm.config.get('parseable')) {
- return parseableOutput(dest, rp || src, cb)
- }
- if (rp === src) rp = null
- output(where + ' -> ' + src + (rp ? ' -> ' + rp : ''))
- cb()
- }
- function parseableOutput (dest, rp, cb) {
- // XXX this should match ls --parseable and install --parseable
- // look up the data from package.json, format it the same way.
- //
- // link is always effectively 'long', since it doesn't help much to
- // *just* print the target folder.
- // However, we don't actually ever read the version number, so
- // the second field is always blank.
- output(dest + '::' + rp)
- cb()
- }
|