Welcome to mirror list, hosted at ThFree Co, Russian Federation.

install.js « lib - github.com/npm/cli.git - Unnamed repository; edit this file 'description' to name the repository.
summaryrefslogtreecommitdiff
blob: 9cda3e26a179a6330499677f53a87089de50f7ce (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
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276

// npm install <pkg> <pkg> <pkg>
// npm install <pkg@version> <pkg@"1.0.0 - 1.99.99"> <pkg[@stable]> <pkg@tagname>

// 1. fetch the data for that package/tag
// 2. if it has any dependents, which are not yet installed,
// then add those to the list, and fetch their data.
// 3. when all the datas have been fetched, and we have a set
// of packages that are either installed or fetchable which
// will satisfy everyone's dependencies, then create the
// target root directories for each (so they're link-able)
// 4. download all the tarballs, and do npm install on each.

module.exports = registryInstall

var registry = require("./utils/registry")
  , npm = require("../npm")
  , readInstalled = require("./utils/read-installed")
  , installedPackages
  , semver = require("./utils/semver")
  , url = require("url")
  , fetch = require("./utils/fetch")
  , exec = require("./utils/exec")
  , rm = require("./utils/rm-rf")
  , mkdir = require("./utils/mkdir-p")
  , readJson = require("./utils/read-json")
  , log = require("./utils/log")
  , path = require("path")
  , chain = require("./utils/chain")
  , fs = require("fs")

function registryInstall (pkglist, cb) {
  // it's helpful to know what we have already
  if (!installedPackages) return readInstalled([], function (er, data) {
    if (er) return cb(er)
    installedPackages = data || {}
    registryInstall(pkglist, cb)
  })

  var defTag = npm.config.get("tag") || "stable"

  // While the list is not empty:
  // a. If it's a range, and a satisfying version is already installed,
  //    then move on to the next
  // b. If it's a tag version, then fetch the json, and add the specific
  //    version to the list.
  // c. If it's not already installed, then fetch its
  //    json, and find the satisfying version. Add
  //    dependencies and tarball url to the list.
  // d. If it's a url, fetch and unpack it to the appropriate pkg/version
  //    folder, add it to the "installed" list, add its deps to the list
  //    and move on

  // When the list is empty, build all the package folders that were created.
  var installList = []
    , seen = {}
    , installReg = {}
  ;(function F (pkg) {
    if (!pkg) {
      installList = installList.map(function (i) {
        return path.join(npm.dir, i, "package")
      })
      log(installList, "install")
      if (installList.length === 0) {
        return log("nothing to install", "install", cb)
      }
      return npm.commands.build
        ( installList
        , cb
        )
    }
    log(pkg, "install pkg")
    if (seen[pkg]) {
      log("seen it", "install")
      return F(pkglist.shift())
    }
    seen[pkg] = true

    function tarballHandler (er, data) {
      if (er) return cb(er)
      log(data, "fetched tarball")
      npm.set(data)
      installedPackages[data.name] = installedPackages[data.name] || {}
      installedPackages[data.name][data.version] = data
      installReg[data.name] = installReg[data.name] || {}
      installReg[data.name][data.version] = data
      installList.push(path.join(data.name, data.version))
      // also make sure to get any dependencies.
      if (data.dependencies) for (var dep in data.dependencies) {
        dep = dep.trim()+"@"+data.dependencies[dep]
        if (!seen[dep]) pkglist.push(dep)
      }
      F(pkglist.shift())
    }
    if (pkg.match(/^(file:|\.|\/)/)) {
      pkg = pkg.replace(/^file:(\/\/)?/, '')
      if (pkg.charAt(0) !== "/") pkg = path.join(process.cwd(), pkg)
      return installLocalTarball(pkg, tarballHandler)
    }
    if (pkg.match(/^https?:\/\//)) return fetchTarball(pkg, tarballHandler)

    // now we know it's not a URL, so handle it like a tag, version, or range.
    pkg = pkg.split("@")
    var name = pkg[0]
      , ver = pkg.slice(1).join("@").trim()
      , range = semver.validRange(ver)
      , exact = semver.valid(ver)
      , tag = (!exact && !range)
    pkg = pkg.join("@")
    // must fetch data to know how to solve this.
    var data = npm.get(name)
    if (!data) {
      log(name, "fetch data")
      return registry.get(name, function (er, data) {
        if (!data) er = new Error("not found in registry: "+name)
        if (er) return cb(er)
        npm.set(name, data)
        seen[pkg] = false
        return F(pkg)
      })
    }
    // now we know that we have the data.

    if (tag) {
      tag = ver || defTag
      if (!data["dist-tags"] || !(tag in data["dist-tags"])) {
        return cb(new Error(
          "Tag "+tag+" not found for package "+name))
      }
      data = data.versions[data["dist-tags"][tag]]
      npm.set(data)
      // add that version to the list, and its dependencies
      if (!seen[name+"@"+data.version]) pkglist.push(name+"@"+data.version)
      if (data.dependencies) for (var dep in data.dependencies) {
        dep = dep.trim()+"@"+data.dependencies[dep]
        if (!seen[dep]) pkglist.push(dep)
      }
      return F(pkglist.shift())
    }
    // now we know it's not a tag.  Either a real version, or a range,
    // and possibly already installed.
    if (exact) {
      if ((name in installedPackages) && (ver in installedPackages[name])) {
        // already installed.  Continue.
        return F(pkglist.shift())
      }
      var data = npm.get(name)
      // make sure this version exists.
      if (!(ver in data.versions)) {
        return cb(new Error(
          "Required version "+name+"@"+ver+" not found in registry"))
      }
      data = data.versions[ver]
      // get the tarball, and add the deps.
      var tarball = data.dist.tarball
      if (!tarball) return cb(new Error(
        "No tarball URL found for "+name+"@"+ver))
      if (!seen[tarball]) pkglist.push(tarball)
      if (data.dependencies) for (var dep in data.dependencies) {
        dep = dep.trim()+"@"+data.dependencies[dep]
        if (!seen[dep]) pkglist.push(dep)
      }
      return F(pkglist.shift())
    }
    // now we know it's a range.  get the data for this package, and then
    // look for a matching version, and add that to the list.
    // satisfying version can come from any packages we've seen so far,
    // or anything on the "to be installed" list
    // this way, we prefer things that are already here, rather than adding
    // unnecessarily
    var satis = semver.maxSatisfying
        ( Object.keys(installedPackages[name] || {})
          .concat(Object.keys(installReg[name] || {}))
        , range)
    if (satis) {
      satis = name+"@"+satis
      if (!seen[satis]) pkglist.push(satis)
      return F(pkglist.shift())
    }
    // new thing.  fetch from registry.  favor defTag version.
    var data = npm.get(name)
      , stable = data["dist-tags"] && data["dist-tags"][defTag]
    if (stable && semver.satisfies(stable, range)) {
      stable = name + "@" + stable
      if (!seen[stable]) pkglist.push(stable)
      return F(pkglist.shift())
    }
    var satis = semver.maxSatisfying(Object.keys(data.versions), range)
    if (!satis) return cb(new Error(
      "No satisfying version found for "+name+"@"+range))
    data = data[satis]
    satis = name+"@"+satis
    if (!seen[satis]) pkglist.push(satis)
    if (data.dependencies) for (var dep in data.dependencies) {
      dep = dep.trim()+"@"+data.dependencies[dep]
      if (!seen[dep]) pkglist.push(dep)
    }
    return F(pkglist.shift())
  })(pkglist.shift())
}

// download the tarball, and move the contents into 
// the appropriate name/version folder.
function fetchTarball (tarball, cb) {
  var target = path.join(npm.tmp, tarball.replace(/[^a-zA-Z0-9]/g, "-")+"-"+
                         Date.now()+"-"+Math.random()) + ".tgz"
  chain
    ( [mkdir, npm.tmp]
    , [fetch, tarball, target]
    , function (er, ok) {
        if (er) return cb(er)
        installLocalTarball(target, function (er, data) {
          if (er) return cb(er)
          rm(target, function (er) {
            if (er) return cb(er)
            cb(null, data)
          })
        })
      }
    )
}
function installLocalTarball (tarball, cb) {
  var folder = path.join(npm.tmp, tarball.replace(/[^a-zA-Z0-9]/g, "-")+"-"+
                         Date.now()+"-"+Math.random())
  chain
    ( [mkdir, folder]
    , [unpackTar, tarball, folder]
    , function (er) {
        if (er) return cb(er)
        readJson(path.join(folder, "package.json"), function (er,data) {
          chain
            ( [moveIntoPlace, folder, data]
            , [log, data._id, "moved into place"]
            , [rm, folder]
            , [log, folder, "removed"]
            , function (er) { return cb(er, data) }
            )
       })
     }
    )

}

function unpackTar (tarball, unpackTarget, cb) {
  exec("tar", ["xzvf", tarball, "--strip", "1", "-C", unpackTarget], cb)
}

// move to ROOT/.npm/{name}/{version}/package
function moveIntoPlace (dir, data, cb) {
  if (!data.name || !data.version) {
    return cb(new Error("Name or version not found in package info."))
  }
  var target = path.join(npm.dir, data.name, data.version)
  
  chain
    ( function (cb) {
        fs.lstat(target, function (e) {
          log((e?"remove":"creating") + " " +target, "moveIntoPlace")
          if (e) rm(target, function (er, ok) {
            if (er) {
              log("could not remove " + target, "moveIntoPlace")
              cb(new Error(target+" exists, and can't be removed"))
            } else {
              log("unlinked "+target,"moveIntoPlace")
              cb()
            }
          })
          else cb()
        })
      }
    , [mkdir, target]
    , [fs, "rename", dir, path.join(target, "package")]
    , [log, "done", "moveIntoPlace"]
    , cb
    )
}