verdaccio/lib/storage.js

557 lines
14 KiB
JavaScript
Raw Normal View History

2013-10-26 14:18:36 +02:00
var async = require('async')
, assert = require('assert')
2013-10-26 14:18:36 +02:00
, UError = require('./error').UserError
, Local = require('./local-storage')
, Proxy = require('./up-storage')
, mystreams = require('./streams')
, utils = require('./utils')
2013-12-07 23:55:50 +01:00
, Logger = require('./logger')
2014-05-07 17:10:59 +02:00
, localList = require('./local-list');
2013-05-31 08:26:11 +02:00
2013-09-25 11:12:33 +02:00
//
// Implements Storage interface
// (same for storage.js, local-storage.js, up-storage.js)
//
2013-06-08 03:16:28 +02:00
function Storage(config) {
2013-10-26 14:18:36 +02:00
if (!(this instanceof Storage)) return new Storage(config)
2013-06-08 03:16:28 +02:00
2013-10-26 14:18:36 +02:00
this.config = config
2013-09-25 11:12:33 +02:00
// we support a number of uplinks, but only one local storage
// Proxy and Local classes should have similar API interfaces
2013-10-26 14:18:36 +02:00
this.uplinks = {}
2013-06-08 03:16:28 +02:00
for (var p in config.uplinks) {
2013-10-26 14:18:36 +02:00
this.uplinks[p] = new Proxy(config.uplinks[p], config)
this.uplinks[p].upname = p
2013-06-01 00:57:28 +02:00
}
2013-10-26 14:18:36 +02:00
this.local = new Local(config)
2013-12-07 23:55:50 +01:00
this.logger = Logger.logger.child()
2013-06-08 03:16:28 +02:00
2013-10-26 14:18:36 +02:00
return this
2013-05-31 08:26:11 +02:00
}
2013-09-25 11:12:33 +02:00
//
// Add a {name} package to a system
//
// Function checks if package with the same name is available from uplinks.
// If it isn't, we create package locally
//
// Used storages: local (write) && uplinks
2013-09-25 11:12:33 +02:00
//
2013-06-08 03:16:28 +02:00
Storage.prototype.add_package = function(name, metadata, callback) {
2013-10-26 14:18:36 +02:00
var self = this
2013-11-24 18:07:18 +01:00
// NOTE:
// - when we checking package for existance, we ask ALL uplinks
// - when we publishing package, we only publish it to some of them
// so all requests are necessary
2013-06-14 10:34:29 +02:00
check_package_local(function(err) {
2013-11-24 18:07:18 +01:00
if (err) return callback(err)
check_package_remote(function(err) {
2013-11-24 18:07:18 +01:00
if (err) return callback(err)
publish_package(function(err) {
if (err) return callback(err)
callback()
})
2013-10-26 14:18:36 +02:00
})
2013-11-24 18:07:18 +01:00
})
function check_package_local(cb) {
self.local.get_package(name, {}, function(err, results) {
if (err && err.status !== 404) return cb(err)
if (results) {
return cb(new UError({
status: 409,
message: 'this package is already present'
}))
}
cb()
})
}
function check_package_remote(cb) {
self._sync_package_with_uplinks(name, null, {}, function(err, results, err_results) {
2013-11-24 18:07:18 +01:00
// something weird
if (err && err.status !== 404) return cb(err)
// checking package
if (results) {
return cb(new UError({
status: 409,
message: 'this package is already present'
}))
}
2013-11-24 18:07:18 +01:00
for (var i=0; i<err_results.length; i++) {
// checking error
// if uplink fails with a status other than 404, we report failure
if (err_results[i][0] != null) {
if (err_results[i][0].status !== 404) {
return cb(new UError({
status: 503,
message: 'one of the uplinks is down, refuse to publish'
2013-11-24 18:07:18 +01:00
}))
}
2013-06-14 10:34:29 +02:00
}
}
2013-11-24 18:07:18 +01:00
return cb()
})
}
function publish_package(cb) {
self.local.add_package(name, metadata, callback)
2013-11-24 18:07:18 +01:00
}
2013-06-01 00:57:28 +02:00
}
2013-09-25 11:12:33 +02:00
//
// Add a new version of package {name} to a system
//
// Used storages: local (write)
2013-09-25 11:12:33 +02:00
//
2013-06-08 03:16:28 +02:00
Storage.prototype.add_version = function(name, version, metadata, tag, callback) {
return this.local.add_version(name, version, metadata, tag, callback)
2013-06-08 03:16:28 +02:00
}
2013-06-01 00:57:28 +02:00
2013-12-27 14:06:30 +01:00
//
// Tags a package version with a provided tag
2013-12-27 14:06:30 +01:00
//
// Used storages: local (write)
2013-12-27 14:06:30 +01:00
//
Storage.prototype.add_tags = function(name, tag_hash, callback) {
return this.local.add_tags(name, tag_hash, callback)
2013-12-27 14:06:30 +01:00
}
//
// Change an existing package (i.e. unpublish one version)
//
// Function changes a package info from local storage and all uplinks with
// write access.
//
// Used storages: local (write)
//
Storage.prototype.change_package = function(name, metadata, revision, callback) {
return this.local.change_package(name, metadata, revision, callback)
}
//
// Remove a package from a system
//
// Function removes a package from local storage
//
// Used storages: local (write)
//
Storage.prototype.remove_package = function(name, callback) {
return this.local.remove_package(name, callback)
}
//
// Remove a tarball from a system
//
// Function removes a tarball from local storage.
// Tarball in question should not be linked to in any existing
// versions, i.e. package version should be unpublished first.
//
// Used storages: local (write)
//
Storage.prototype.remove_tarball = function(name, filename, revision, callback) {
return this.local.remove_tarball(name, filename, revision, callback)
}
2013-09-25 11:12:33 +02:00
//
2013-09-28 14:19:40 +02:00
// Upload a tarball for {name} package
2013-09-25 11:12:33 +02:00
//
// Function is syncronous and returns a WritableStream
//
// Used storages: local (write)
2013-09-25 11:12:33 +02:00
//
2013-06-20 15:07:34 +02:00
Storage.prototype.add_tarball = function(name, filename) {
return this.local.add_tarball(name, filename)
2013-06-01 00:57:28 +02:00
}
2014-05-07 21:28:10 +02:00
Storage.prototype.get_readme = function(name, version, callback) {
return this.local.get_readme(name, version, callback);
};
2013-09-25 11:12:33 +02:00
//
// Get a tarball from a storage for {name} package
//
// Function is syncronous and returns a ReadableStream
//
2013-09-27 02:26:15 +02:00
// Function tries to read tarball locally, if it fails then it reads package
// information in order to figure out where we can get this tarball from
//
// Used storages: local || uplink (just one)
2013-09-25 11:12:33 +02:00
//
Storage.prototype.get_tarball = function(name, filename) {
2013-10-26 14:18:36 +02:00
var stream = new mystreams.ReadTarballStream()
stream.abort = function() {}
var self = this
2013-06-19 18:58:16 +02:00
2013-06-18 20:14:55 +02: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-10-26 14:18:36 +02:00
var rstream = self.local.get_tarball(name, filename)
var is_open = false
2013-06-20 15:07:34 +02:00
rstream.on('error', function(err) {
if (is_open || err.status !== 404) {
2013-10-26 14:18:36 +02:00
return stream.emit('error', err)
2013-06-20 15:07:34 +02:00
}
2013-06-20 15:07:34 +02:00
// local reported 404
2013-10-26 14:18:36 +02:00
var err404 = err
var uplink = null
rstream.abort()
rstream = null // gc
2013-06-18 20:14:55 +02:00
2013-06-19 18:58:16 +02:00
self.local.get_package(name, function(err, info) {
if (!err && info._distfiles && info._distfiles[filename] != null) {
// information about this file exists locally
serve_file(info._distfiles[filename])
2013-06-18 20:14:55 +02:00
} else {
// we know nothing about this file, trying to get information elsewhere
2013-06-19 18:58:16 +02:00
self._sync_package_with_uplinks(name, info, {}, function(err, info) {
if (err) return stream.emit('error', err)
2013-06-19 18:58:16 +02:00
if (!info._distfiles || info._distfiles[filename] == null) {
return stream.emit('error', err404)
}
serve_file(info._distfiles[filename])
})
}
2013-10-26 14:18:36 +02:00
})
})
rstream.on('content-length', function(v) {
stream.emit('content-length', v)
})
2013-06-20 15:07:34 +02:00
rstream.on('open', function() {
2013-10-26 14:18:36 +02:00
is_open = true
rstream.pipe(stream)
})
return stream
function serve_file(file) {
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({
url: file.url,
_autogenerated: true,
}, self.config)
}
var savestream = self.local.add_tarball(name, filename)
function on_open() {
var rstream2 = uplink.get_url(file.url)
rstream2.on('error', function(err) {
if (savestream) savestream.abort()
savestream = null
stream.emit('error', err)
})
rstream2.on('end', function() {
if (savestream) savestream.done()
})
rstream2.on('content-length', function(v) {
stream.emit('content-length', v)
if (savestream) savestream.emit('content-length', v)
})
rstream2.pipe(stream)
if (savestream) rstream2.pipe(savestream)
}
savestream.on('open', function() {
on_open()
})
savestream.on('error', function() {
if (savestream) savestream.abort()
savestream = null
on_open()
})
}
2013-06-01 00:57:28 +02:00
}
2013-09-25 11:12:33 +02:00
//
// Retrieve a package metadata for {name} package
//
// Function invokes local.get_package and uplink.get_package for every
// uplink with proxy_access rights against {name} and combines results
// into one json object
//
// Used storages: local && uplink (proxy_access)
//
2013-12-09 04:58:25 +01:00
Storage.prototype.get_package = function(name, options, callback) {
if (typeof(options) === 'function') callback = options, options = {}
2013-10-22 11:31:48 +02:00
var self = this
2013-06-08 03:16:28 +02:00
2013-12-09 04:58:25 +01:00
self.local.get_package(name, options, function(err, data) {
2013-10-22 11:31:48 +02:00
if (err && (!err.status || err.status >= 500)) {
// report internal errors right away
2013-12-23 01:14:57 +01:00
return callback(err)
2013-10-22 11:31:48 +02:00
}
2013-05-31 08:26:11 +02:00
self._sync_package_with_uplinks(name, data, options, function(err, result, uplink_errors) {
if (err) return callback(err)
var whitelist = ['_rev', 'name', 'versions', 'dist-tags']
for (var i in result) {
2013-12-23 01:14:57 +01:00
if (whitelist.indexOf(i) === -1) delete result[i]
}
2013-06-08 03:16:28 +02:00
if (self.config.ignore_latest_tag || !result['dist-tags'].latest) {
result['dist-tags'].latest = utils.semver_sort(Object.keys(result.versions))
}
for (var i in result['dist-tags']) {
if (Array.isArray(result['dist-tags'][i])) {
result['dist-tags'][i] = result['dist-tags'][i][result['dist-tags'][i].length-1]
if (result['dist-tags'][i] == null) delete result['dist-tags'][i]
}
}
2013-12-27 14:06:30 +01:00
// npm can throw if this field doesn't exist
result._attachments = {}
callback(null, result, uplink_errors)
})
})
}
2014-04-12 19:20:26 +02:00
//
// Retrieve remote and local packages more recent than {startkey}
//
// Function invokes uplink.request for npm and local.get_recent_packages for
// local ones then sum up the result in a json object
//
// Used storages: local && uplink (proxy_access)
//
Storage.prototype.search = function(startkey, options, callback) {
var self = this
var uplinks = []
var i = 0
var uplinks
for (var p in self.uplinks) {
uplinks.push(p)
}
function merge_with_local_packages(err, res, body) {
2014-06-26 18:21:23 +02:00
if (err) return callback(err)
2014-04-12 19:20:26 +02:00
var j = 0
self.local.get_recent_packages(startkey, function(err, list) {
if (err) return callback(err)
var listL = list.length
if (!listL) return callback(null, body)
list.forEach(function(item) {
self.local.get_package(item.name, options, function(err, data) {
if (err) return callback(err)
var versions = utils.semver_sort(Object.keys(data.versions))
var latest = versions[versions.length - 1]
if (data.versions[latest]) {
body[item.name] = {
name : data.versions[latest].name,
description : data.versions[latest].description,
'dist-tags' : {
latest: latest
},
2014-08-01 22:08:44 +02:00
maintainers : data.versions[latest].maintainers || [data.versions[latest]._npmUser].filter(Boolean),
2014-04-12 19:20:26 +02:00
readmeFilename: data.versions[latest].readmeFilename || '',
time : {
modified: new Date(item.time).toISOString()
},
versions : {},
repository : data.versions[latest].repository,
keywords : data.versions[latest].keywords
}
body[item.name].versions[latest] = 'latest'
}
if (++j !== listL) {
return false
}
return callback(null, body)
})
})
})
}
function remote_search() {
var uplink = self.uplinks[uplinks[i]]
if (!uplink) {
return merge_with_local_packages(null, null, {})
}
self.uplinks[uplinks[i]].request({
uri: options.req.url,
timeout: self.uplinks[p].timeout,
json: true
}, function(err, res, body) {
if (err || Math.floor(res.statusCode / 100) > 3) {
i++
return remote_search()
}
return merge_with_local_packages(err, res, body)
})
}
remote_search()
}
2014-05-07 17:10:59 +02:00
Storage.prototype.get_local = function(callback) {
var self = this
, locals = localList.get()
, packages = [];
var getPackage = function(i) {
2014-08-08 04:17:05 +02:00
self.local.get_package(locals[i], function(err, info) {
2014-08-08 03:58:25 +02:00
if (!err) {
var latest = info['dist-tags'].latest;
2014-09-02 02:26:54 +02:00
if (info.versions[latest]) packages.push(info.versions[latest]);
2014-05-07 17:10:59 +02:00
}
2014-08-08 03:58:25 +02:00
if (i >= locals.length - 1) {
callback(null, packages);
} else {
2014-05-07 17:10:59 +02:00
getPackage(i + 1);
}
});
};
2014-05-07 22:31:25 +02:00
if(locals.length) {
getPackage(0);
}
else {
callback(null, []);
}
2014-05-07 17:10:59 +02:00
};
// function fetches package information from uplinks and synchronizes it with local data
// if package is available locally, it MUST be provided in pkginfo
// returns callback(err, result, uplink_errors)
Storage.prototype._sync_package_with_uplinks = function(name, pkginfo, options, callback) {
var self = this
if (!pkginfo) {
var exists = false
pkginfo = {
2013-10-22 11:31:48 +02:00
name: name,
versions: {},
'dist-tags': {},
_uplinks: {},
}
} else {
var exists = true
}
2013-06-14 10:34:29 +02:00
var uplinks = []
for (var i in self.uplinks) {
if (self.config.proxy_access(name, i)) {
uplinks.push(self.uplinks[i])
}
}
2013-10-22 11:31:48 +02:00
async.map(uplinks, function(up, cb) {
var _options = Object.create(options)
2014-03-02 10:47:09 +01:00
if (utils.is_object(pkginfo._uplinks[up.upname])) {
var fetched = pkginfo._uplinks[up.upname].fetched
if (fetched && fetched > (Date.now() - up.maxage)) {
return cb()
}
_options.etag = pkginfo._uplinks[up.upname].etag
2014-03-02 10:47:09 +01:00
}
up.get_package(name, _options, function(err, up_res, etag) {
if (err && err.status === 304)
2014-03-02 10:47:09 +01:00
pkginfo._uplinks[up.upname].fetched = Date.now()
if (err || !up_res) return cb(null, [err || new Error('no data')])
try {
utils.validate_metadata(up_res, name)
} catch(err) {
self.logger.error({
sub: 'out',
err: err,
}, 'package.json validating error @{!err.message}\n@{err.stack}')
return cb(null, [err])
}
2013-10-22 11:31:48 +02:00
pkginfo._uplinks[up.upname] = {
2014-03-02 10:47:09 +01:00
etag: etag,
fetched: Date.now()
2013-10-22 11:31:48 +02:00
}
try {
Storage._merge_versions(pkginfo, up_res, self.config)
} catch(err) {
self.logger.error({
sub: 'out',
err: err,
}, 'package.json parsing error @{!err.message}\n@{err.stack}')
return cb(null, [err])
}
2013-10-22 11:31:48 +02:00
// if we got to this point, assume that the correct package exists
// on the uplink
exists = true
cb()
})
}, function(err, uplink_errors) {
assert(!err && Array.isArray(uplink_errors))
2013-12-07 23:55:50 +01:00
if (!exists) {
return callback(new UError({
status: 404,
message: 'no such package available'
}), null, uplink_errors)
}
2013-12-07 23:55:50 +01:00
self.local.update_versions(name, pkginfo, function(err, pkginfo) {
if (err) return callback(err)
return callback(null, pkginfo, uplink_errors)
2013-10-22 11:31:48 +02:00
})
})
2013-05-31 08:26:11 +02:00
}
2013-12-07 23:55:50 +01:00
// function gets a local info and an info from uplinks and tries to merge it
// exported for unit tests only
Storage._merge_versions = function(local, up, config) {
2013-12-07 23:55:50 +01:00
// copy new versions to a cache
// NOTE: if a certain version was updated, we can't refresh it reliably
for (var i in up.versions) {
if (local.versions[i] == null) {
local.versions[i] = up.versions[i]
}
}
// refresh dist-tags
for (var i in up['dist-tags']) {
utils.tag_version(local, up['dist-tags'][i], i, config || {})
2013-12-07 23:55:50 +01:00
}
}
2013-10-26 14:18:36 +02:00
module.exports = Storage
2013-06-08 03:16:28 +02:00