import fs from 'fs'; import Path from 'path'; import buildDebug from 'debug'; import _ from 'lodash'; import async from 'async'; import mkdirp from 'mkdirp'; import { Callback, Config, IPackageStorage, IPluginStorage, LocalStorage, Logger, StorageList, } from '@verdaccio/types'; import { getInternalError } from '@verdaccio/commons-api'; import LocalDriver, { noSuchFile } from './local-fs'; import { loadPrivatePackages } from './pkg-utils'; import TokenActions from './token'; import { _dbGenPath } from './utils'; const DB_NAME = '.verdaccio-db.json'; const debug = buildDebug('verdaccio:plugin:local-storage'); class LocalDatabase extends TokenActions implements IPluginStorage<{}> { public path: string; public logger: Logger; // @ts-ignore public data: LocalStorage; public config: Config; public locked: boolean; public constructor(config: Config, logger: Logger) { super(config); this.config = config; this.logger = logger; this.locked = false; this.path = _dbGenPath(DB_NAME, config); debug('plugin storage path %o', this.path); } public async init(): Promise { debug('plugin init'); this.data = await this._fetchLocalPackages(); this._sync(); } public getSecret(): Promise { return Promise.resolve(this.data.secret); } public setSecret(secret: string): Promise { return new Promise((resolve): void => { this.data.secret = secret; resolve(this._sync()); }); } public add(name: string, cb: Callback): void { if (this.data.list.indexOf(name) === -1) { this.data.list.push(name); debug('the private package %o has been added', name); cb(this._sync()); } else { debug('the private package %o was not added', name); cb(null); } } public search( onPackage: Callback, onEnd: Callback, validateName: (name: string) => boolean ): void { const storages = this._getCustomPackageLocalStorages(); debug(`search custom local packages: %o`, JSON.stringify(storages)); const base = Path.dirname(this.config.config_path); const self = this; const storageKeys = Object.keys(storages); debug(`search base: %o keys: %o`, base, storageKeys); async.eachSeries( storageKeys, function (storage, cb) { const position = storageKeys.indexOf(storage); const base2 = Path.join(position !== 0 ? storageKeys[0] : ''); const storagePath: string = Path.resolve(base, base2, storage); debug('search path: %o : %o', storagePath, storage); fs.readdir(storagePath, (err, files) => { if (err) { return cb(err); } async.eachSeries( files, function (file, cb) { debug('local-storage: [search] search file path: %o', file); if (storageKeys.includes(file)) { return cb(); } if (file.match(/^@/)) { // scoped const fileLocation = Path.resolve(base, storage, file); debug('search scoped file location: %o', fileLocation); fs.readdir(fileLocation, function (err, files) { if (err) { return cb(err); } async.eachSeries( files, (file2, cb) => { if (validateName(file2)) { const packagePath = Path.resolve(base, storage, file, file2); fs.stat(packagePath, (err, stats) => { if (_.isNil(err) === false) { return cb(err); } const item = { name: `${file}/${file2}`, path: packagePath, time: stats.mtime.getTime(), }; onPackage(item, cb); }); } else { cb(); } }, cb ); }); } else if (validateName(file)) { const base2 = Path.join(position !== 0 ? storageKeys[0] : ''); const packagePath = Path.resolve(base, base2, storage, file); debug('search file location: %o', packagePath); fs.stat(packagePath, (err, stats) => { if (_.isNil(err) === false) { return cb(err); } onPackage( { name: file, path: packagePath, time: self.getTime(stats.mtime.getTime(), stats.mtime), }, cb ); }); } else { cb(); } }, cb ); }); }, // @ts-ignore onEnd ); } public remove(name: string, cb: Callback): void { this.get((err, data) => { if (err) { cb(getInternalError('error remove private package')); this.logger.error( { err }, '[local-storage/remove]: remove the private package has failed @{err}' ); debug('error on remove package %o', name); } const pkgName = data.indexOf(name); if (pkgName !== -1) { this.data.list.splice(pkgName, 1); debug('remove package %o has been removed', name); } cb(this._sync()); }); } /** * Return all database elements. * @return {Array} */ public get(cb: Callback): void { const list = this.data.list; const totalItems = this.data.list.length; cb(null, list); debug('get full list of packages (%o) has been fetched', totalItems); } public getPackageStorage(packageName: string): IPackageStorage { const packageAccess = this.config.getMatchedPackagesSpec(packageName); const packagePath: string = this._getLocalStoragePath( packageAccess ? packageAccess.storage : undefined ); debug('storage path selected: ', packagePath); if (_.isString(packagePath) === false) { debug('the package %o has no storage defined ', packageName); return; } const packageStoragePath: string = Path.join( Path.resolve(Path.dirname(this.config.config_path || ''), packagePath), packageName ); debug('storage absolute path: ', packageStoragePath); return new LocalDriver(packageStoragePath, this.logger); } public clean(): void { this._sync(); } private getTime(time: number, mtime: Date): number | Date { return time ? time : mtime; } private _getCustomPackageLocalStorages(): object { const storages = {}; // add custom storage if exist if (this.config.storage) { storages[this.config.storage] = true; } const { packages } = this.config; if (packages) { const listPackagesConf = Object.keys(packages || {}); listPackagesConf.map((pkg) => { const storage = packages[pkg].storage; if (storage) { storages[storage] = false; } }); } return storages; } /** * Syncronize {create} database whether does not exist. * @return {Error|*} */ private _sync(): Error | null { debug('sync database started'); if (this.locked) { this.logger.error( 'Database is locked, please check error message printed during startup to ' + 'prevent data loss.' ); return new Error( 'Verdaccio database is locked, please contact your administrator to checkout ' + 'logs during verdaccio startup.' ); } // Uses sync to prevent ugly race condition try { // https://www.npmjs.com/package/mkdirp#mkdirpsyncdir-opts const folderName = Path.dirname(this.path); debug('creating folder %o', folderName); mkdirp.sync(folderName); debug('sync folder %o created succeed', folderName); } catch (err) { debug('sync create folder has failed with error: %o', err); return null; } try { fs.writeFileSync(this.path, JSON.stringify(this.data)); debug('sync write succeed'); return null; } catch (err) { debug('sync failed %o', err); return err; } } /** * Verify the right local storage location. * @param {String} path * @return {String} * @private */ private _getLocalStoragePath(storage: string | void): string { const globalConfigStorage = this.config ? this.config.storage : undefined; if (_.isNil(globalConfigStorage)) { throw new Error('global storage is required for this plugin'); } else { if (_.isNil(storage) === false && _.isString(storage)) { return Path.join(globalConfigStorage as string, storage as string); } return globalConfigStorage as string; } } /** * Fetch local packages. * @private * @return {Object} */ private async _fetchLocalPackages(): Promise { const list: StorageList = []; const emptyDatabase = { list, secret: '' }; try { return await loadPrivatePackages(this.path, this.logger); } catch (err) { // readFileSync is platform specific, macOS, Linux and Windows thrown an error // Only recreate if file not found to prevent data loss debug('error on fetch local packages %o', err); if (err.code !== noSuchFile) { this.locked = true; this.logger.error( 'Failed to read package database file, please check the error printed below:\n', `File Path: ${this.path}\n\n ${err.message}` ); } return emptyDatabase; } } } export default LocalDatabase;