2013-06-07 20:16:28 -05:00
|
|
|
var async = require('async');
|
|
|
|
var semver = require('semver');
|
2013-06-20 08:07:34 -05:00
|
|
|
var through = require('through');
|
2013-05-31 17:57:28 -05:00
|
|
|
var UError = require('./error').UserError;
|
2013-06-13 09:21:14 -05:00
|
|
|
var Local = require('./st-local');
|
2013-06-07 20:16:28 -05:00
|
|
|
var Proxy = require('./st-proxy');
|
|
|
|
var utils = require('./utils');
|
2013-05-31 01:26:11 -05:00
|
|
|
|
2013-06-07 20:16:28 -05:00
|
|
|
function Storage(config) {
|
|
|
|
if (!(this instanceof Storage)) return new Storage(config);
|
|
|
|
|
|
|
|
this.config = config;
|
|
|
|
this.uplinks = {};
|
|
|
|
for (var p in config.uplinks) {
|
2013-06-19 11:58:16 -05:00
|
|
|
this.uplinks[p] = new Proxy(config.uplinks[p], config);
|
2013-05-31 17:57:28 -05:00
|
|
|
}
|
2013-06-13 09:21:14 -05:00
|
|
|
this.local = new Local(config);
|
2013-06-07 20:16:28 -05:00
|
|
|
|
|
|
|
return this;
|
2013-05-31 01:26:11 -05:00
|
|
|
}
|
|
|
|
|
2013-06-07 20:16:28 -05:00
|
|
|
Storage.prototype.add_package = function(name, metadata, callback) {
|
2013-06-14 03:34:29 -05:00
|
|
|
var self = this;
|
|
|
|
|
|
|
|
var uplinks = [];
|
|
|
|
for (var i in this.uplinks) {
|
|
|
|
if (this.config.allow_proxy(name, i)) {
|
|
|
|
uplinks.push(this.uplinks[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async.map(uplinks, function(up, cb) {
|
|
|
|
up.get_package(name, function(err, res) {
|
|
|
|
cb(null, [err, res]);
|
|
|
|
});
|
|
|
|
}, function(err, results) {
|
|
|
|
for (var i=0; i<results.length; i++) {
|
|
|
|
// checking error
|
|
|
|
// if uplink fails with a status other than 404, we report failure
|
|
|
|
if (results[i][0] != null) {
|
|
|
|
if (results[i][0].status !== 404) {
|
|
|
|
return callback(new UError({
|
|
|
|
status: 503,
|
|
|
|
msg: 'one of the uplinks is down, refuse to publish'
|
|
|
|
}));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// checking package
|
|
|
|
if (results[i][1] != null) {
|
|
|
|
return callback(new UError({
|
|
|
|
status: 409,
|
|
|
|
msg: 'this package is already present'
|
|
|
|
}));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
self.local.add_package(name, metadata, callback);
|
|
|
|
});
|
2013-05-31 17:57:28 -05:00
|
|
|
}
|
|
|
|
|
2013-06-07 20:16:28 -05:00
|
|
|
Storage.prototype.add_version = function(name, version, metadata, tag, callback) {
|
2013-06-13 09:21:14 -05:00
|
|
|
this.local.add_version(name, version, metadata, tag, callback);
|
2013-06-07 20:16:28 -05:00
|
|
|
}
|
2013-05-31 17:57:28 -05:00
|
|
|
|
2013-06-20 08:07:34 -05:00
|
|
|
Storage.prototype.add_tarball = function(name, filename) {
|
|
|
|
return this.local.add_tarball(name, filename);
|
2013-05-31 17:57:28 -05:00
|
|
|
}
|
|
|
|
|
2013-06-07 20:16:28 -05:00
|
|
|
Storage.prototype.get_tarball = function(name, filename, callback) {
|
2013-06-20 08:07:34 -05:00
|
|
|
var stream = through(function(data) {
|
|
|
|
this.queue(data);
|
|
|
|
}, function() {
|
|
|
|
this.queue(null);
|
|
|
|
});
|
|
|
|
|
2013-06-19 11:58:16 -05:00
|
|
|
var self = this;
|
|
|
|
|
2013-06-18 13:14:55 -05:00
|
|
|
// if someone requesting tarball, it means that we should already have some
|
|
|
|
// information about it, so fetching package info is unnecessary
|
|
|
|
|
|
|
|
// trying local first
|
2013-06-20 08:07:34 -05:00
|
|
|
var rstream = self.local.get_tarball(name, filename);
|
|
|
|
var is_open = false;
|
|
|
|
rstream.on('error', function(err) {
|
|
|
|
if (is_open || err.status !== 404) {
|
|
|
|
return stream.emit('error', err);
|
|
|
|
}
|
|
|
|
|
|
|
|
// local reported 404
|
2013-06-19 11:58:16 -05:00
|
|
|
var err404 = err;
|
2013-06-20 08:07:34 -05:00
|
|
|
var uplink = null;
|
|
|
|
rstream.destroy();
|
2013-06-18 13:14:55 -05:00
|
|
|
|
2013-06-19 11:58:16 -05:00
|
|
|
self.local.get_package(name, function(err, info) {
|
2013-06-20 08:07:34 -05:00
|
|
|
if (err) return stream.emit('error', err);
|
2013-06-18 13:14:55 -05:00
|
|
|
|
2013-06-19 11:58:16 -05:00
|
|
|
if (info._distfiles[filename] == null) {
|
2013-06-20 08:07:34 -05:00
|
|
|
return stream.emit('error', err404);
|
2013-06-19 11:58:16 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
var file = info._distfiles[filename];
|
|
|
|
var uplink = null;
|
|
|
|
for (var p in self.uplinks) {
|
|
|
|
if (self.uplinks[p].can_fetch_url(file.url)) {
|
|
|
|
uplink = self.uplinks[p];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (uplink == null) {
|
|
|
|
uplink = new Proxy({
|
2013-06-20 08:41:07 -05:00
|
|
|
url: file.url,
|
2013-06-19 11:58:16 -05:00
|
|
|
_autogenerated: true,
|
|
|
|
}, self.config);
|
|
|
|
}
|
|
|
|
|
2013-06-20 08:41:07 -05:00
|
|
|
var rstream2 = uplink.get_url(file.url);
|
2013-06-20 08:07:34 -05:00
|
|
|
rstream2.on('error', function(err) {
|
|
|
|
stream.emit('error', err);
|
|
|
|
});
|
|
|
|
rstream2.on('data', function(data) {
|
|
|
|
stream.write(data);
|
|
|
|
});
|
|
|
|
rstream2.on('end', function() {
|
|
|
|
stream.end();
|
2013-06-19 11:58:16 -05:00
|
|
|
});
|
|
|
|
});
|
2013-06-18 13:14:55 -05:00
|
|
|
});
|
2013-06-20 08:07:34 -05:00
|
|
|
rstream.on('open', function() {
|
|
|
|
is_open = true;
|
|
|
|
rstream.pipe(stream);
|
|
|
|
});
|
|
|
|
return stream;
|
2013-05-31 17:57:28 -05:00
|
|
|
}
|
|
|
|
|
2013-06-07 20:16:28 -05:00
|
|
|
Storage.prototype.get_package = function(name, callback) {
|
2013-06-18 13:14:55 -05:00
|
|
|
var self = this;
|
2013-06-13 09:21:14 -05:00
|
|
|
var uplinks = [this.local];
|
2013-06-07 20:16:28 -05:00
|
|
|
for (var i in this.uplinks) {
|
|
|
|
if (this.config.allow_proxy(name, i)) {
|
|
|
|
uplinks.push(this.uplinks[i]);
|
2013-05-31 17:57:28 -05:00
|
|
|
}
|
2013-06-07 20:16:28 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
var result = {
|
|
|
|
name: name,
|
|
|
|
versions: {},
|
|
|
|
'dist-tags': {},
|
|
|
|
};
|
2013-06-14 03:34:29 -05:00
|
|
|
var exists = false;
|
2013-06-07 20:16:28 -05:00
|
|
|
var latest;
|
2013-05-31 01:26:11 -05:00
|
|
|
|
2013-06-07 20:16:28 -05:00
|
|
|
async.map(uplinks, function(up, cb) {
|
|
|
|
up.get_package(name, function(err, up_res) {
|
|
|
|
if (err) return cb();
|
|
|
|
|
2013-06-20 08:07:34 -05:00
|
|
|
if (up === self.local) {
|
|
|
|
// file exists in local repo
|
|
|
|
exists = true;
|
|
|
|
}
|
|
|
|
|
2013-06-07 20:16:28 -05:00
|
|
|
try {
|
|
|
|
utils.validate_metadata(up_res, name);
|
|
|
|
} catch(err) {
|
|
|
|
return cb();
|
|
|
|
}
|
|
|
|
|
2013-06-14 02:56:02 -05:00
|
|
|
var this_version = up_res['dist-tags'].latest;
|
2013-06-20 08:07:34 -05:00
|
|
|
if (latest == null
|
|
|
|
|| (!semver.gt(latest, this_version) && this_version)) {
|
2013-06-14 02:56:02 -05:00
|
|
|
latest = this_version;
|
|
|
|
var is_latest = true;
|
|
|
|
}
|
|
|
|
|
2013-06-07 20:16:28 -05:00
|
|
|
['versions', 'dist-tags'].forEach(function(key) {
|
|
|
|
for (var i in up_res[key]) {
|
|
|
|
if (!result[key][i] || is_latest) {
|
|
|
|
result[key][i] = up_res[key][i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
2013-06-14 03:34:29 -05:00
|
|
|
|
|
|
|
// if we got to this point, assume that the correct package exists
|
|
|
|
// on the uplink
|
|
|
|
exists = true;
|
2013-06-07 20:16:28 -05:00
|
|
|
cb();
|
|
|
|
});
|
|
|
|
}, function(err) {
|
|
|
|
if (err) return callback(err);
|
2013-06-14 03:34:29 -05:00
|
|
|
if (!exists) {
|
2013-05-31 17:57:28 -05:00
|
|
|
return callback(new UError({
|
|
|
|
status: 404,
|
|
|
|
msg: 'no such package available'
|
|
|
|
}));
|
|
|
|
}
|
2013-06-07 20:16:28 -05:00
|
|
|
callback(null, result);
|
2013-06-18 13:14:55 -05:00
|
|
|
|
|
|
|
self.local.update_versions(name, result, function(){});
|
2013-05-31 17:57:28 -05:00
|
|
|
});
|
2013-05-31 01:26:11 -05:00
|
|
|
}
|
|
|
|
|
2013-06-07 20:16:28 -05:00
|
|
|
module.exports = Storage;
|
|
|
|
|