verdaccio
Version:
A lightweight private npm proxy registry
630 lines (501 loc) • 68 kB
JavaScript
"use strict";
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.default = void 0;
var _assert = _interopRequireDefault(require("assert"));
var _stream = _interopRequireDefault(require("stream"));
var _lodash = _interopRequireDefault(require("lodash"));
var _async = _interopRequireDefault(require("async"));
var _streams = require("@verdaccio/streams");
var _logger = require("../lib/logger");
var _upStorage = _interopRequireDefault(require("./up-storage"));
var _search = _interopRequireDefault(require("./search"));
var _constants = require("./constants");
var _localStorage = _interopRequireDefault(require("./local-storage"));
var _storageUtils = require("./storage-utils");
var _uplinkUtil = require("./uplink-util");
var _metadataUtils = require("./metadata-utils");
var _utils = require("./utils");
var _configUtils = require("./config-utils");
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
class Storage {
constructor(config) {
_defineProperty(this, "localStorage", void 0);
_defineProperty(this, "config", void 0);
_defineProperty(this, "logger", void 0);
_defineProperty(this, "uplinks", void 0);
_defineProperty(this, "filters", void 0);
this.config = config;
this.uplinks = (0, _uplinkUtil.setupUpLinks)(config);
this.logger = _logger.logger.child();
this.filters = []; // @ts-ignore
this.localStorage = null;
}
init(config, filters = []) {
this.filters = filters;
this.localStorage = new _localStorage.default(this.config, _logger.logger);
return this.localStorage.getSecret(config);
}
/**
* 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
*/
async addPackage(name, metadata, callback) {
try {
await (0, _storageUtils.checkPackageLocal)(name, this.localStorage);
await (0, _storageUtils.checkPackageRemote)(name, this._isAllowPublishOffline(), this._syncUplinksMetadata.bind(this));
await (0, _storageUtils.publishPackage)(name, metadata, this.localStorage);
callback();
} catch (err) {
callback(err);
}
}
_isAllowPublishOffline() {
return typeof this.config.publish !== 'undefined' && _lodash.default.isBoolean(this.config.publish.allow_offline) && this.config.publish.allow_offline;
}
readTokens(filter) {
return this.localStorage.readTokens(filter);
}
saveToken(token) {
return this.localStorage.saveToken(token);
}
deleteToken(user, tokenKey) {
return this.localStorage.deleteToken(user, tokenKey);
}
/**
* Add a new version of package {name} to a system
Used storages: local (write)
*/
addVersion(name, version, metadata, tag, callback) {
this.localStorage.addVersion(name, version, metadata, tag, callback);
}
/**
* Tags a package version with a provided tag
Used storages: local (write)
*/
mergeTags(name, tagHash, callback) {
this.localStorage.mergeTags(name, tagHash, callback);
}
/**
* 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)
*/
changePackage(name, metadata, revision, callback) {
this.localStorage.changePackage(name, metadata, revision, callback);
}
/**
* Remove a package from a system
Function removes a package from local storage
Used storages: local (write)
*/
removePackage(name, callback) {
this.localStorage.removePackage(name, callback); // update the indexer
_search.default.remove(name);
}
/**
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 storage: local (write)
*/
removeTarball(name, filename, revision, callback) {
this.localStorage.removeTarball(name, filename, revision, callback);
}
/**
* Upload a tarball for {name} package
Function is synchronous and returns a WritableStream
Used storages: local (write)
*/
addTarball(name, filename) {
return this.localStorage.addTarball(name, filename);
}
/**
Get a tarball from a storage for {name} package
Function is synchronous and returns a ReadableStream
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)
*/
getTarball(name, filename) {
const readStream = new _streams.ReadTarball({});
readStream.abort = function () {};
const self = this; // if someone requesting tarball, it means that we should already have some
// information about it, so fetching package info is unnecessary
// trying local first
// flow: should be IReadTarball
let localStream = self.localStorage.getTarball(name, filename);
let isOpen = false;
localStream.on('error', err => {
if (isOpen || err.status !== _constants.HTTP_STATUS.NOT_FOUND) {
return readStream.emit('error', err);
} // local reported 404
const err404 = err;
localStream.abort();
localStream = null; // we force for garbage collector
self.localStorage.getPackageMetadata(name, (err, info) => {
if (_lodash.default.isNil(err) && info._distfiles && _lodash.default.isNil(info._distfiles[filename]) === false) {
// information about this file exists locally
serveFile(info._distfiles[filename]);
} else {
// we know nothing about this file, trying to get information elsewhere
self._syncUplinksMetadata(name, info, {}, (err, info) => {
if (_lodash.default.isNil(err) === false) {
return readStream.emit('error', err);
}
if (_lodash.default.isNil(info._distfiles) || _lodash.default.isNil(info._distfiles[filename])) {
return readStream.emit('error', err404);
}
serveFile(info._distfiles[filename]);
});
}
});
});
localStream.on('content-length', function (v) {
readStream.emit('content-length', v);
});
localStream.on('open', function () {
isOpen = true;
localStream.pipe(readStream);
});
return readStream;
/**
* Fetch and cache local/remote packages.
* @param {Object} file define the package shape
*/
function serveFile(file) {
let uplink = null;
for (const uplinkId in self.uplinks) {
if ((0, _configUtils.hasProxyTo)(name, uplinkId, self.config.packages)) {
uplink = self.uplinks[uplinkId];
}
}
if (uplink == null) {
uplink = new _upStorage.default({
url: file.url,
cache: true,
_autogenerated: true
}, self.config);
}
let savestream = null;
if (uplink.config.cache) {
savestream = self.localStorage.addTarball(name, filename);
}
let on_open = function () {
// prevent it from being called twice
on_open = function () {};
const rstream2 = uplink.fetchTarball(file.url);
rstream2.on('error', function (err) {
if (savestream) {
savestream.abort();
}
savestream = null;
readStream.emit('error', err);
});
rstream2.on('end', function () {
if (savestream) {
savestream.done();
}
});
rstream2.on('content-length', function (v) {
readStream.emit('content-length', v);
if (savestream) {
savestream.emit('content-length', v);
}
});
rstream2.pipe(readStream);
if (savestream) {
rstream2.pipe(savestream);
}
};
if (savestream) {
savestream.on('open', function () {
on_open();
});
savestream.on('error', function (err) {
self.logger.warn({
err: err,
fileName: file
}, 'error saving file @{fileName}: @{err.message}\n@{err.stack}');
if (savestream) {
savestream.abort();
}
savestream = null;
on_open();
});
} else {
on_open();
}
}
}
/**
Retrieve a package metadata for {name} package
Function invokes localStorage.getPackage 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)
* @param {object} options
* @property {string} options.name Package Name
* @property {object} options.req Express `req` object
* @property {boolean} options.keepUpLinkData keep up link info in package meta, last update, etc.
* @property {function} options.callback Callback for receive data
*/
getPackage(options) {
this.localStorage.getPackageMetadata(options.name, (err, data) => {
if (err && (!err.status || err.status >= _constants.HTTP_STATUS.INTERNAL_ERROR)) {
// report internal errors right away
return options.callback(err);
}
this._syncUplinksMetadata(options.name, data, {
req: options.req,
uplinksLook: options.uplinksLook
}, function getPackageSynUpLinksCallback(err, result, uplinkErrors) {
if (err) {
return options.callback(err);
}
(0, _utils.normalizeDistTags)((0, _storageUtils.cleanUpLinksRef)(options.keepUpLinkData, result)); // npm can throw if this field doesn't exist
result._attachments = {};
options.callback(null, result, uplinkErrors);
});
});
}
/**
Retrieve remote and local packages more recent than {startkey}
Function streams all packages from all uplinks first, and then
local packages.
Note that local packages could override registry ones just because
they appear in JSON last. That's a trade-off we make to avoid
memory issues.
Used storages: local && uplink (proxy_access)
* @param {*} startkey
* @param {*} options
* @return {Stream}
*/
search(startkey, options) {
const self = this; // stream to write a tarball
const stream = new _stream.default.PassThrough({
objectMode: true
});
_async.default.eachSeries(Object.keys(this.uplinks), function (up_name, cb) {
// shortcut: if `local=1` is supplied, don't call uplinks
if (options.req.query.local !== undefined) {
return cb();
} // search by keyword for each uplink
const lstream = self.uplinks[up_name].search(options); // join streams
lstream.pipe(stream, {
end: false
});
lstream.on('error', function (err) {
self.logger.error({
err: err
}, 'uplink error: @{err.message}');
cb();
cb = function () {};
});
lstream.on('end', function () {
cb();
cb = function () {};
});
stream.abort = function () {
if (lstream.abort) {
lstream.abort();
}
cb();
cb = function () {};
};
}, // executed after all series
function () {
// attach a local search results
const lstream = self.localStorage.search(startkey, options);
stream.abort = function () {
lstream.abort();
};
lstream.pipe(stream, {
end: true
});
lstream.on('error', function (err) {
self.logger.error({
err: err
}, 'search error: @{err.message}');
stream.end();
});
});
return stream;
}
/**
* Retrieve only private local packages
* @param {*} callback
*/
getLocalDatabase(callback) {
const self = this;
this.localStorage.storagePlugin.get((err, locals) => {
if (err) {
callback(err);
}
const packages = [];
const getPackage = function (itemPkg) {
self.localStorage.getPackageMetadata(locals[itemPkg], function (err, pkgMetadata) {
if (_lodash.default.isNil(err)) {
const latest = pkgMetadata[_constants.DIST_TAGS].latest;
if (latest && pkgMetadata.versions[latest]) {
const version = pkgMetadata.versions[latest];
const timeList = pkgMetadata.time;
const time = timeList[latest]; // @ts-ignore
version.time = time; // Add for stars api
// @ts-ignore
version.users = pkgMetadata.users;
packages.push(version);
} else {
self.logger.warn({
package: locals[itemPkg]
}, 'package @{package} does not have a "latest" tag?');
}
}
if (itemPkg >= locals.length - 1) {
callback(null, packages);
} else {
getPackage(itemPkg + 1);
}
});
};
if (locals.length) {
getPackage(0);
} else {
callback(null, []);
}
});
}
/**
* Function fetches package metadata 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)
*/
_syncUplinksMetadata(name, packageInfo, options, callback) {
let found = true;
const self = this;
const upLinks = [];
const hasToLookIntoUplinks = _lodash.default.isNil(options.uplinksLook) || options.uplinksLook;
if (!packageInfo) {
found = false;
packageInfo = (0, _storageUtils.generatePackageTemplate)(name);
}
for (const uplink in this.uplinks) {
if ((0, _configUtils.hasProxyTo)(name, uplink, this.config.packages) && hasToLookIntoUplinks) {
upLinks.push(this.uplinks[uplink]);
}
}
_async.default.map(upLinks, (upLink, cb) => {
const _options = Object.assign({}, options);
const upLinkMeta = packageInfo._uplinks[upLink.upname];
if ((0, _utils.isObject)(upLinkMeta)) {
const fetched = upLinkMeta.fetched;
if (fetched && Date.now() - fetched < upLink.maxage) {
return cb();
}
_options.etag = upLinkMeta.etag;
}
upLink.getRemoteMetadata(name, _options, (err, upLinkResponse, eTag) => {
if (err && err.remoteStatus === 304) {
upLinkMeta.fetched = Date.now();
}
if (err || !upLinkResponse) {
return cb(null, [err || _utils.ErrorCode.getInternalError('no data')]);
}
try {
(0, _utils.validateMetadata)(upLinkResponse, name);
} catch (err) {
self.logger.error({
sub: 'out',
err: err
}, 'package.json validating error @{!err.message}\n@{err.stack}');
return cb(null, [err]);
}
packageInfo._uplinks[upLink.upname] = {
etag: eTag,
fetched: Date.now()
};
packageInfo.time = (0, _storageUtils.mergeUplinkTimeIntoLocal)(packageInfo, upLinkResponse);
(0, _uplinkUtil.updateVersionsHiddenUpLink)(upLinkResponse.versions, upLink);
try {
(0, _metadataUtils.mergeVersions)(packageInfo, upLinkResponse);
} catch (err) {
self.logger.error({
sub: 'out',
err: err
}, 'package.json parsing error @{!err.message}\n@{err.stack}');
return cb(null, [err]);
} // if we got to this point, assume that the correct package exists
// on the uplink
found = true;
cb();
});
}, // @ts-ignore
(err, upLinksErrors) => {
(0, _assert.default)(!err && Array.isArray(upLinksErrors)); // Check for connection timeout or reset errors with uplink(s)
// (these should be handled differently from the package not being found)
if (!found) {
let uplinkTimeoutError;
for (let i = 0; i < upLinksErrors.length; i++) {
if (upLinksErrors[i]) {
for (let j = 0; j < upLinksErrors[i].length; j++) {
if (upLinksErrors[i][j]) {
const code = upLinksErrors[i][j].code;
if (code === 'ETIMEDOUT' || code === 'ESOCKETTIMEDOUT' || code === 'ECONNRESET') {
uplinkTimeoutError = true;
break;
}
}
}
}
}
if (uplinkTimeoutError) {
return callback(_utils.ErrorCode.getServiceUnavailable(), null, upLinksErrors);
}
return callback(_utils.ErrorCode.getNotFound(_constants.API_ERROR.NO_PACKAGE), null, upLinksErrors);
}
if (upLinks.length === 0) {
return callback(null, packageInfo);
}
self.localStorage.updateVersions(name, packageInfo, async (err, packageJsonLocal) => {
if (err) {
return callback(err);
} // Any error here will cause a 404, like an uplink error. This is likely the right thing to do
// as a broken filter is a security risk.
const filterErrors = []; // This MUST be done serially and not in parallel as they modify packageJsonLocal
for (const filter of self.filters) {
try {
// These filters can assume it's save to modify packageJsonLocal and return it directly for
// performance (i.e. need not be pure)
packageJsonLocal = await filter.filter_metadata(packageJsonLocal);
} catch (err) {
filterErrors.push(err);
}
}
callback(null, packageJsonLocal, _lodash.default.concat(upLinksErrors, filterErrors));
});
});
}
/**
* Set a hidden value for each version.
* @param {Array} versions list of version
* @param {String} upLink uplink name
* @private
*/
_updateVersionsHiddenUpLink(versions, upLink) {
for (const i in versions) {
if (Object.prototype.hasOwnProperty.call(versions, i)) {
const version = versions[i]; // holds a "hidden" value to be used by the package storage.
// $FlowFixMe
version[Symbol.for('__verdaccio_uplink')] = upLink.upname;
}
}
}
}
var _default = Storage;
exports.default = _default;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/lib/storage.ts"],"names":["Storage","constructor","config","uplinks","logger","child","filters","localStorage","init","LocalStorage","getSecret","addPackage","name","metadata","callback","_isAllowPublishOffline","_syncUplinksMetadata","bind","err","publish","_","isBoolean","allow_offline","readTokens","filter","saveToken","token","deleteToken","user","tokenKey","addVersion","version","tag","mergeTags","tagHash","changePackage","revision","removePackage","Search","remove","removeTarball","filename","addTarball","getTarball","readStream","ReadTarball","abort","self","localStream","isOpen","on","status","HTTP_STATUS","NOT_FOUND","emit","err404","getPackageMetadata","info","isNil","_distfiles","serveFile","v","pipe","file","uplink","uplinkId","packages","ProxyStorage","url","cache","_autogenerated","savestream","on_open","rstream2","fetchTarball","done","warn","fileName","getPackage","options","data","INTERNAL_ERROR","req","uplinksLook","getPackageSynUpLinksCallback","result","uplinkErrors","keepUpLinkData","_attachments","search","startkey","stream","Stream","PassThrough","objectMode","async","eachSeries","Object","keys","up_name","cb","query","local","undefined","lstream","end","error","getLocalDatabase","storagePlugin","get","locals","itemPkg","pkgMetadata","latest","DIST_TAGS","versions","timeList","time","users","push","package","length","packageInfo","found","upLinks","hasToLookIntoUplinks","map","upLink","_options","assign","upLinkMeta","_uplinks","upname","fetched","Date","now","maxage","etag","getRemoteMetadata","upLinkResponse","eTag","remoteStatus","ErrorCode","getInternalError","sub","upLinksErrors","Array","isArray","uplinkTimeoutError","i","j","code","getServiceUnavailable","getNotFound","API_ERROR","NO_PACKAGE","updateVersions","packageJsonLocal","filterErrors","filter_metadata","concat","_updateVersionsHiddenUpLink","prototype","hasOwnProperty","call","Symbol","for"],"mappings":"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","sourcesContent":["import assert from 'assert';\nimport Stream from 'stream';\nimport _ from 'lodash';\nimport async, { AsyncResultArrayCallback } from 'async';\nimport { ReadTarball } from '@verdaccio/streams';\nimport {\n  IReadTarball,\n  IUploadTarball,\n  Versions,\n  Package,\n  Config,\n  MergeTags,\n  Version,\n  DistFile,\n  Callback,\n  Logger\n} from '@verdaccio/types';\nimport { GenericBody, TokenFilter, Token } from '@verdaccio/types';\nimport { VerdaccioError } from '@verdaccio/commons-api';\nimport {\n  IStorage,\n  IProxy,\n  IStorageHandler,\n  ProxyList,\n  StringValue,\n  IGetPackageOptions,\n  ISyncUplinks,\n  IPluginFilters\n} from '../../types';\nimport { logger } from '../lib/logger';\nimport ProxyStorage from './up-storage';\nimport Search from './search';\nimport { API_ERROR, HTTP_STATUS, DIST_TAGS } from './constants';\nimport LocalStorage from './local-storage';\nimport {\n  checkPackageLocal,\n  publishPackage,\n  checkPackageRemote,\n  cleanUpLinksRef,\n  mergeUplinkTimeIntoLocal,\n  generatePackageTemplate\n} from './storage-utils';\nimport { setupUpLinks, updateVersionsHiddenUpLink } from './uplink-util';\nimport { mergeVersions } from './metadata-utils';\nimport { ErrorCode, normalizeDistTags, validateMetadata, isObject } from './utils';\nimport { hasProxyTo } from './config-utils';\n\nclass Storage implements IStorageHandler {\n  public localStorage: IStorage;\n  public config: Config;\n  public logger: Logger;\n  public uplinks: ProxyList;\n  public filters: IPluginFilters;\n\n  public constructor(config: Config) {\n    this.config = config;\n    this.uplinks = setupUpLinks(config);\n    this.logger = logger.child();\n    this.filters = [];\n    // @ts-ignore\n    this.localStorage = null;\n  }\n\n  public init(config: Config, filters: IPluginFilters = []): Promise<string> {\n    this.filters = filters;\n    this.localStorage = new LocalStorage(this.config, logger);\n\n    return this.localStorage.getSecret(config);\n  }\n\n  /**\n   *  Add a {name} package to a system\n   Function checks if package with the same name is available from uplinks.\n   If it isn't, we create package locally\n   Used storages: local (write) && uplinks\n   */\n  public async addPackage(name: string, metadata: any, callback: Function): Promise<void> {\n    try {\n      await checkPackageLocal(name, this.localStorage);\n      await checkPackageRemote(\n        name,\n        this._isAllowPublishOffline(),\n        this._syncUplinksMetadata.bind(this)\n      );\n      await publishPackage(name, metadata, this.localStorage as IStorage);\n      callback();\n    } catch (err) {\n      callback(err);\n    }\n  }\n\n  private _isAllowPublishOffline(): boolean {\n    return (\n      typeof this.config.publish !== 'undefined' &&\n      _.isBoolean(this.config.publish.allow_offline) &&\n      this.config.publish.allow_offline\n    );\n  }\n\n  public readTokens(filter: TokenFilter): Promise<Token[]> {\n    return this.localStorage.readTokens(filter);\n  }\n\n  public saveToken(token: Token): Promise<void> {\n    return this.localStorage.saveToken(token);\n  }\n\n  public deleteToken(user: string, tokenKey: string): Promise<any> {\n    return this.localStorage.deleteToken(user, tokenKey);\n  }\n\n  /**\n   * Add a new version of package {name} to a system\n   Used storages: local (write)\n   */\n  public addVersion(\n    name: string,\n    version: string,\n    metadata: Version,\n    tag: StringValue,\n    callback: Callback\n  ): void {\n    this.localStorage.addVersion(name, version, metadata, tag, callback);\n  }\n\n  /**\n   * Tags a package version with a provided tag\n   Used storages: local (write)\n   */\n  public mergeTags(name: string, tagHash: MergeTags, callback: Callback): void {\n    this.localStorage.mergeTags(name, tagHash, callback);\n  }\n\n  /**\n   * Change an existing package (i.e. unpublish one version)\n   Function changes a package info from local storage and all uplinks with write access./\n   Used storages: local (write)\n   */\n  public changePackage(\n    name: string,\n    metadata: Package,\n    revision: string,\n    callback: Callback\n  ): void {\n    this.localStorage.changePackage(name, metadata, revision, callback);\n  }\n\n  /**\n   * Remove a package from a system\n   Function removes a package from local storage\n   Used storages: local (write)\n   */\n  public removePackage(name: string, callback: Callback): void {\n    this.localStorage.removePackage(name, callback);\n    // update the indexer\n    Search.remove(name);\n  }\n\n  /**\n   Remove a tarball from a system\n   Function removes a tarball from local storage.\n   Tarball in question should not be linked to in any existing\n   versions, i.e. package version should be unpublished first.\n   Used storage: local (write)\n   */\n  public removeTarball(name: string, filename: string, revision: string, callback: Callback): void {\n    this.localStorage.removeTarball(name, filename, revision, callback);\n  }\n\n  /**\n   * Upload a tarball for {name} package\n   Function is synchronous and returns a WritableStream\n   Used storages: local (write)\n   */\n  public addTarball(name: string, filename: string): IUploadTarball {\n    return this.localStorage.addTarball(name, filename);\n  }\n\n  /**\n   Get a tarball from a storage for {name} package\n   Function is synchronous and returns a ReadableStream\n   Function tries to read tarball locally, if it fails then it reads package\n   information in order to figure out where we can get this tarball from\n   Used storages: local || uplink (just one)\n   */\n  public getTarball(name: string, filename: string): IReadTarball {\n    const readStream = new ReadTarball({});\n    readStream.abort = function () {};\n\n    const self = this;\n\n    // if someone requesting tarball, it means that we should already have some\n    // information about it, so fetching package info is unnecessary\n\n    // trying local first\n    // flow: should be IReadTarball\n    let localStream: any = self.localStorage.getTarball(name, filename);\n    let isOpen = false;\n    localStream.on('error', (err): any => {\n      if (isOpen || err.status !== HTTP_STATUS.NOT_FOUND) {\n        return readStream.emit('error', err);\n      }\n\n      // local reported 404\n      const err404 = err;\n      localStream.abort();\n      localStream = null; // we force for garbage collector\n      self.localStorage.getPackageMetadata(name, (err, info: Package): void => {\n        if (_.isNil(err) && info._distfiles && _.isNil(info._distfiles[filename]) === false) {\n          // information about this file exists locally\n          serveFile(info._distfiles[filename]);\n        } else {\n          // we know nothing about this file, trying to get information elsewhere\n          self._syncUplinksMetadata(name, info, {}, (err, info: Package): any => {\n            if (_.isNil(err) === false) {\n              return readStream.emit('error', err);\n            }\n            if (_.isNil(info._distfiles) || _.isNil(info._distfiles[filename])) {\n              return readStream.emit('error', err404);\n         