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

848 lines
24 KiB
JavaScript
Raw Normal View History

/* eslint prefer-rest-params: "off" */
/* eslint prefer-spread: "off" */
'use strict';
const assert = require('assert');
const async = require('async');
const Crypto = require('crypto');
const fs = require('fs');
const Error = require('http-errors');
const Path = require('path');
const Stream = require('stream');
const URL = require('url');
2017-04-19 13:45:48 -05:00
const fs_storage = require('./local-fs');
const LocalData = require('./local-data');
const Logger = require('./logger');
const MyStreams = require('./streams');
const Utils = require('./utils');
const info_file = 'package.json';
2013-06-13 09:21:14 -05:00
2013-06-18 13:14:55 -05:00
// returns the minimal package file
const get_boilerplate = function(name) {
return {
// standard things
'name': name,
'versions': {},
'dist-tags': {},
// our own object
'_distfiles': {},
'_attachments': {},
'_uplinks': {},
};
};
2013-06-18 13:14:55 -05:00
/**
* Implements Storage interface (same for storage.js, local-storage.js, up-storage.js).
*/
2017-04-19 13:45:48 -05:00
class Storage {
/**
* Constructor
* @param {Object} config config list of properties
*/
constructor(config) {
this.config = config;
// local data handler is linked with the configuration handler
this.localList = new LocalData(Path.join(Path.resolve(Path.dirname(config.self_path || ''), config.storage),
// FUTURE: the database might be parameterizable from config.yaml
'.sinopia-db.json'
)
);
this.logger = Logger.logger.child({sub: 'fs'});
2017-04-19 13:45:48 -05:00
}
2013-10-18 16:17:53 -05:00
2017-04-19 13:45:48 -05:00
/**
* Add a package.
2017-04-19 13:45:48 -05:00
* @param {*} name
* @param {*} info
* @param {*} callback
* @return {Function}
2017-04-19 13:45:48 -05:00
*/
addPackage(name, info, callback) {
const storage = this.storage(name);
if (!storage) {
return callback( Error[404]('this package cannot be added'));
}
2017-04-19 13:45:48 -05:00
storage.create_json(info_file, get_boilerplate(name), function(err) {
if (err && err.code === 'EEXISTS') {
return callback( Error[409]('this package is already present') );
2017-04-19 13:45:48 -05:00
}
const latest = info['dist-tags'].latest;
2017-04-19 13:45:48 -05:00
if (latest && info.versions[latest]) {
return callback(null, info.versions[latest]);
2017-04-19 13:45:48 -05:00
}
return callback();
});
2017-04-19 13:45:48 -05:00
}
2017-04-19 13:45:48 -05:00
/**
* Remove package.
2017-04-19 13:45:48 -05:00
* @param {*} name
* @param {*} callback
* @return {Function}
2017-04-19 13:45:48 -05:00
*/
removePackage(name, callback) {
this.logger.info( {name: name}
, 'unpublishing @{name} (all)');
let storage = this.storage(name);
if (!storage) {
return callback( Error[404]('no such package available') );
}
2017-04-19 13:45:48 -05:00
storage.read_json(info_file, (err, data) => {
if (err) {
if (err.code === 'ENOENT') {
return callback( Error[404]('no such package available') );
2017-04-19 13:45:48 -05:00
} else {
return callback(err);
2017-04-19 13:45:48 -05:00
}
}
this._normalizePackage(data);
2017-04-19 13:45:48 -05:00
storage.unlink(info_file, function(err) {
if (err) {
return callback(err);
}
const files = Object.keys(data._attachments);
const unlinkNext = function(cb) {
if (files.length === 0) {
return cb();
}
let file = files.shift();
2017-04-19 13:45:48 -05:00
storage.unlink(file, function() {
unlinkNext(cb);
});
};
2017-04-19 13:45:48 -05:00
unlinkNext(function() {
// try to unlink the directory, but ignore errors because it can fail
storage.rmdir('.', function(err) {
callback(err);
});
});
});
});
this.localList.remove(name);
}
2017-04-19 13:45:48 -05:00
/**
* Synchronize remote package info with the local one
* @param {*} name
* @param {*} newdata
* @param {*} callback
*/
updateVersions(name, newdata, callback) {
this._readCreatePackage(name, (err, data) => {
if (err) {
return callback(err);
}
let change = false;
for (let ver in newdata.versions) {
2017-04-19 13:45:48 -05:00
if (data.versions[ver] == null) {
let verdata = newdata.versions[ver];
2017-04-19 13:45:48 -05:00
// we don't keep readmes for package versions,
// only one readme per package
delete verdata.readme;
2017-04-19 13:45:48 -05:00
change = true;
data.versions[ver] = verdata;
2017-04-19 13:45:48 -05:00
if (verdata.dist && verdata.dist.tarball) {
let filename = URL.parse(verdata.dist.tarball).pathname.replace(/^.*\//, '');
2017-04-19 13:45:48 -05:00
// we do NOT overwrite any existing records
if (data._distfiles[filename] == null) {
let hash = data._distfiles[filename] = {
2017-04-19 13:45:48 -05:00
url: verdata.dist.tarball,
sha: verdata.dist.shasum,
};
2017-04-19 13:45:48 -05:00
// if (verdata[Symbol('_verdaccio_uplink')]) {
if (verdata._verdaccio_uplink) {
// if we got this information from a known registry,
// use the same protocol for the tarball
//
// see https://github.com/rlidwka/sinopia/issues/166
let tarball_url = URL.parse(hash.url);
let uplink_url = URL.parse(this.config.uplinks[verdata._verdaccio_uplink].url);
2017-04-19 13:45:48 -05:00
if (uplink_url.host === tarball_url.host) {
tarball_url.protocol = uplink_url.protocol;
hash.registry = verdata._verdaccio_uplink;
hash.url = URL.format(tarball_url);
2017-04-19 13:45:48 -05:00
}
}
}
}
}
}
for (let tag in newdata['dist-tags']) {
2017-04-19 13:45:48 -05:00
if (!data['dist-tags'][tag] || data['dist-tags'][tag] !== newdata['dist-tags'][tag]) {
change = true;
data['dist-tags'][tag] = newdata['dist-tags'][tag];
2017-04-19 13:45:48 -05:00
}
}
for (let up in newdata._uplinks) {
if (Object.prototype.hasOwnProperty.call(newdata._uplinks, up)) {
const need_change = !Utils.is_object(data._uplinks[up])
|| newdata._uplinks[up].etag !== data._uplinks[up].etag
|| newdata._uplinks[up].fetched !== data._uplinks[up].fetched;
if (need_change) {
change = true;
data._uplinks[up] = newdata._uplinks[up];
}
2017-04-19 13:45:48 -05:00
}
}
if (newdata.readme !== data.readme) {
data.readme = newdata.readme;
change = true;
}
2013-06-18 13:14:55 -05:00
2017-04-19 13:45:48 -05:00
if (change) {
this.logger.debug('updating package info');
this._writePackage(name, data, function(err) {
callback(err, data);
});
2017-04-19 13:45:48 -05:00
} else {
callback(null, data);
2017-04-19 13:45:48 -05:00
}
});
2017-04-19 13:45:48 -05:00
}
2017-04-19 13:45:48 -05:00
/**
* Add a new version to a previous local package.
2017-04-19 13:45:48 -05:00
* @param {*} name
* @param {*} version
* @param {*} metadata
* @param {*} tag
* @param {*} callback
*/
addVersion(name, version, metadata, tag, callback) {
this._updatePackage(name, (data, cb) => {
2017-04-19 13:45:48 -05:00
// keep only one readme per package
data.readme = metadata.readme;
delete metadata.readme;
2017-04-19 13:45:48 -05:00
if (data.versions[version] != null) {
return cb( Error[409]('this version already present') );
2017-04-19 13:45:48 -05:00
}
2017-04-19 13:45:48 -05:00
// if uploaded tarball has a different shasum, it's very likely that we have some kind of error
if (Utils.is_object(metadata.dist) && typeof(metadata.dist.tarball) === 'string') {
let tarball = metadata.dist.tarball.replace(/.*\//, '');
2017-04-19 13:45:48 -05:00
if (Utils.is_object(data._attachments[tarball])) {
if (data._attachments[tarball].shasum != null && metadata.dist.shasum != null) {
if (data._attachments[tarball].shasum != metadata.dist.shasum) {
return cb( Error[400]('shasum error, '
+ data._attachments[tarball].shasum
+ ' != ' + metadata.dist.shasum) );
2017-04-19 13:45:48 -05:00
}
}
data._attachments[tarball].version = version;
}
}
data.versions[version] = metadata;
Utils.tag_version(data, version, tag);
this.localList.add(name);
cb();
}, callback);
2017-04-19 13:45:48 -05:00
}
2013-06-07 20:16:28 -05:00
2017-04-19 13:45:48 -05:00
/**
* Merge a new list of tags for a local packages with the existing one.
2017-04-19 13:45:48 -05:00
* @param {*} name
* @param {*} tags
* @param {*} callback
*/
mergeTags(name, tags, callback) {
this._updatePackage(name, function updater(data, cb) {
2017-04-19 13:45:48 -05:00
for (let t in tags) {
if (tags[t] === null) {
delete data['dist-tags'][t];
continue;
2017-04-19 13:45:48 -05:00
}
// be careful here with == (cast)
2017-04-19 13:45:48 -05:00
if (data.versions[tags[t]] == null) {
return cb( Error[404]('this version doesn\'t exist') );
2017-04-19 13:45:48 -05:00
}
Utils.tag_version(data, tags[t], t);
2015-05-10 10:23:31 -05:00
}
cb();
}, callback);
2017-04-19 13:45:48 -05:00
}
2015-05-10 10:23:31 -05:00
/**
* Replace the complete list of tags for a local package.
2017-04-19 13:45:48 -05:00
* @param {*} name
* @param {*} tags
* @param {*} callback
*/
replaceTags(name, tags, callback) {
this._updatePackage(name, function updater(data, cb) {
data['dist-tags'] = {};
2015-05-10 10:23:31 -05:00
2017-04-19 13:45:48 -05:00
for (let t in tags) {
if (tags[t] === null) {
delete data['dist-tags'][t];
continue;
2017-04-19 13:45:48 -05:00
}
2015-05-10 10:23:31 -05:00
2017-04-19 13:45:48 -05:00
if (data.versions[tags[t]] == null) {
return cb( Error[404]('this version doesn\'t exist') );
2017-04-19 13:45:48 -05:00
}
2015-05-10 10:23:31 -05:00
Utils.tag_version(data, tags[t], t);
}
cb();
}, callback);
2017-04-19 13:45:48 -05:00
}
2017-04-19 13:45:48 -05:00
/**
* Update the package metadata, tags and attachments (tarballs).
* Note: Currently supports unpublishing only.
2017-04-19 13:45:48 -05:00
* @param {*} name
* @param {*} metadata
* @param {*} revision
* @param {*} callback
* @return {Function}
2017-04-19 13:45:48 -05:00
*/
changePackage(name, metadata, revision, callback) {
2017-04-19 13:45:48 -05:00
if (!Utils.is_object(metadata.versions) || !Utils.is_object(metadata['dist-tags'])) {
return callback( Error[422]('bad data') );
}
this._updatePackage(name, (data, cb) => {
2017-04-19 13:45:48 -05:00
for (let ver in data.versions) {
if (metadata.versions[ver] == null) {
this.logger.info( {name: name, version: ver},
'unpublishing @{name}@@{version}');
delete data.versions[ver];
for (let file in data._attachments) {
2017-04-19 13:45:48 -05:00
if (data._attachments[file].version === ver) {
delete data._attachments[file].version;
2017-04-19 13:45:48 -05:00
}
}
}
}
data['dist-tags'] = metadata['dist-tags'];
cb();
2017-04-19 13:45:48 -05:00
}, function(err) {
if (err) {
return callback(err);
}
callback();
});
2017-04-19 13:45:48 -05:00
}
2017-04-19 13:45:48 -05:00
/**
* Remove a tarball.
2017-04-19 13:45:48 -05:00
* @param {*} name
* @param {*} filename
* @param {*} revision
* @param {*} callback
*/
removeTarball(name, filename, revision, callback) {
assert(Utils.validate_name(filename));
2017-04-19 13:45:48 -05:00
this._updatePackage(name, (data, cb) => {
2017-04-19 13:45:48 -05:00
if (data._attachments[filename]) {
delete data._attachments[filename];
cb();
2017-04-19 13:45:48 -05:00
} else {
cb(Error[404]('no such file available'));
2017-04-19 13:45:48 -05:00
}
}, (err) => {
if (err) {
return callback(err);
}
let storage = this.storage(name);
if (storage) {
storage.unlink(filename, callback);
}
});
2017-04-19 13:45:48 -05:00
}
2017-04-19 13:45:48 -05:00
/**
* Add a tarball.
* @param {String} name
* @param {String} filename
* @return {Function}
2017-04-19 13:45:48 -05:00
*/
addTarball(name, filename) {
assert(Utils.validate_name(filename));
2017-04-19 13:45:48 -05:00
let stream = new MyStreams.UploadTarball();
let _transform = stream._transform;
let length = 0;
let shasum = Crypto.createHash('sha1');
2017-04-19 13:45:48 -05:00
stream.abort = stream.done = function() {};
2017-04-19 13:45:48 -05:00
stream._transform = function(data) {
shasum.update(data);
length += data.length;
_transform.apply(stream, arguments);
};
2017-04-19 13:45:48 -05:00
if (name === info_file || name === '__proto__') {
process.nextTick(function() {
stream.emit('error', Error[403]('can\'t use this filename'));
});
return stream;
2017-04-19 13:45:48 -05:00
}
let storage = this.storage(name);
2017-04-19 13:45:48 -05:00
if (!storage) {
process.nextTick(function() {
stream.emit('error', Error[404]('can\'t upload this package'));
});
return stream;
2017-04-19 13:45:48 -05:00
}
let wstream = storage.write_stream(filename);
2017-04-19 13:45:48 -05:00
wstream.on('error', (err) => {
if (err.code === 'EEXISTS') {
stream.emit('error', Error[409]('this tarball is already present'));
2017-04-19 13:45:48 -05:00
} else if (err.code === 'ENOENT') {
// check if package exists to throw an appropriate message
this.getPackage(name, function(_err, res) {
2017-04-19 13:45:48 -05:00
if (_err) {
stream.emit('error', _err);
2017-04-19 13:45:48 -05:00
} else {
stream.emit('error', err);
2017-04-19 13:45:48 -05:00
}
});
2017-04-19 13:45:48 -05:00
} else {
stream.emit('error', err);
2017-04-19 13:45:48 -05:00
}
});
2017-04-19 13:45:48 -05:00
wstream.on('open', function() {
// re-emitting open because it's handled in storage.js
stream.emit('open');
});
2017-04-19 13:45:48 -05:00
wstream.on('success', () => {
this._updatePackage(name, function updater(data, cb) {
2017-04-19 13:45:48 -05:00
data._attachments[filename] = {
shasum: shasum.digest('hex'),
};
cb();
2017-04-19 13:45:48 -05:00
}, function(err) {
if (err) {
stream.emit('error', err);
2017-04-19 13:45:48 -05:00
} else {
stream.emit('success');
}
});
});
2017-04-19 13:45:48 -05:00
stream.abort = function() {
wstream.abort();
};
2017-04-19 13:45:48 -05:00
stream.done = function() {
if (!length) {
stream.emit('error', Error[422]('refusing to accept zero-length file'));
wstream.abort();
} else {
wstream.done();
}
};
stream.pipe(wstream);
return stream;
}
2017-04-19 13:45:48 -05:00
/**
* Get a tarball.
2017-04-19 13:45:48 -05:00
* @param {*} name
* @param {*} filename
* @param {*} callback
* @return {Function}
2017-04-19 13:45:48 -05:00
*/
getTarball(name, filename, callback) {
assert(Utils.validate_name(filename));
const stream = new MyStreams.ReadTarball();
2017-04-19 13:45:48 -05:00
stream.abort = function() {
if (rstream) {
rstream.abort();
}
};
let storage = this.storage(name);
2017-04-19 13:45:48 -05:00
if (!storage) {
process.nextTick(function() {
stream.emit('error', Error[404]('no such file available'));
});
return stream;
2017-04-19 13:45:48 -05:00
}
/* eslint no-var: "off" */
var rstream = storage.read_stream(filename);
2017-04-19 13:45:48 -05:00
rstream.on('error', function(err) {
if (err && err.code === 'ENOENT') {
stream.emit('error', Error(404, 'no such file available'));
} else {
stream.emit('error', err);
}
});
2017-04-19 13:45:48 -05:00
rstream.on('content-length', function(v) {
stream.emit('content-length', v);
});
2017-04-19 13:45:48 -05:00
rstream.on('open', function() {
// re-emitting open because it's handled in storage.js
stream.emit('open');
rstream.pipe(stream);
});
return stream;
2017-04-19 13:45:48 -05:00
}
2017-04-19 13:45:48 -05:00
/**
* Retrieve a package by name.
2017-04-19 13:45:48 -05:00
* @param {*} name
* @param {*} options
* @param {*} callback
* @return {Function}
2017-04-19 13:45:48 -05:00
*/
getPackage(name, options, callback) {
2017-04-19 13:45:48 -05:00
if (typeof(options) === 'function') {
callback = options, options = {};
}
let storage = this.storage(name);
if (!storage) {
return callback( Error[404]('no such package available') );
}
2017-04-19 13:45:48 -05:00
storage.read_json(info_file, (err, result) => {
if (err) {
if (err.code === 'ENOENT') {
return callback( Error[404]('no such package available') );
2017-04-19 13:45:48 -05:00
} else {
return callback(this._internalError(err, info_file, 'error reading'));
2017-04-19 13:45:48 -05:00
}
}
this._normalizePackage(result);
callback(err, result);
});
}
2017-04-19 13:45:48 -05:00
/**
* Search a local package.
* @param {*} startKey
2017-04-19 13:45:48 -05:00
* @param {*} options
* @return {Function}
2017-04-19 13:45:48 -05:00
*/
search(startKey, options) {
const stream = new Stream.PassThrough({objectMode: true});
2017-04-19 13:45:48 -05:00
this._eachPackage((item, cb) => {
2017-04-19 13:45:48 -05:00
fs.stat(item.path, (err, stats) => {
if (err) {
return cb(err);
}
if (stats.mtime > startKey) {
this.getPackage(item.name, options, function(err, data) {
2017-04-19 13:45:48 -05:00
if (err) {
return cb(err);
}
let versions = Utils.semver_sort(Object.keys(data.versions));
let latest = data['dist-tags'] && data['dist-tags'].latest ? data['dist-tags'].latest : versions.pop();
2017-04-19 13:45:48 -05:00
if (data.versions[latest]) {
stream.push({
'name': data.versions[latest].name,
'description': data.versions[latest].description,
'dist-tags': {latest: latest},
'maintainers': data.versions[latest].maintainers
|| [data.versions[latest].author].filter(Boolean),
'author': data.versions[latest].author,
'repository': data.versions[latest].repository,
'readmeFilename': data.versions[latest].readmeFilename || '',
'homepage': data.versions[latest].homepage,
'keywords': data.versions[latest].keywords,
'bugs': data.versions[latest].bugs,
'license': data.versions[latest].license,
'time': {
modified: item.time ? new Date(item.time).toISOString() : undefined,
2017-04-19 13:45:48 -05:00
},
'versions': {},
});
2017-04-19 13:45:48 -05:00
}
cb();
});
2017-04-19 13:45:48 -05:00
} else {
cb();
}
});
2017-04-19 13:45:48 -05:00
}, function on_end(err) {
if (err) return stream.emit('error', err);
stream.end();
});
return stream;
2017-04-19 13:45:48 -05:00
}
2013-10-22 02:00:04 -05:00
/**
* Retrieve a wrapper that provide access to the package location.
* @param {*} pkg package name.
* @return {Object}
*/
storage(pkg) {
let path = this.config.get_package_spec(pkg).storage;
if (path == null) {
path = this.config.storage;
}
if (path == null || path === false) {
this.logger.debug( {name: pkg}
, 'this package has no storage defined: @{name}' );
return null;
}
return new PathWrapper(
Path.join(
Path.resolve(Path.dirname(this.config.self_path || ''), path),
pkg
)
);
}
/**
* Walks through each package and calls `on_package` on them.
* @param {*} on_package
* @param {*} on_end
*/
_eachPackage(on_package, on_end) {
let storages = {};
storages[this.config.storage] = true;
if (this.config.packages) {
Object.keys(this.packages || {}).map( (pkg) => {
if (this.config.packages[pkg].storage) {
storages[this.config.packages[pkg].storage] = true;
}
});
}
const base = Path.dirname(this.config.self_path);
async.eachSeries(Object.keys(storages), function(storage, cb) {
fs.readdir(Path.resolve(base, storage), function(err, files) {
if (err) {
return cb(err);
}
async.eachSeries(files, function(file, cb) {
if (file.match(/^@/)) {
// scoped
fs.readdir(Path.resolve(base, storage, file), function(err, files) {
if (err) {
return cb(err);
}
async.eachSeries(files, function(file2, cb) {
if (Utils.validate_name(file2)) {
on_package({
name: `${file}/${file2}`,
path: Path.resolve(base, storage, file, file2),
}, cb);
} else {
cb();
}
}, cb);
});
} else if (Utils.validate_name(file)) {
on_package({
name: file,
path: Path.resolve(base, storage, file),
}, cb);
} else {
cb();
}
}, cb);
});
}, on_end);
}
2017-04-19 13:45:48 -05:00
/**
* Normalise package properties, tags, revision id.
* @param {Object} pkg package reference.
2017-04-19 13:45:48 -05:00
*/
_normalizePackage(pkg) {
['versions', 'dist-tags', '_distfiles', '_attachments', '_uplinks'].forEach(function(key) {
if (!Utils.is_object(pkg[key])) {
pkg[key] = {};
}
});
2017-04-19 13:45:48 -05:00
if (typeof(pkg._rev) !== 'string') {
pkg._rev = '0-0000000000000000';
}
// normalize dist-tags
Utils.normalize_dist_tags(pkg);
2017-04-19 13:45:48 -05:00
}
2013-10-22 02:00:04 -05:00
/**
* Retrieve either a previous created local package or a boilerplate.
* @param {*} name
* @param {*} callback
* @return {Function}
*/
_readCreatePackage(name, callback) {
const storage = this.storage(name);
if (!storage) {
let data = get_boilerplate(name);
this._normalizePackage(data);
return callback(null, data);
}
storage.read_json(info_file, (err, data) => {
// TODO: race condition
if (err) {
if (err.code === 'ENOENT') {
// if package doesn't exist, we create it here
data = get_boilerplate(name);
} else {
return callback(this._internalError(err, info_file, 'error reading'));
}
}
this._normalizePackage(data);
callback(null, data);
});
}
/**
* Handle internal error
* @param {*} err
* @param {*} file
* @param {*} message
* @return {Object} Error instance
*/
_internalError(err, file, message) {
this.logger.error( {err: err, file: file},
message + ' @{file}: @{!err.message}' );
return Error[500]();
}
/**
* This function allows to update the package thread-safely
Algorithm:
1. lock package.json for writing
2. read package.json
3. updateFn(pkg, cb), and wait for cb
4. write package.json.tmp
5. move package.json.tmp package.json
6. callback(err?)
* @param {*} name package name
* @param {*} updateFn function(package, cb) - update function
* @param {*} _callback callback that gets invoked after it's all updated
* @return {Function}
*/
_updatePackage(name, updateFn, _callback) {
const storage = this.storage(name);
if (!storage) {
return _callback( Error[404]('no such package available') );
}
storage.lock_and_read_json(info_file, (err, json) => {
let locked = false;
// callback that cleans up lock first
const callback = function(err) {
let _args = arguments;
if (locked) {
storage.unlock_file(info_file, function() {
// ignore any error from the unlock
_callback.apply(err, _args);
});
} else {
_callback.apply(null, _args);
}
};
if (!err) {
locked = true;
}
if (err) {
if (err.code === 'EAGAIN') {
return callback( Error[503]('resource temporarily unavailable') );
} else if (err.code === 'ENOENT') {
return callback( Error[404]('no such package available') );
} else {
return callback(err);
}
}
this._normalizePackage(json);
updateFn(json, (err) => {
if (err) {
return callback(err);
}
this._writePackage(name, json, callback);
});
});
}
2017-04-19 13:45:48 -05:00
/**
* Update the revision (_rev) string for a package.
2017-04-19 13:45:48 -05:00
* @param {*} name
* @param {*} json
* @param {*} callback
* @return {Function}
2017-04-19 13:45:48 -05:00
*/
_writePackage(name, json, callback) {
2017-04-19 13:45:48 -05:00
// calculate revision a la couchdb
if (typeof(json._rev) !== 'string') {
json._rev = '0-0000000000000000';
}
let rev = json._rev.split('-');
json._rev = ((+rev[0] || 0) + 1) + '-' + Crypto.pseudoRandomBytes(8).toString('hex');
2013-10-22 02:00:04 -05:00
let storage = this.storage(name);
if (!storage) {
return callback();
}
storage.write_json(info_file, json, callback);
2017-04-19 13:45:48 -05:00
}
2013-10-22 02:00:04 -05:00
}
const PathWrapper = (function() {
/**
* A wrapper adding paths to fs_storage methods.
*/
class Wrapper {
/**
* @param {*} path
*/
constructor(path) {
this.path = path;
}
}
const wrapLocalStorageMethods = function(method) {
return function() {
let args = Array.prototype.slice.apply(arguments);
/* eslint no-invalid-this: off */
args[0] = Path.join(this.path, args[0] || '');
return fs_storage[method].apply(null, args);
};
};
for (let i in fs_storage) {
if (fs_storage.hasOwnProperty(i)) {
Wrapper.prototype[i] = wrapLocalStorageMethods(i);
}
}
return Wrapper;
})();
2014-01-13 11:48:51 -05:00
module.exports = Storage;