1 | ;
|
2 |
|
3 | Object.defineProperty(exports, "__esModule", {
|
4 | value: true
|
5 | });
|
6 | exports.default = void 0;
|
7 |
|
8 | var _fs = _interopRequireDefault(require("fs"));
|
9 |
|
10 | var _path = _interopRequireDefault(require("path"));
|
11 |
|
12 | var _lodash = _interopRequireDefault(require("lodash"));
|
13 |
|
14 | var _async = _interopRequireDefault(require("async"));
|
15 |
|
16 | var _mkdirp = _interopRequireDefault(require("mkdirp"));
|
17 |
|
18 | var _level = _interopRequireDefault(require("level"));
|
19 |
|
20 | var _lib = require("@verdaccio/commons-api/lib");
|
21 |
|
22 | var _localFs = _interopRequireWildcard(require("./local-fs"));
|
23 |
|
24 | var _pkgUtils = require("./pkg-utils");
|
25 |
|
26 | function _getRequireWildcardCache() { if (typeof WeakMap !== "function") return null; var cache = new WeakMap(); _getRequireWildcardCache = function () { return cache; }; return cache; }
|
27 |
|
28 | function _interopRequireWildcard(obj) { if (obj && obj.__esModule) { return obj; } if (obj === null || typeof obj !== "object" && typeof obj !== "function") { return { default: obj }; } var cache = _getRequireWildcardCache(); if (cache && cache.has(obj)) { return cache.get(obj); } var newObj = {}; var hasPropertyDescriptor = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var key in obj) { if (Object.prototype.hasOwnProperty.call(obj, key)) { var desc = hasPropertyDescriptor ? Object.getOwnPropertyDescriptor(obj, key) : null; if (desc && (desc.get || desc.set)) { Object.defineProperty(newObj, key, desc); } else { newObj[key] = obj[key]; } } } newObj.default = obj; if (cache) { cache.set(obj, newObj); } return newObj; }
|
29 |
|
30 | function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
31 |
|
32 | 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; }
|
33 |
|
34 | const DEPRECATED_DB_NAME = '.sinopia-db.json';
|
35 | const DB_NAME = '.verdaccio-db.json';
|
36 | const TOKEN_DB_NAME = '.token-db';
|
37 |
|
38 | /**
|
39 | * Handle local database.
|
40 | */
|
41 | class LocalDatabase {
|
42 | /**
|
43 | * Load an parse the local json database.
|
44 | * @param {*} path the database path
|
45 | */
|
46 | constructor(config, logger) {
|
47 | _defineProperty(this, "path", void 0);
|
48 |
|
49 | _defineProperty(this, "logger", void 0);
|
50 |
|
51 | _defineProperty(this, "data", void 0);
|
52 |
|
53 | _defineProperty(this, "config", void 0);
|
54 |
|
55 | _defineProperty(this, "locked", void 0);
|
56 |
|
57 | _defineProperty(this, "tokenDb", void 0);
|
58 |
|
59 | this.config = config;
|
60 | this.path = this._buildStoragePath(config);
|
61 | this.logger = logger;
|
62 | this.locked = false;
|
63 | this.data = this._fetchLocalPackages();
|
64 | this.logger.trace({
|
65 | config: this.config
|
66 | }, '[local-storage]: configuration: @{config}');
|
67 |
|
68 | this._sync();
|
69 | }
|
70 |
|
71 | getSecret() {
|
72 | return Promise.resolve(this.data.secret);
|
73 | }
|
74 |
|
75 | setSecret(secret) {
|
76 | return new Promise(resolve => {
|
77 | this.data.secret = secret;
|
78 | resolve(this._sync());
|
79 | });
|
80 | }
|
81 | /**
|
82 | * Add a new element.
|
83 | * @param {*} name
|
84 | * @return {Error|*}
|
85 | */
|
86 |
|
87 |
|
88 | add(name, cb) {
|
89 | if (this.data.list.indexOf(name) === -1) {
|
90 | this.data.list.push(name);
|
91 | this.logger.debug({
|
92 | name
|
93 | }, '[local-storage]: the private package @{name} has been added');
|
94 | cb(this._sync());
|
95 | } else {
|
96 | cb(null);
|
97 | }
|
98 | }
|
99 |
|
100 | search(onPackage, onEnd, validateName) {
|
101 | const storages = this._getCustomPackageLocalStorages();
|
102 |
|
103 | this.logger.trace(`local-storage: [search]: ${JSON.stringify(storages)}`);
|
104 |
|
105 | const base = _path.default.dirname(this.config.self_path);
|
106 |
|
107 | const self = this;
|
108 | const storageKeys = Object.keys(storages);
|
109 | this.logger.trace(`local-storage: [search] base: ${base} keys ${storageKeys}`);
|
110 |
|
111 | _async.default.eachSeries(storageKeys, function (storage, cb) {
|
112 | const position = storageKeys.indexOf(storage);
|
113 |
|
114 | const base2 = _path.default.join(position !== 0 ? storageKeys[0] : '');
|
115 |
|
116 | const storagePath = _path.default.resolve(base, base2, storage);
|
117 |
|
118 | self.logger.trace({
|
119 | storagePath,
|
120 | storage
|
121 | }, 'local-storage: [search] search path: @{storagePath} : @{storage}');
|
122 |
|
123 | _fs.default.readdir(storagePath, (err, files) => {
|
124 | if (err) {
|
125 | return cb(err);
|
126 | }
|
127 |
|
128 | _async.default.eachSeries(files, function (file, cb) {
|
129 | self.logger.trace({
|
130 | file
|
131 | }, 'local-storage: [search] search file path: @{file}');
|
132 |
|
133 | if (storageKeys.includes(file)) {
|
134 | return cb();
|
135 | }
|
136 |
|
137 | if (file.match(/^@/)) {
|
138 | // scoped
|
139 | const fileLocation = _path.default.resolve(base, storage, file);
|
140 |
|
141 | self.logger.trace({
|
142 | fileLocation
|
143 | }, 'local-storage: [search] search scoped file location: @{fileLocation}');
|
144 |
|
145 | _fs.default.readdir(fileLocation, function (err, files) {
|
146 | if (err) {
|
147 | return cb(err);
|
148 | }
|
149 |
|
150 | _async.default.eachSeries(files, (file2, cb) => {
|
151 | if (validateName(file2)) {
|
152 | const packagePath = _path.default.resolve(base, storage, file, file2);
|
153 |
|
154 | _fs.default.stat(packagePath, (err, stats) => {
|
155 | if (_lodash.default.isNil(err) === false) {
|
156 | return cb(err);
|
157 | }
|
158 |
|
159 | const item = {
|
160 | name: `${file}/${file2}`,
|
161 | path: packagePath,
|
162 | time: stats.mtime.getTime()
|
163 | };
|
164 | onPackage(item, cb);
|
165 | });
|
166 | } else {
|
167 | cb();
|
168 | }
|
169 | }, cb);
|
170 | });
|
171 | } else if (validateName(file)) {
|
172 | const base2 = _path.default.join(position !== 0 ? storageKeys[0] : '');
|
173 |
|
174 | const packagePath = _path.default.resolve(base, base2, storage, file);
|
175 |
|
176 | self.logger.trace({
|
177 | packagePath
|
178 | }, 'local-storage: [search] search file location: @{packagePath}');
|
179 |
|
180 | _fs.default.stat(packagePath, (err, stats) => {
|
181 | if (_lodash.default.isNil(err) === false) {
|
182 | return cb(err);
|
183 | }
|
184 |
|
185 | onPackage({
|
186 | name: file,
|
187 | path: packagePath,
|
188 | time: self._getTime(stats.mtime.getTime(), stats.mtime)
|
189 | }, cb);
|
190 | });
|
191 | } else {
|
192 | cb();
|
193 | }
|
194 | }, cb);
|
195 | });
|
196 | }, onEnd);
|
197 | }
|
198 | /**
|
199 | * Remove an element from the database.
|
200 | * @param {*} name
|
201 | * @return {Error|*}
|
202 | */
|
203 |
|
204 |
|
205 | remove(name, cb) {
|
206 | this.get((err, data) => {
|
207 | if (err) {
|
208 | cb((0, _lib.getInternalError)('error remove private package'));
|
209 | this.logger.error({
|
210 | err
|
211 | }, '[local-storage/remove]: remove the private package has failed @{err}');
|
212 | }
|
213 |
|
214 | const pkgName = data.indexOf(name);
|
215 |
|
216 | if (pkgName !== -1) {
|
217 | this.data.list.splice(pkgName, 1);
|
218 | this.logger.trace({
|
219 | name
|
220 | }, 'local-storage: [remove] package @{name} has been removed');
|
221 | }
|
222 |
|
223 | cb(this._sync());
|
224 | });
|
225 | }
|
226 | /**
|
227 | * Return all database elements.
|
228 | * @return {Array}
|
229 | */
|
230 |
|
231 |
|
232 | get(cb) {
|
233 | const list = this.data.list;
|
234 | const totalItems = this.data.list.length;
|
235 | cb(null, list);
|
236 | this.logger.trace({
|
237 | totalItems
|
238 | }, 'local-storage: [get] full list of packages (@{totalItems}) has been fetched');
|
239 | }
|
240 |
|
241 | getPackageStorage(packageName) {
|
242 | const packageAccess = this.config.getMatchedPackagesSpec(packageName);
|
243 |
|
244 | const packagePath = this._getLocalStoragePath(packageAccess ? packageAccess.storage : undefined);
|
245 |
|
246 | this.logger.trace({
|
247 | packagePath
|
248 | }, '[local-storage/getPackageStorage]: storage selected: @{packagePath}');
|
249 |
|
250 | if (_lodash.default.isString(packagePath) === false) {
|
251 | this.logger.debug({
|
252 | name: packageName
|
253 | }, 'this package has no storage defined: @{name}');
|
254 | return;
|
255 | }
|
256 |
|
257 | const packageStoragePath = _path.default.join(_path.default.resolve(_path.default.dirname(this.config.self_path || ''), packagePath), packageName);
|
258 |
|
259 | this.logger.trace({
|
260 | packageStoragePath
|
261 | }, '[local-storage/getPackageStorage]: storage path: @{packageStoragePath}');
|
262 | return new _localFs.default(packageStoragePath, this.logger);
|
263 | }
|
264 |
|
265 | clean() {
|
266 | this._sync();
|
267 | }
|
268 |
|
269 | saveToken(token) {
|
270 | const key = this._getTokenKey(token);
|
271 |
|
272 | const db = this.getTokenDb();
|
273 | return new Promise((resolve, reject) => {
|
274 | db.put(key, token, err => {
|
275 | if (err) {
|
276 | reject(err);
|
277 | return;
|
278 | }
|
279 |
|
280 | resolve();
|
281 | });
|
282 | });
|
283 | }
|
284 |
|
285 | deleteToken(user, tokenKey) {
|
286 | const key = this._compoundTokenKey(user, tokenKey);
|
287 |
|
288 | const db = this.getTokenDb();
|
289 | return new Promise((resolve, reject) => {
|
290 | db.del(key, err => {
|
291 | if (err) {
|
292 | reject(err);
|
293 | return;
|
294 | }
|
295 |
|
296 | resolve();
|
297 | });
|
298 | });
|
299 | }
|
300 |
|
301 | readTokens(filter) {
|
302 | return new Promise((resolve, reject) => {
|
303 | const tokens = [];
|
304 | const key = filter.user + ':';
|
305 | const db = this.getTokenDb();
|
306 | const stream = db.createReadStream({
|
307 | gte: key,
|
308 | lte: String.fromCharCode(key.charCodeAt(0) + 1)
|
309 | });
|
310 | stream.on('data', data => {
|
311 | tokens.push(data.value);
|
312 | });
|
313 | stream.once('end', () => resolve(tokens));
|
314 | stream.once('error', err => reject(err));
|
315 | });
|
316 | }
|
317 |
|
318 | _getTime(time, mtime) {
|
319 | return time ? time : mtime;
|
320 | }
|
321 |
|
322 | _getCustomPackageLocalStorages() {
|
323 | const storages = {}; // add custom storage if exist
|
324 |
|
325 | if (this.config.storage) {
|
326 | storages[this.config.storage] = true;
|
327 | }
|
328 |
|
329 | const {
|
330 | packages
|
331 | } = this.config;
|
332 |
|
333 | if (packages) {
|
334 | const listPackagesConf = Object.keys(packages || {});
|
335 | listPackagesConf.map(pkg => {
|
336 | const storage = packages[pkg].storage;
|
337 |
|
338 | if (storage) {
|
339 | storages[storage] = false;
|
340 | }
|
341 | });
|
342 | }
|
343 |
|
344 | return storages;
|
345 | }
|
346 | /**
|
347 | * Syncronize {create} database whether does not exist.
|
348 | * @return {Error|*}
|
349 | */
|
350 |
|
351 |
|
352 | _sync() {
|
353 | this.logger.debug('[local-storage/_sync]: init sync database');
|
354 |
|
355 | if (this.locked) {
|
356 | this.logger.error('Database is locked, please check error message printed during startup to prevent data loss.');
|
357 | return new Error('Verdaccio database is locked, please contact your administrator to checkout logs during verdaccio startup.');
|
358 | } // Uses sync to prevent ugly race condition
|
359 |
|
360 |
|
361 | try {
|
362 | // https://www.npmjs.com/package/mkdirp#mkdirpsyncdir-opts
|
363 | const folderName = _path.default.dirname(this.path);
|
364 |
|
365 | _mkdirp.default.sync(folderName);
|
366 |
|
367 | this.logger.debug({
|
368 | folderName
|
369 | }, '[local-storage/_sync]: folder @{folderName} created succeed');
|
370 | } catch (err) {
|
371 | // perhaps a logger instance?
|
372 | this.logger.debug({
|
373 | err
|
374 | }, '[local-storage/_sync/mkdirp.sync]: sync failed @{err}');
|
375 | return null;
|
376 | }
|
377 |
|
378 | try {
|
379 | _fs.default.writeFileSync(this.path, JSON.stringify(this.data));
|
380 |
|
381 | this.logger.debug('[local-storage/_sync/writeFileSync]: sync write succeed');
|
382 | return null;
|
383 | } catch (err) {
|
384 | this.logger.debug({
|
385 | err
|
386 | }, '[local-storage/_sync/writeFileSync]: sync failed @{err}');
|
387 | return err;
|
388 | }
|
389 | }
|
390 | /**
|
391 | * Verify the right local storage location.
|
392 | * @param {String} path
|
393 | * @return {String}
|
394 | * @private
|
395 | */
|
396 |
|
397 |
|
398 | _getLocalStoragePath(storage) {
|
399 | const globalConfigStorage = this.config ? this.config.storage : undefined;
|
400 |
|
401 | if (_lodash.default.isNil(globalConfigStorage)) {
|
402 | throw new Error('global storage is required for this plugin');
|
403 | } else {
|
404 | if (_lodash.default.isNil(storage) === false && _lodash.default.isString(storage)) {
|
405 | return _path.default.join(globalConfigStorage, storage);
|
406 | }
|
407 |
|
408 | return globalConfigStorage;
|
409 | }
|
410 | }
|
411 | /**
|
412 | * Build the local database path.
|
413 | * @param {Object} config
|
414 | * @return {string|String|*}
|
415 | * @private
|
416 | */
|
417 |
|
418 |
|
419 | _buildStoragePath(config) {
|
420 | const sinopiadbPath = this._dbGenPath(DEPRECATED_DB_NAME, config);
|
421 |
|
422 | try {
|
423 | _fs.default.accessSync(sinopiadbPath, _fs.default.constants.F_OK);
|
424 |
|
425 | return sinopiadbPath;
|
426 | } catch (err) {
|
427 | if (err.code === _localFs.noSuchFile) {
|
428 | return this._dbGenPath(DB_NAME, config);
|
429 | }
|
430 |
|
431 | throw err;
|
432 | }
|
433 | }
|
434 |
|
435 | _dbGenPath(dbName, config) {
|
436 | return _path.default.join(_path.default.resolve(_path.default.dirname(config.self_path || ''), config.storage, dbName));
|
437 | }
|
438 | /**
|
439 | * Fetch local packages.
|
440 | * @private
|
441 | * @return {Object}
|
442 | */
|
443 |
|
444 |
|
445 | _fetchLocalPackages() {
|
446 | const list = [];
|
447 | const emptyDatabase = {
|
448 | list,
|
449 | secret: ''
|
450 | };
|
451 |
|
452 | try {
|
453 | const db = (0, _pkgUtils.loadPrivatePackages)(this.path, this.logger);
|
454 | return db;
|
455 | } catch (err) {
|
456 | // readFileSync is platform specific, macOS, Linux and Windows thrown an error
|
457 | // Only recreate if file not found to prevent data loss
|
458 | if (err.code !== _localFs.noSuchFile) {
|
459 | this.locked = true;
|
460 | this.logger.error('Failed to read package database file, please check the error printed below:\n', `File Path: ${this.path}\n\n ${err.message}`);
|
461 | }
|
462 |
|
463 | return emptyDatabase;
|
464 | }
|
465 | }
|
466 |
|
467 | getTokenDb() {
|
468 | if (!this.tokenDb) {
|
469 | this.tokenDb = (0, _level.default)(this._dbGenPath(TOKEN_DB_NAME, this.config), {
|
470 | valueEncoding: 'json'
|
471 | });
|
472 | }
|
473 |
|
474 | return this.tokenDb;
|
475 | }
|
476 |
|
477 | _getTokenKey(token) {
|
478 | const {
|
479 | user,
|
480 | key
|
481 | } = token;
|
482 | return this._compoundTokenKey(user, key);
|
483 | }
|
484 |
|
485 | _compoundTokenKey(user, key) {
|
486 | return `${user}:${key}`;
|
487 | }
|
488 |
|
489 | }
|
490 |
|
491 | var _default = LocalDatabase;
|
492 | exports.default = _default;
|
493 | //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/local-database.ts"],"names":["DEPRECATED_DB_NAME","DB_NAME","TOKEN_DB_NAME","LocalDatabase","constructor","config","logger","path","_buildStoragePath","locked","data","_fetchLocalPackages","trace","_sync","getSecret","Promise","resolve","secret","setSecret","add","name","cb","list","indexOf","push","debug","search","onPackage","onEnd","validateName","storages","_getCustomPackageLocalStorages","JSON","stringify","base","Path","dirname","self_path","self","storageKeys","Object","keys","async","eachSeries","storage","position","base2","join","storagePath","fs","readdir","err","files","file","includes","match","fileLocation","file2","packagePath","stat","stats","_","isNil","item","time","mtime","getTime","_getTime","remove","get","error","pkgName","splice","totalItems","length","getPackageStorage","packageName","packageAccess","getMatchedPackagesSpec","_getLocalStoragePath","undefined","isString","packageStoragePath","LocalDriver","clean","saveToken","token","key","_getTokenKey","db","getTokenDb","reject","put","deleteToken","user","tokenKey","_compoundTokenKey","del","readTokens","filter","tokens","stream","createReadStream","gte","lte","String","fromCharCode","charCodeAt","on","value","once","packages","listPackagesConf","map","pkg","Error","folderName","mkdirp","sync","writeFileSync","globalConfigStorage","sinopiadbPath","_dbGenPath","accessSync","constants","F_OK","code","noSuchFile","dbName","emptyDatabase","message","tokenDb","valueEncoding"],"mappings":"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","sourcesContent":["import fs from 'fs';\nimport Path from 'path';\nimport stream from 'stream';\n\nimport _ from 'lodash';\nimport async from 'async';\nimport mkdirp from 'mkdirp';\nimport {\n  Callback,\n  Config,\n  IPackageStorage,\n  IPluginStorage,\n  LocalStorage,\n  Logger,\n  StorageList,\n  Token,\n  TokenFilter,\n} from '@verdaccio/types';\nimport level from 'level';\nimport { getInternalError } from '@verdaccio/commons-api/lib';\n\nimport LocalDriver, { noSuchFile } from './local-fs';\nimport { loadPrivatePackages } from './pkg-utils';\n\nconst DEPRECATED_DB_NAME = '.sinopia-db.json';\nconst DB_NAME = '.verdaccio-db.json';\nconst TOKEN_DB_NAME = '.token-db';\n\ninterface Level {\n  put(key: string, token, fn?: Function): void;\n\n  get(key: string, fn?: Function): void;\n\n  del(key: string, fn?: Function): void;\n\n  createReadStream(options?: object): stream.Readable;\n}\n\n/**\n * Handle local database.\n */\nclass LocalDatabase implements IPluginStorage<{}> {\n  public path: string;\n  public logger: Logger;\n  public data: LocalStorage;\n  public config: Config;\n  public locked: boolean;\n  public tokenDb;\n\n  /**\n   * Load an parse the local json database.\n   * @param {*} path the database path\n   */\n  public constructor(config: Config, logger: Logger) {\n    this.config = config;\n    this.path = this._buildStoragePath(config);\n    this.logger = logger;\n    this.locked = false;\n    this.data = this._fetchLocalPackages();\n\n    this.logger.trace({ config: this.config }, '[local-storage]: configuration: @{config}');\n\n    this._sync();\n  }\n\n  public getSecret(): Promise<string> {\n    return Promise.resolve(this.data.secret);\n  }\n\n  public setSecret(secret: string): Promise<Error | null> {\n    return new Promise((resolve): void => {\n      this.data.secret = secret;\n\n      resolve(this._sync());\n    });\n  }\n\n  /**\n   * Add a new element.\n   * @param {*} name\n   * @return {Error|*}\n   */\n  public add(name: string, cb: Callback): void {\n    if (this.data.list.indexOf(name) === -1) {\n      this.data.list.push(name);\n\n      this.logger.debug({ name }, '[local-storage]: the private package @{name} has been added');\n      cb(this._sync());\n    } else {\n      cb(null);\n    }\n  }\n\n  public search(onPackage: Callback, onEnd: Callback, validateName: (name: string) => boolean): void {\n    const storages = this._getCustomPackageLocalStorages();\n    this.logger.trace(`local-storage: [search]: ${JSON.stringify(storages)}`);\n    const base = Path.dirname(this.config.self_path);\n    const self = this;\n    const storageKeys = Object.keys(storages);\n    this.logger.trace(`local-storage: [search] base: ${base} keys ${storageKeys}`);\n\n    async.eachSeries(\n      storageKeys,\n      function(storage, cb) {\n        const position = storageKeys.indexOf(storage);\n        const base2 = Path.join(position !== 0 ? storageKeys[0] : '');\n        const storagePath: string = Path.resolve(base, base2, storage);\n        self.logger.trace({ storagePath, storage }, 'local-storage: [search] search path: @{storagePath} : @{storage}');\n        fs.readdir(storagePath, (err, files) => {\n          if (err) {\n            return cb(err);\n          }\n\n          async.eachSeries(\n            files,\n            function(file, cb) {\n              self.logger.trace({ file }, 'local-storage: [search] search file path: @{file}');\n              if (storageKeys.includes(file)) {\n                return cb();\n              }\n\n              if (file.match(/^@/)) {\n                // scoped\n                const fileLocation = Path.resolve(base, storage, file);\n                self.logger.trace(\n                  { fileLocation },\n                  'local-storage: [search] search scoped file location: @{fileLocation}'\n                );\n                fs.readdir(fileLocation, function(err, files) {\n                  if (err) {\n                    return cb(err);\n                  }\n\n                  async.eachSeries(\n                    files,\n                    (file2, cb) => {\n                      if (validateName(file2)) {\n                        const packagePath = Path.resolve(base, storage, file, file2);\n\n                        fs.stat(packagePath, (err, stats) => {\n                          if (_.isNil(err) === false) {\n                            return cb(err);\n                          }\n                          const item = {\n                            name: `${file}/${file2}`,\n                            path: packagePath,\n                            time: stats.mtime.getTime(),\n                          };\n                          onPackage(item, cb);\n                        });\n                      } else {\n                        cb();\n                      }\n                    },\n                    cb\n                  );\n                });\n              } else if (validateName(file)) {\n                const base2 = Path.join(position !== 0 ? storageKeys[0] : '');\n                const packagePath = Path.resolve(base, base2, storage, file);\n                self.logger.trace({ packagePath }, 'local-storage: [search] search file location: @{packagePath}');\n                fs.stat(packagePath, (err, stats) => {\n                  if (_.isNil(err) === false) {\n                    return cb(err);\n                  }\n                  onPackage(\n                    {\n                      name: file,\n                      path: packagePath,\n                      time: self._getTime(stats.mtime.getTime(), stats.mtime),\n                    },\n                    cb\n                  );\n                });\n              } else {\n                cb();\n              }\n            },\n            cb\n          );\n        });\n      },\n      onEnd\n    );\n  }\n\n  /**\n   * Remove an element from the database.\n   * @param {*} name\n   * @return {Error|*}\n   */\n  public remove(name: string, cb: Callback): void {\n    this.get((err, data) => {\n      if (err) {\n        cb(getInternalError('error remove private package'));\n        this.logger.error({ err }, '[local-storage/remove]: remove the private package has failed @{err}');\n      }\n\n      const pkgName = data.indexOf(name);\n      if (pkgName !== -1) {\n        this.data.list.splice(pkgName, 1);\n\n        this.logger.trace({ name }, 'local-storage: [remove] package @{name} has been removed');\n      }\n\n      cb(this._sync());\n    });\n  }\n\n  /**\n   * Return all database elements.\n   * @return {Array}\n   */\n  public get(cb: Callback): void {\n    const list = this.data.list;\n    const totalItems = this.data.list.length;\n\n    cb(null, list);\n\n    this.logger.trace({ totalItems }, 'local-storage: [get] full list of packages (@{totalItems}) has been fetched');\n  }\n\n  public getPackageStorage(packageName: string): IPackageStorage {\n    const packageAccess = this.config.getMatchedPackagesSpec(packageName);\n\n    const packagePath: string = this._getLocalStoragePath(packageAccess ? packageAccess.storage : undefined);\n    this.logger.trace({ packagePath }, '[local-storage/getPackageStorage]: storage selected: @{packagePath}');\n\n    if (_.isString(packagePath) === false) {\n      this.logger.debug({ name: packageName }, 'this package has no storage defined: @{name}');\n      return;\n    }\n\n    const packageStoragePath: string = Path.join(\n      Path.resolve(Path.dirname(this.config.self_path || ''), packagePath),\n      packageName\n    );\n\n    this.logger.trace({ packageStoragePath }, '[local-storage/getPackageStorage]: storage path: @{packageStoragePath}');\n\n    return new LocalDriver(packageStoragePath, this.logger);\n  }\n\n  public clean(): void {\n    this._sync();\n  }\n\n  public saveToken(token: Token): Promise<void> {\n    const key = this._getTokenKey(token);\n    const db = this.getTokenDb();\n\n    return new Promise((resolve, reject): void => {\n      db.put(key, token, err => {\n        if (err) {\n          reject(err);\n          return;\n        }\n        resolve();\n      });\n    });\n  }\n\n  public deleteToken(user: string, tokenKey: string): Promise<void> {\n    const key = this._compoundTokenKey(user, tokenKey);\n    const db = this.getTokenDb();\n    return new Promise((resolve, reject): void => {\n      db.del(key, err => {\n        if (err) {\n          reject(err);\n          return;\n        }\n        resolve();\n      });\n    });\n  }\n\n  public readTokens(filter: TokenFilter): Promise<Token[]> {\n    return new Promise((resolve, reject): void => {\n      const tokens: Token[] = [];\n      const key = filter.user + ':';\n      const db = this.getTokenDb();\n      const stream = db.createReadStream({\n        gte: key,\n        lte: String.fromCharCode(key.charCodeAt(0) + 1),\n      });\n\n      stream.on('data', data => {\n        tokens.push(data.value);\n      });\n\n      stream.once('end', () => resolve(tokens));\n\n      stream.once('error', err => reject(err));\n    });\n  }\n\n  private _getTime(time: number, mtime: Date): number | Date {\n    return time ? time : mtime;\n  }\n\n  private _getCustomPackageLocalStorages(): object {\n    const storages = {};\n\n    // add custom storage if exist\n    if (this.config.storage) {\n      storages[this.config.storage] = true;\n    }\n\n    const { packages } = this.config;\n\n    if (packages) {\n      const listPackagesConf = Object.keys(packages || {});\n\n      listPackagesConf.map(pkg => {\n        const storage = packages[pkg].storage;\n        if (storage) {\n          storages[storage] = false;\n        }\n      });\n    }\n\n    return storages;\n  }\n\n  /**\n   * Syncronize {create} database whether does not exist.\n   * @return {Error|*}\n   */\n  private _sync(): Error | null {\n    this.logger.debug('[local-storage/_sync]: init sync database');\n\n    if (this.locked) {\n      this.logger.error('Database is locked, please check error message printed during startup to prevent data loss.');\n      return new Error(\n        'Verdaccio database is locked, please contact your administrator to checkout logs during verdaccio startup.'\n      );\n    }\n    // Uses sync to prevent ugly race condition\n    try {\n      // https://www.npmjs.com/package/mkdirp#mkdirpsyncdir-opts\n      const folderName = Path.dirname(this.path);\n      mkdirp.sync(folderName);\n      this.logger.debug({ folderName }, '[local-storage/_sync]: folder @{folderName} created succeed');\n    } catch (err) {\n      // perhaps a logger instance?\n      this.logger.debug({ err }, '[local-storage/_sync/mkdirp.sync]: sync failed @{err}');\n\n      return null;\n    }\n\n    try {\n      fs.writeFileSync(this.path, JSON.stringify(this.data));\n      this.logger.debug('[local-storage/_sync/writeFileSync]: sync write succeed');\n\n      return null;\n    } catch (err) {\n      this.logger.debug({ err }, '[local-storage/_sync/writeFileSync]: sync failed @{err}');\n\n      return err;\n    }\n  }\n\n  /**\n   * Verify the right local storage location.\n   * @param {String} path\n   * @return {String}\n   * @private\n   */\n  private _getLocalStoragePath(storage: string | void): string {\n    const globalConfigStorage = this.config ? this.config.storage : undefined;\n    if (_.isNil(globalConfigStorage)) {\n      throw new Error('global storage is required for this plugin');\n    } else {\n      if (_.isNil(storage) === false && _.isString(storage)) {\n        return Path.join(globalConfigStorage as string, storage as string);\n      }\n\n      return globalConfigStorage as string;\n    }\n  }\n\n  /**\n   * Build the local database path.\n   * @param {Object} config\n   * @return {string|String|*}\n   * @private\n   */\n  private _buildStoragePath(config: Config): string {\n    const sinopiadbPath: string = this._dbGenPath(DEPRECATED_DB_NAME, config);\n    try {\n      fs.accessSync(sinopiadbPath, fs.constants.F_OK);\n      return sinopiadbPath;\n    } catch (err) {\n      if (err.code === noSuchFile) {\n        return this._dbGenPath(DB_NAME, config);\n      }\n\n      throw err;\n    }\n  }\n\n  private _dbGenPath(dbName: string, config: Config): string {\n    return Path.join(Path.resolve(Path.dirname(config.self_path || ''), config.storage as string, dbName));\n  }\n\n  /**\n   * Fetch local packages.\n   * @private\n   * @return {Object}\n   */\n  private _fetchLocalPackages(): LocalStorage {\n    const list: StorageList = [];\n    const emptyDatabase = { list, secret: '' };\n\n    try {\n      const db = loadPrivatePackages(this.path, this.logger);\n\n      return db;\n    } catch (err) {\n      // readFileSync is platform specific, macOS, Linux and Windows thrown an error\n      // Only recreate if file not found to prevent data loss\n      if (err.code !== noSuchFile) {\n        this.locked = true;\n        this.logger.error(\n          'Failed to read package database file, please check the error printed below:\\n',\n          `File Path: ${this.path}\\n\\n ${err.message}`\n        );\n      }\n\n      return emptyDatabase;\n    }\n  }\n\n  private getTokenDb(): Level {\n    if (!this.tokenDb) {\n      this.tokenDb = level(this._dbGenPath(TOKEN_DB_NAME, this.config), {\n        valueEncoding: 'json',\n      });\n    }\n\n    return this.tokenDb;\n  }\n\n  private _getTokenKey(token: Token): string {\n    const { user, key } = token;\n    return this._compoundTokenKey(user, key);\n  }\n\n  private _compoundTokenKey(user: string, key: string): string {\n    return `${user}:${key}`;\n  }\n}\n\nexport default LocalDatabase;\n"]} |
\ | No newline at end of file |