summaryrefslogtreecommitdiff
path: root/deps/npm/lib/install/action/extract.js
blob: fd9562c1849c0a6b8d180b7379d7b16c3f65b43d (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
'use strict'
var path = require('path')
var iferr = require('iferr')
var asyncMap = require('slide').asyncMap
var fs = require('graceful-fs')
var mkdirp = require('mkdirp')
var move = require('../../utils/move.js')
var gentlyRm = require('../../utils/gently-rm.js')
var updatePackageJson = require('../update-package-json')
var npm = require('../../npm.js')
var moduleName = require('../../utils/module-name.js')
var packageId = require('../../utils/package-id.js')
var cache = require('../../cache.js')
var moduleStagingPath = require('../module-staging-path.js')
var readPackageJson = require('read-package-json')

module.exports = function (staging, pkg, log, next) {
  log.silly('extract', packageId(pkg))
  var up = npm.config.get('unsafe-perm')
  var user = up ? null : npm.config.get('user')
  var group = up ? null : npm.config.get('group')
  var extractTo = moduleStagingPath(staging, pkg)
  cache.unpack(pkg.package.name, pkg.package.version, extractTo, null, null, user, group,
      andUpdatePackageJson(pkg, staging, extractTo,
      andStageBundledChildren(pkg, staging, extractTo, log,
      andRemoveExtraneousBundles(extractTo, next))))
}

function andUpdatePackageJson (pkg, staging, extractTo, next) {
  return iferr(next, function () {
    readPackageJson(path.join(extractTo, 'package.json'), false, function (err, metadata) {
      if (!err) {
        // Copy _ keys (internal to npm) and any missing keys from the possibly incomplete
        // registry metadata over to the full package metadata read off of disk.
        Object.keys(pkg.package).forEach(function (key) {
          if (key[0] === '_' || !(key in metadata)) metadata[key] = pkg.package[key]
        })
        metadata.name = pkg.package.name // things go wrong if these don't match
        pkg.package = metadata
      }
      updatePackageJson(pkg, extractTo, next)
    })
  })
}

function andStageBundledChildren (pkg, staging, extractTo, log, next) {
  return iferr(next, function () {
    if (!pkg.package.bundleDependencies) return next()

    asyncMap(pkg.children, andStageBundledModule(pkg, staging, extractTo), next)
  })
}

function andRemoveExtraneousBundles (extractTo, next) {
  return iferr(next, function () {
    gentlyRm(path.join(extractTo, 'node_modules'), next)
  })
}

function andStageBundledModule (bundler, staging, parentPath) {
  return function (child, next) {
    if (child.error) return next(child.error)
    stageBundledModule(bundler, child, staging, parentPath, next)
  }
}

function getTree (pkg) {
  while (pkg.parent) pkg = pkg.parent
  return pkg
}

function warn (pkg, code, msg) {
  var tree = getTree(pkg)
  var err = new Error(msg)
  err.code = code
  tree.warnings.push(err)
}

function stageBundledModule (bundler, child, staging, parentPath, next) {
  var stageFrom = path.join(parentPath, 'node_modules', child.package.name)
  var stageTo = moduleStagingPath(staging, child)

  return asyncMap(child.children, andStageBundledModule(bundler, staging, stageFrom), iferr(next, finishModule))

  function finishModule () {
    // If we were the one's who bundled this module…
    if (child.fromBundle === bundler) {
      return moveModule()
    } else {
      return checkForReplacement()
    }
  }

  function moveModule () {
    return mkdirp(path.dirname(stageTo), iferr(next, function () {
      return move(stageFrom, stageTo, iferr(next, updateMovedPackageJson))
    }))
  }

  function checkForReplacement () {
    return fs.stat(stageFrom, function (notExists, exists) {
      if (exists) {
        warn(bundler, 'EBUNDLEOVERRIDE', 'In ' + packageId(bundler) +
          ' replacing bundled version of ' + moduleName(child) +
          ' with ' + packageId(child))
        return gentlyRm(stageFrom, next)
      } else {
        return next()
      }
    })
  }

  function updateMovedPackageJson () {
    updatePackageJson(child, stageTo, next)
  }
}