node/deps/npm/lib/utils/fetch.js

94 lines
2.7 KiB
JavaScript
Raw Normal View History

/**
* Fetch an HTTP url to a local file.
**/
var request = require("request")
, fs = require("graceful-fs")
, npm = require("../npm.js")
, url = require("url")
2012-06-11 12:29:47 +08:00
, log = require("npmlog")
, path = require("path")
, mkdir = require("mkdirp")
, chownr = require("chownr")
, regHost
2013-03-21 08:49:57 +08:00
, once = require("once")
module.exports = fetch
function fetch (remote, local, headers, cb) {
if (typeof cb !== "function") cb = headers, headers = {}
2013-03-21 08:49:57 +08:00
cb = once(cb)
2012-06-11 12:29:47 +08:00
log.verbose("fetch", "to=", local)
mkdir(path.dirname(local), function (er, made) {
if (er) return cb(er)
fetch_(remote, local, headers, cb)
})
}
function fetch_ (remote, local, headers, cb) {
var fstr = fs.createWriteStream(local, { mode : npm.modes.file })
var response = null
2013-03-21 08:49:57 +08:00
fstr.on("error", function (er) {
2013-03-21 08:49:57 +08:00
cb(er)
2013-05-11 08:45:37 +08:00
fstr.destroy()
})
2013-03-21 08:49:57 +08:00
var req = makeRequest(remote, fstr, headers)
req.on("response", function (res) {
log.http(res.statusCode, remote)
response = res
response.resume()
// Work around bug in node v0.10.0 where the CryptoStream
// gets stuck and never starts reading again.
if (process.version === "v0.10.0") {
response.resume = function (orig) { return function() {
var ret = orig.apply(response, arguments)
if (response.socket.encrypted)
response.socket.encrypted.read(0)
return ret
}}(response.resume)
}
})
2013-03-21 08:49:57 +08:00
fstr.on("close", function () {
2013-03-21 08:49:57 +08:00
var er
if (response && response.statusCode && response.statusCode >= 400) {
2013-03-21 08:49:57 +08:00
er = new Error(response.statusCode + " "
+ require("http").STATUS_CODES[response.statusCode])
}
2013-03-21 08:49:57 +08:00
cb(er, response)
})
}
function makeRequest (remote, fstr, headers) {
remote = url.parse(remote)
2012-06-11 12:29:47 +08:00
log.http("GET", remote.href)
regHost = regHost || url.parse(npm.config.get("registry")).host
if (remote.host === regHost && npm.config.get("always-auth")) {
remote.auth = new Buffer( npm.config.get("_auth")
, "base64" ).toString("utf8")
if (!remote.auth) return fstr.emit("error", new Error(
"Auth required and none provided. Please run 'npm adduser'"))
}
2012-08-22 06:29:03 +08:00
var proxy
if (remote.protocol !== "https:" || !(proxy = npm.config.get("https-proxy"))) {
proxy = npm.config.get("proxy")
}
var opts = { url: remote
, proxy: proxy
, strictSSL: npm.config.get("strict-ssl")
2013-03-21 08:49:57 +08:00
, rejectUnauthorized: npm.config.get("strict-ssl")
, ca: remote.host === regHost ? npm.config.get("ca") : undefined
, headers: { "user-agent": npm.config.get("user-agent") }}
var req = request(opts)
req.on("error", function (er) {
fstr.emit("error", er)
})
req.pipe(fstr)
2013-03-21 08:49:57 +08:00
return req
}