UNPKG

48.1 kBJavaScriptView Raw
1"use strict";
2
3Object.defineProperty(exports, "__esModule", {
4 value: true
5});
6exports.default = void 0;
7
8var _fs = _interopRequireDefault(require("fs"));
9
10var _path = _interopRequireDefault(require("path"));
11
12var _lodash = _interopRequireDefault(require("lodash"));
13
14var _async = _interopRequireDefault(require("async"));
15
16var _mkdirp = _interopRequireDefault(require("mkdirp"));
17
18var _level = _interopRequireDefault(require("level"));
19
20var _lib = require("@verdaccio/commons-api/lib");
21
22var _localFs = _interopRequireWildcard(require("./local-fs"));
23
24var _pkgUtils = require("./pkg-utils");
25
26function _getRequireWildcardCache() { if (typeof WeakMap !== "function") return null; var cache = new WeakMap(); _getRequireWildcardCache = function () { return cache; }; return cache; }
27
28function _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
30function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
31
32function _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
34const DEPRECATED_DB_NAME = '.sinopia-db.json';
35const DB_NAME = '.verdaccio-db.json';
36const TOKEN_DB_NAME = '.token-db';
37
38/**
39 * Handle local database.
40 */
41class 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
491var _default = LocalDatabase;
492exports.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