0
Fork 0
mirror of https://github.com/verdaccio/verdaccio.git synced 2025-01-06 22:40:26 -05:00
verdaccio/lib/st-local.js

186 lines
4.6 KiB
JavaScript
Raw Normal View History

2013-06-19 11:58:16 -05:00
var fs = require('fs');
2013-06-18 13:14:55 -05:00
var semver = require('semver');
2013-06-14 03:34:29 -05:00
var fs_storage = require('./fs-storage');
2013-06-07 20:16:28 -05:00
var UError = require('./error').UserError;
2013-06-19 11:58:16 -05:00
var utils = require('./utils');
2013-06-07 20:16:28 -05:00
var info_file = 'package.json';
2013-06-13 09:21:14 -05:00
function Storage(config) {
if (!(this instanceof Storage)) return new Storage(config);
this.config = config;
2013-06-14 03:34:29 -05:00
this.storage = new fs_storage(this.config.storage);
2013-06-13 09:21:14 -05:00
return this;
}
2013-06-18 13:14:55 -05:00
// returns the minimal package file
function get_boilerplate(name) {
return {
2013-06-19 11:58:16 -05:00
// standard things
2013-06-18 13:14:55 -05:00
name: name,
versions: {},
'dist-tags': {},
2013-06-19 11:58:16 -05:00
// our own object
// type: "filename"->"metadata"
'_distfiles': {},
2013-06-18 13:14:55 -05:00
};
}
2013-06-13 09:21:14 -05:00
Storage.prototype.add_package = function(name, metadata, callback) {
2013-06-18 13:14:55 -05:00
this.storage.create_json(name + '/' + info_file, get_boilerplace(name), function(err) {
2013-06-07 20:16:28 -05:00
if (err && err.code === 'EEXISTS') {
return callback(new UError({
status: 409,
msg: 'this package is already present'
}));
}
callback();
});
}
2013-06-18 13:14:55 -05:00
Storage.prototype._read_create_package = function(name, callback) {
2013-06-14 03:34:29 -05:00
var self = this;
self.storage.read_json(name + '/' + info_file, function(err, data) {
2013-06-07 20:16:28 -05:00
// TODO: race condition
2013-06-18 13:14:55 -05:00
if (err) {
if (err.code === 'ENOENT') {
// if package doesn't exist, we create it here
data = get_boilerplate(name);
} else {
return callback(err);
}
}
callback(null, data);
});
}
// synchronize remote package info with the local one
// TODO: readfile called twice
Storage.prototype.update_versions = function(name, newdata, callback) {
var self = this;
self._read_create_package(name, function(err, data) {
if (err) return callback(err);
var change = false;
for (var ver in newdata.versions) {
if (data.versions[ver] == null) {
2013-06-19 11:58:16 -05:00
var verdata = newdata.versions[ver];
// why does anyone need to keep that in database?
delete verdata.readme;
2013-06-18 13:14:55 -05:00
change = true;
2013-06-19 11:58:16 -05:00
data.versions[ver] = verdata;
if (verdata.dist && verdata.dist.tarball) {
var url = utils.parse_tarball_url(
verdata.dist.__sinopia_orig_tarball || verdata.dist.tarball
);
// we do NOT overwrite any existing records
if (url != null && data._distfiles[url.filename] == null) {
data._distfiles[url.filename] = {
url: verdata.dist.tarball,
prefix: url.protocol + '//' + url.host + url.prepath,
sha: verdata.dist.shasum,
};
}
}
2013-06-18 13:14:55 -05:00
}
}
for (var tag in newdata['dist-tags']) {
// if tag is updated to reference latter version, that's fine
var need_change =
(data['dist-tags'][tag] == null) ||
(!semver.gt(newdata['dist-tags'], data['dist-tags'][tag]));
if (need_change) {
change = true;
data['dist-tags'][tag] = newdata['dist-tags'][tag];
}
}
if (change) {
self.storage.write_json(name + '/' + info_file, data, callback);
} else {
callback();
}
});
}
Storage.prototype.add_version = function(name, version, metadata, tag, callback) {
var self = this;
self._read_create_package(name, function(err, data) {
2013-06-19 11:58:16 -05:00
// why does anyone need to keep that in database?
delete metadata.readme;
2013-06-07 20:16:28 -05:00
if (err) return callback(err);
if (data.versions[version] != null) {
return callback(new UError({
status: 409,
msg: 'this version already present'
}));
}
data.versions[version] = metadata;
data['dist-tags'][tag] = version;
2013-06-14 03:34:29 -05:00
self.storage.update_json(name + '/' + info_file, data, callback);
2013-06-07 20:16:28 -05:00
});
}
2013-06-13 09:21:14 -05:00
Storage.prototype.add_tarball = function(name, filename, stream, callback) {
2013-06-14 03:34:29 -05:00
var self = this;
2013-06-07 20:16:28 -05:00
if (name === info_file) {
return callback(new UError({
status: 403,
msg: 'can\'t use this filename'
}));
}
var data = new Buffer(0);
stream.on('data', function(d) {
var tmp = data;
data = new Buffer(tmp.length+d.length);
tmp.copy(data, 0);
d.copy(data, tmp.length);
});
stream.on('end', function(d) {
2013-06-14 03:34:29 -05:00
self.storage.create(name + '/' + filename, data, function(err) {
2013-06-07 20:16:28 -05:00
if (err && err.code === 'EEXISTS') {
return callback(new UError({
status: 409,
msg: 'this tarball is already present'
}));
}
callback.apply(null, arguments);
});
});
}
2013-06-13 09:21:14 -05:00
Storage.prototype.get_tarball = function(name, filename, callback) {
2013-06-14 03:34:29 -05:00
this.storage.read(name + '/' + filename, function(err) {
2013-06-07 20:16:28 -05:00
if (err && err.code === 'ENOENT') {
return callback(new UError({
status: 404,
msg: 'no such package available'
}));
}
callback.apply(null, arguments);
});
}
2013-06-13 09:21:14 -05:00
Storage.prototype.get_package = function(name, callback) {
2013-06-14 03:34:29 -05:00
this.storage.read_json(name + '/' + info_file, function(err) {
2013-06-07 20:16:28 -05:00
if (err && err.code === 'ENOENT') {
return callback(new UError({
status: 404,
msg: 'no such package available'
}));
}
callback.apply(null, arguments);
});
}
2013-06-13 09:21:14 -05:00
module.exports = Storage;