1 | ;
|
2 |
|
3 | Object.defineProperty(exports, "__esModule", {
|
4 | value: true
|
5 | });
|
6 | exports.default = exports.fSError = exports.pkgFileName = exports.resourceNotAvailable = exports.noSuchFile = exports.fileExist = 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 _mkdirp = _interopRequireDefault(require("mkdirp"));
|
15 |
|
16 | var _streams = require("@verdaccio/streams");
|
17 |
|
18 | var _fileLocking = require("@verdaccio/file-locking");
|
19 |
|
20 | var _lib = require("@verdaccio/commons-api/lib");
|
21 |
|
22 | function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
23 |
|
24 | 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; }
|
25 |
|
26 | const fileExist = 'EEXISTS';
|
27 | exports.fileExist = fileExist;
|
28 | const noSuchFile = 'ENOENT';
|
29 | exports.noSuchFile = noSuchFile;
|
30 | const resourceNotAvailable = 'EAGAIN';
|
31 | exports.resourceNotAvailable = resourceNotAvailable;
|
32 | const pkgFileName = 'package.json';
|
33 | exports.pkgFileName = pkgFileName;
|
34 |
|
35 | const fSError = function (message, code = 409) {
|
36 | const err = (0, _lib.getCode)(code, message); // FIXME: we should return http-status codes here instead, future improvement
|
37 | // @ts-ignore
|
38 |
|
39 | err.code = message;
|
40 | return err;
|
41 | };
|
42 |
|
43 | exports.fSError = fSError;
|
44 |
|
45 | const tempFile = function (str) {
|
46 | return `${str}.tmp${String(Math.random()).substr(2)}`;
|
47 | };
|
48 |
|
49 | const renameTmp = function (src, dst, _cb) {
|
50 | const cb = err => {
|
51 | if (err) {
|
52 | _fs.default.unlink(src, () => {});
|
53 | }
|
54 |
|
55 | _cb(err);
|
56 | };
|
57 |
|
58 | if (process.platform !== 'win32') {
|
59 | return _fs.default.rename(src, dst, cb);
|
60 | } // windows can't remove opened file,
|
61 | // but it seem to be able to rename it
|
62 |
|
63 |
|
64 | const tmp = tempFile(dst);
|
65 |
|
66 | _fs.default.rename(dst, tmp, function (err) {
|
67 | _fs.default.rename(src, dst, cb);
|
68 |
|
69 | if (!err) {
|
70 | _fs.default.unlink(tmp, () => {});
|
71 | }
|
72 | });
|
73 | };
|
74 |
|
75 | class LocalFS {
|
76 | constructor(path, logger) {
|
77 | _defineProperty(this, "path", void 0);
|
78 |
|
79 | _defineProperty(this, "logger", void 0);
|
80 |
|
81 | this.path = path;
|
82 | this.logger = logger;
|
83 | }
|
84 | /**
|
85 | * This function allows to update the package thread-safely
|
86 | Algorithm:
|
87 | 1. lock package.json for writing
|
88 | 2. read package.json
|
89 | 3. updateFn(pkg, cb), and wait for cb
|
90 | 4. write package.json.tmp
|
91 | 5. move package.json.tmp package.json
|
92 | 6. callback(err?)
|
93 | * @param {*} name
|
94 | * @param {*} updateHandler
|
95 | * @param {*} onWrite
|
96 | * @param {*} transformPackage
|
97 | * @param {*} onEnd
|
98 | */
|
99 |
|
100 |
|
101 | updatePackage(name, updateHandler, onWrite, transformPackage, onEnd) {
|
102 | this._lockAndReadJSON(pkgFileName, (err, json) => {
|
103 | let locked = false;
|
104 | const self = this; // callback that cleans up lock first
|
105 |
|
106 | const unLockCallback = function (lockError) {
|
107 | // eslint-disable-next-line prefer-rest-params
|
108 | const _args = arguments;
|
109 |
|
110 | if (locked) {
|
111 | self._unlockJSON(pkgFileName, () => {
|
112 | // ignore any error from the unlock
|
113 | if (lockError !== null) {
|
114 | self.logger.trace({
|
115 | name,
|
116 | lockError
|
117 | }, '[local-storage/updatePackage/unLockCallback] file: @{name} lock has failed lockError: @{lockError}');
|
118 | }
|
119 |
|
120 | onEnd.apply(lockError, _args);
|
121 | });
|
122 | } else {
|
123 | self.logger.trace({
|
124 | name
|
125 | }, '[local-storage/updatePackage/unLockCallback] file: @{name} has been updated');
|
126 | onEnd(..._args);
|
127 | }
|
128 | };
|
129 |
|
130 | if (!err) {
|
131 | locked = true;
|
132 | this.logger.trace({
|
133 | name
|
134 | }, '[local-storage/updatePackage] file: @{name} has been locked');
|
135 | }
|
136 |
|
137 | if (_lodash.default.isNil(err) === false) {
|
138 | if (err.code === resourceNotAvailable) {
|
139 | return unLockCallback((0, _lib.getInternalError)('resource temporarily unavailable'));
|
140 | } else if (err.code === noSuchFile) {
|
141 | return unLockCallback((0, _lib.getNotFound)());
|
142 | } else {
|
143 | return unLockCallback(err);
|
144 | }
|
145 | }
|
146 |
|
147 | updateHandler(json, err => {
|
148 | if (err) {
|
149 | return unLockCallback(err);
|
150 | }
|
151 |
|
152 | onWrite(name, transformPackage(json), unLockCallback);
|
153 | });
|
154 | });
|
155 | }
|
156 |
|
157 | deletePackage(packageName, callback) {
|
158 | this.logger.debug({
|
159 | packageName
|
160 | }, '[local-storage/deletePackage] delete a package @{packageName}');
|
161 | return _fs.default.unlink(this._getStorage(packageName), callback);
|
162 | }
|
163 |
|
164 | removePackage(callback) {
|
165 | this.logger.debug({
|
166 | packageName: this.path
|
167 | }, '[local-storage/removePackage] remove a package: @{packageName}');
|
168 |
|
169 | _fs.default.rmdir(this._getStorage('.'), callback);
|
170 | }
|
171 |
|
172 | createPackage(name, value, cb) {
|
173 | this.logger.debug({
|
174 | packageName: name
|
175 | }, '[local-storage/createPackage] create a package: @{packageName}');
|
176 |
|
177 | this._createFile(this._getStorage(pkgFileName), this._convertToString(value), cb);
|
178 | }
|
179 |
|
180 | savePackage(name, value, cb) {
|
181 | this.logger.debug({
|
182 | packageName: name
|
183 | }, '[local-storage/savePackage] save a package: @{packageName}');
|
184 |
|
185 | this._writeFile(this._getStorage(pkgFileName), this._convertToString(value), cb);
|
186 | }
|
187 |
|
188 | readPackage(name, cb) {
|
189 | this.logger.debug({
|
190 | packageName: name
|
191 | }, '[local-storage/readPackage] read a package: @{packageName}');
|
192 |
|
193 | this._readStorageFile(this._getStorage(pkgFileName)).then(res => {
|
194 | try {
|
195 | const data = JSON.parse(res.toString('utf8'));
|
196 | this.logger.trace({
|
197 | packageName: name
|
198 | }, '[local-storage/readPackage/_readStorageFile] read a package succeed: @{packageName}');
|
199 | cb(null, data);
|
200 | } catch (err) {
|
201 | this.logger.trace({
|
202 | err
|
203 | }, '[local-storage/readPackage/_readStorageFile] error on read a package: @{err}');
|
204 | cb(err);
|
205 | }
|
206 | }, err => {
|
207 | this.logger.trace({
|
208 | err
|
209 | }, '[local-storage/readPackage/_readStorageFile] error on read a package: @{err}');
|
210 | return cb(err);
|
211 | });
|
212 | }
|
213 |
|
214 | writeTarball(name) {
|
215 | const uploadStream = new _streams.UploadTarball({});
|
216 | this.logger.debug({
|
217 | packageName: name
|
218 | }, '[local-storage/writeTarball] write a tarball for package: @{packageName}');
|
219 | let _ended = 0;
|
220 | uploadStream.on('end', function () {
|
221 | _ended = 1;
|
222 | });
|
223 |
|
224 | const pathName = this._getStorage(name);
|
225 |
|
226 | _fs.default.access(pathName, fileNotFound => {
|
227 | const exists = !fileNotFound;
|
228 |
|
229 | if (exists) {
|
230 | uploadStream.emit('error', fSError(fileExist));
|
231 | } else {
|
232 | const temporalName = _path.default.join(this.path, `${name}.tmp-${String(Math.random()).replace(/^0\./, '')}`);
|
233 |
|
234 | const file = _fs.default.createWriteStream(temporalName);
|
235 |
|
236 | const removeTempFile = () => _fs.default.unlink(temporalName, () => {});
|
237 |
|
238 | let opened = false;
|
239 | uploadStream.pipe(file);
|
240 |
|
241 | uploadStream.done = function () {
|
242 | const onend = function () {
|
243 | file.on('close', function () {
|
244 | renameTmp(temporalName, pathName, function (err) {
|
245 | if (err) {
|
246 | uploadStream.emit('error', err);
|
247 | } else {
|
248 | uploadStream.emit('success');
|
249 | }
|
250 | });
|
251 | });
|
252 | file.end();
|
253 | };
|
254 |
|
255 | if (_ended) {
|
256 | onend();
|
257 | } else {
|
258 | uploadStream.on('end', onend);
|
259 | }
|
260 | };
|
261 |
|
262 | uploadStream.abort = function () {
|
263 | if (opened) {
|
264 | opened = false;
|
265 | file.on('close', function () {
|
266 | removeTempFile();
|
267 | });
|
268 | } else {
|
269 | // if the file does not recieve any byte never is opened and has to be removed anyway.
|
270 | removeTempFile();
|
271 | }
|
272 |
|
273 | file.end();
|
274 | };
|
275 |
|
276 | file.on('open', function () {
|
277 | opened = true; // re-emitting open because it's handled in storage.js
|
278 |
|
279 | uploadStream.emit('open');
|
280 | });
|
281 | file.on('error', function (err) {
|
282 | uploadStream.emit('error', err);
|
283 | });
|
284 | }
|
285 | });
|
286 |
|
287 | return uploadStream;
|
288 | }
|
289 |
|
290 | readTarball(name) {
|
291 | const pathName = this._getStorage(name);
|
292 |
|
293 | this.logger.debug({
|
294 | packageName: name
|
295 | }, '[local-storage/readTarball] read a tarball for package: @{packageName}');
|
296 | const readTarballStream = new _streams.ReadTarball({});
|
297 |
|
298 | const readStream = _fs.default.createReadStream(pathName);
|
299 |
|
300 | readStream.on('error', function (err) {
|
301 | readTarballStream.emit('error', err);
|
302 | });
|
303 | readStream.on('open', function (fd) {
|
304 | _fs.default.fstat(fd, function (err, stats) {
|
305 | if (_lodash.default.isNil(err) === false) {
|
306 | return readTarballStream.emit('error', err);
|
307 | }
|
308 |
|
309 | readTarballStream.emit('content-length', stats.size);
|
310 | readTarballStream.emit('open');
|
311 | readStream.pipe(readTarballStream);
|
312 | });
|
313 | });
|
314 |
|
315 | readTarballStream.abort = function () {
|
316 | readStream.close();
|
317 | };
|
318 |
|
319 | return readTarballStream;
|
320 | }
|
321 |
|
322 | _createFile(name, contents, callback) {
|
323 | this.logger.trace({
|
324 | name
|
325 | }, '[local-storage/_createFile] create a new file: @{name}');
|
326 |
|
327 | _fs.default.open(name, 'wx', err => {
|
328 | if (err) {
|
329 | // native EEXIST used here to check exception on fs.open
|
330 | if (err.code === 'EEXIST') {
|
331 | this.logger.trace({
|
332 | name
|
333 | }, '[local-storage/_createFile] file cannot be created, it already exists: @{name}');
|
334 | return callback(fSError(fileExist));
|
335 | }
|
336 | }
|
337 |
|
338 | this._writeFile(name, contents, callback);
|
339 | });
|
340 | }
|
341 |
|
342 | _readStorageFile(name) {
|
343 | return new Promise((resolve, reject) => {
|
344 | this.logger.trace({
|
345 | name
|
346 | }, '[local-storage/_readStorageFile] read a file: @{name}');
|
347 |
|
348 | _fs.default.readFile(name, (err, data) => {
|
349 | if (err) {
|
350 | this.logger.trace({
|
351 | err
|
352 | }, '[local-storage/_readStorageFile] error on read the file: @{name}');
|
353 | reject(err);
|
354 | } else {
|
355 | this.logger.trace({
|
356 | name
|
357 | }, '[local-storage/_readStorageFile] read file succeed: @{name}');
|
358 | resolve(data);
|
359 | }
|
360 | });
|
361 | });
|
362 | }
|
363 |
|
364 | _convertToString(value) {
|
365 | return JSON.stringify(value, null, '\t');
|
366 | }
|
367 |
|
368 | _getStorage(fileName = '') {
|
369 | const storagePath = _path.default.join(this.path, fileName);
|
370 |
|
371 | return storagePath;
|
372 | }
|
373 |
|
374 | _writeFile(dest, data, cb) {
|
375 | const createTempFile = cb => {
|
376 | const tempFilePath = tempFile(dest);
|
377 |
|
378 | _fs.default.writeFile(tempFilePath, data, err => {
|
379 | if (err) {
|
380 | this.logger.trace({
|
381 | name: dest
|
382 | }, '[local-storage/_writeFile] new file: @{name} has been created');
|
383 | return cb(err);
|
384 | }
|
385 |
|
386 | this.logger.trace({
|
387 | name: dest
|
388 | }, '[local-storage/_writeFile] creating a new file: @{name}');
|
389 | renameTmp(tempFilePath, dest, cb);
|
390 | });
|
391 | };
|
392 |
|
393 | createTempFile(err => {
|
394 | if (err && err.code === noSuchFile) {
|
395 | (0, _mkdirp.default)(_path.default.dirname(dest), function (err) {
|
396 | if (err) {
|
397 | return cb(err);
|
398 | }
|
399 |
|
400 | createTempFile(cb);
|
401 | });
|
402 | } else {
|
403 | cb(err);
|
404 | }
|
405 | });
|
406 | }
|
407 |
|
408 | _lockAndReadJSON(name, cb) {
|
409 | const fileName = this._getStorage(name);
|
410 |
|
411 | (0, _fileLocking.readFile)(fileName, {
|
412 | lock: true,
|
413 | parse: true
|
414 | }, (err, res) => {
|
415 | if (err) {
|
416 | this.logger.trace({
|
417 | name
|
418 | }, '[local-storage/_lockAndReadJSON] read new file: @{name} has failed');
|
419 | return cb(err);
|
420 | }
|
421 |
|
422 | this.logger.trace({
|
423 | name
|
424 | }, '[local-storage/_lockAndReadJSON] file: @{name} read');
|
425 | return cb(null, res);
|
426 | });
|
427 | }
|
428 |
|
429 | _unlockJSON(name, cb) {
|
430 | (0, _fileLocking.unlockFile)(this._getStorage(name), cb);
|
431 | }
|
432 |
|
433 | }
|
434 |
|
435 | exports.default = LocalFS;
|
436 | //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/local-fs.ts"],"names":["fileExist","noSuchFile","resourceNotAvailable","pkgFileName","fSError","message","code","err","tempFile","str","String","Math","random","substr","renameTmp","src","dst","_cb","cb","fs","unlink","process","platform","rename","tmp","LocalFS","constructor","path","logger","updatePackage","name","updateHandler","onWrite","transformPackage","onEnd","_lockAndReadJSON","json","locked","self","unLockCallback","lockError","_args","arguments","_unlockJSON","trace","apply","_","isNil","deletePackage","packageName","callback","debug","_getStorage","removePackage","rmdir","createPackage","value","_createFile","_convertToString","savePackage","_writeFile","readPackage","_readStorageFile","then","res","data","JSON","parse","toString","writeTarball","uploadStream","UploadTarball","_ended","on","pathName","access","fileNotFound","exists","emit","temporalName","join","replace","file","createWriteStream","removeTempFile","opened","pipe","done","onend","end","abort","readTarball","readTarballStream","ReadTarball","readStream","createReadStream","fd","fstat","stats","size","close","contents","open","Promise","resolve","reject","readFile","stringify","fileName","storagePath","dest","createTempFile","tempFilePath","writeFile","dirname","lock"],"mappings":"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","sourcesContent":["import fs from 'fs';\nimport path from 'path';\n\nimport _ from 'lodash';\nimport mkdirp from 'mkdirp';\nimport { UploadTarball, ReadTarball } from '@verdaccio/streams';\nimport { unlockFile, readFile } from '@verdaccio/file-locking';\nimport { Callback, Logger, Package, ILocalPackageManager, IUploadTarball } from '@verdaccio/types';\nimport { getCode, getInternalError, getNotFound, VerdaccioError } from '@verdaccio/commons-api/lib';\n\nexport const fileExist = 'EEXISTS';\nexport const noSuchFile = 'ENOENT';\nexport const resourceNotAvailable = 'EAGAIN';\nexport const pkgFileName = 'package.json';\n\nexport const fSError = function(message: string, code = 409): VerdaccioError {\n  const err: VerdaccioError = getCode(code, message);\n  // FIXME: we should return http-status codes here instead, future improvement\n  // @ts-ignore\n  err.code = message;\n\n  return err;\n};\n\nconst tempFile = function(str): string {\n  return `${str}.tmp${String(Math.random()).substr(2)}`;\n};\n\nconst renameTmp = function(src, dst, _cb): void {\n  const cb = (err): void => {\n    if (err) {\n      fs.unlink(src, () => {});\n    }\n    _cb(err);\n  };\n\n  if (process.platform !== 'win32') {\n    return fs.rename(src, dst, cb);\n  }\n\n  // windows can't remove opened file,\n  // but it seem to be able to rename it\n  const tmp = tempFile(dst);\n  fs.rename(dst, tmp, function(err) {\n    fs.rename(src, dst, cb);\n    if (!err) {\n      fs.unlink(tmp, () => {});\n    }\n  });\n};\n\nexport type ILocalFSPackageManager = ILocalPackageManager & { path: string };\n\nexport default class LocalFS implements ILocalFSPackageManager {\n  public path: string;\n  public logger: Logger;\n\n  public constructor(path: string, logger: Logger) {\n    this.path = path;\n    this.logger = logger;\n  }\n\n  /**\n    *  This function allows to update the package thread-safely\n      Algorithm:\n      1. lock package.json for writing\n      2. read package.json\n      3. updateFn(pkg, cb), and wait for cb\n      4. write package.json.tmp\n      5. move package.json.tmp package.json\n      6. callback(err?)\n    * @param {*} name\n    * @param {*} updateHandler\n    * @param {*} onWrite\n    * @param {*} transformPackage\n    * @param {*} onEnd\n    */\n  public updatePackage(\n    name: string,\n    updateHandler: Callback,\n    onWrite: Callback,\n    transformPackage: Function,\n    onEnd: Callback\n  ): void {\n    this._lockAndReadJSON(pkgFileName, (err, json) => {\n      let locked = false;\n      const self = this;\n      // callback that cleans up lock first\n      const unLockCallback = function(lockError: Error): void {\n        // eslint-disable-next-line prefer-rest-params\n        const _args = arguments;\n\n        if (locked) {\n          self._unlockJSON(pkgFileName, () => {\n            // ignore any error from the unlock\n            if (lockError !== null) {\n              self.logger.trace(\n                {\n                  name,\n                  lockError,\n                },\n                '[local-storage/updatePackage/unLockCallback] file: @{name} lock has failed lockError: @{lockError}'\n              );\n            }\n\n            onEnd.apply(lockError, _args);\n          });\n        } else {\n          self.logger.trace({ name }, '[local-storage/updatePackage/unLockCallback] file: @{name} has been updated');\n\n          onEnd(..._args);\n        }\n      };\n\n      if (!err) {\n        locked = true;\n        this.logger.trace(\n          {\n            name,\n          },\n          '[local-storage/updatePackage] file: @{name} has been locked'\n        );\n      }\n\n      if (_.isNil(err) === false) {\n        if (err.code === resourceNotAvailable) {\n          return unLockCallback(getInternalError('resource temporarily unavailable'));\n        } else if (err.code === noSuchFile) {\n          return unLockCallback(getNotFound());\n        } else {\n          return unLockCallback(err);\n        }\n      }\n\n      updateHandler(json, err => {\n        if (err) {\n          return unLockCallback(err);\n        }\n\n        onWrite(name, transformPackage(json), unLockCallback);\n      });\n    });\n  }\n\n  public deletePackage(packageName: string, callback: (err: NodeJS.ErrnoException | null) => void): void {\n    this.logger.debug({ packageName }, '[local-storage/deletePackage] delete a package @{packageName}');\n\n    return fs.unlink(this._getStorage(packageName), callback);\n  }\n\n  public removePackage(callback: (err: NodeJS.ErrnoException | null) => void): void {\n    this.logger.debug({ packageName: this.path }, '[local-storage/removePackage] remove a package: @{packageName}');\n\n    fs.rmdir(this._getStorage('.'), callback);\n  }\n\n  public createPackage(name: string, value: Package, cb: Callback): void {\n    this.logger.debug({ packageName: name }, '[local-storage/createPackage] create a package: @{packageName}');\n\n    this._createFile(this._getStorage(pkgFileName), this._convertToString(value), cb);\n  }\n\n  public savePackage(name: string, value: Package, cb: Callback): void {\n    this.logger.debug({ packageName: name }, '[local-storage/savePackage] save a package: @{packageName}');\n\n    this._writeFile(this._getStorage(pkgFileName), this._convertToString(value), cb);\n  }\n\n  public readPackage(name: string, cb: Callback): void {\n    this.logger.debug({ packageName: name }, '[local-storage/readPackage] read a package: @{packageName}');\n\n    this._readStorageFile(this._getStorage(pkgFileName)).then(\n      res => {\n        try {\n          const data: any = JSON.parse(res.toString('utf8'));\n\n          this.logger.trace(\n            { packageName: name },\n            '[local-storage/readPackage/_readStorageFile] read a package succeed: @{packageName}'\n          );\n          cb(null, data);\n        } catch (err) {\n          this.logger.trace({ err }, '[local-storage/readPackage/_readStorageFile] error on read a package: @{err}');\n          cb(err);\n        }\n      },\n      err => {\n        this.logger.trace({ err }, '[local-storage/readPackage/_readStorageFile] error on read a package: @{err}');\n\n        return cb(err);\n      }\n    );\n  }\n\n  public writeTarball(name: string): IUploadTarball {\n    const uploadStream = new UploadTarball({});\n    this.logger.debug(\n      { packageName: name },\n      '[local-storage/writeTarball] write a tarball for package: @{packageName}'\n    );\n\n    let _ended = 0;\n    uploadStream.on('end', function() {\n      _ended = 1;\n    });\n\n    const pathName: string = this._getStorage(name);\n\n    fs.access(pathName, fileNotFound => {\n      const exists = !fileNotFound;\n      if (exists) {\n        uploadStream.emit('error', fSError(fileExist));\n      } else {\n        const temporalName = path.join(this.path, `${name}.tmp-${String(Math.random()).replace(/^0\\./, '')}`);\n        const file = fs.createWriteStream(temporalName);\n        const removeTempFile = (): void => fs.unlink(temporalName, () => {});\n        let opened = false;\n        uploadStream.pipe(file);\n\n        uploadStream.done = function(): void {\n          const onend = function(): void {\n            file.on('close', function() {\n              renameTmp(temporalName, pathName, function(err) {\n                if (err) {\n                  uploadStream.emit('error', err);\n                } else {\n                  uploadStream.emit('success');\n                }\n              });\n            });\n            file.end();\n          };\n          if (_ended) {\n            onend();\n          } else {\n            uploadStream.on('end', onend);\n          }\n        };\n\n        uploadStream.abort = function(): void {\n          if (opened) {\n            opened = false;\n            file.on('close', function() {\n              removeTempFile();\n            });\n          } else {\n            // if the file does not recieve any byte never is opened and has to be removed anyway.\n            removeTempFile();\n          }\n          file.end();\n        };\n\n        file.on('open', function() {\n          opened = true;\n          // re-emitting open because it's handled in storage.js\n          uploadStream.emit('open');\n        });\n\n        file.on('error', function(err) {\n          uploadStream.emit('error', err);\n        });\n      }\n    });\n\n    return uploadStream;\n  }\n\n  public readTarball(name: string): ReadTarball {\n    const pathName: string = this._getStorage(name);\n    this.logger.debug({ packageName: name }, '[local-storage/readTarball] read a tarball for package: @{packageName}');\n\n    const readTarballStream = new ReadTarball({});\n\n    const readStream = fs.createReadStream(pathName);\n\n    readStream.on('error', function(err) {\n      readTarballStream.emit('error', err);\n    });\n\n    readStream.on('open', function(fd) {\n      fs.fstat(fd, function(err, stats) {\n        if (_.isNil(err) === false) {\n          return readTarballStream.emit('error', err);\n        }\n        readTarballStream.emit('content-length', stats.size);\n        readTarballStream.emit('open');\n        readStream.pipe(readTarballStream);\n      });\n    });\n\n    readTarballStream.abort = function(): void {\n      readStream.close();\n    };\n\n    return readTarballStream;\n  }\n\n  private _createFile(name: string, contents: any, callback: Function): void {\n    this.logger.trace({ name }, '[local-storage/_createFile] create a new file: @{name}');\n\n    fs.open(name, 'wx', err => {\n      if (err) {\n        // native EEXIST used here to check exception on fs.open\n        if (err.code === 'EEXIST') {\n          this.logger.trace({ name }, '[local-storage/_createFile] file cannot be created, it already exists: @{name}');\n          return callback(fSError(fileExist));\n        }\n      }\n\n      this._writeFile(name, contents, callback);\n    });\n  }\n\n  private _readStorageFile(name: string): Promise<any> {\n    return new Promise((resolve, reject): void => {\n      this.logger.trace({ name }, '[local-storage/_readStorageFile] read a file: @{name}');\n\n      fs.readFile(name, (err, data) => {\n        if (err) {\n          this.logger.trace({ err }, '[local-storage/_readStorageFile] error on read the file: @{name}');\n          reject(err);\n        } else {\n          this.logger.trace({ name }, '[local-storage/_readStorageFile] read file succeed: @{name}');\n\n          resolve(data);\n        }\n      });\n    });\n  }\n\n  private _convertToString(value: Package): string {\n    return JSON.stringify(value, null, '\\t');\n  }\n\n  private _getStorage(fileName = ''): string {\n    const storagePath: string = path.join(this.path, fileName);\n\n    return storagePath;\n  }\n\n  private _writeFile(dest: string, data: string, cb: Callback): void {\n    const createTempFile = (cb): void => {\n      const tempFilePath = tempFile(dest);\n\n      fs.writeFile(tempFilePath, data, err => {\n        if (err) {\n          this.logger.trace({ name: dest }, '[local-storage/_writeFile] new file: @{name} has been created');\n\n          return cb(err);\n        }\n\n        this.logger.trace({ name: dest }, '[local-storage/_writeFile] creating a new file: @{name}');\n        renameTmp(tempFilePath, dest, cb);\n      });\n    };\n\n    createTempFile(err => {\n      if (err && err.code === noSuchFile) {\n        mkdirp(path.dirname(dest), function(err) {\n          if (err) {\n            return cb(err);\n          }\n          createTempFile(cb);\n        });\n      } else {\n        cb(err);\n      }\n    });\n  }\n\n  private _lockAndReadJSON(name: string, cb: Function): void {\n    const fileName: string = this._getStorage(name);\n\n    readFile(\n      fileName,\n      {\n        lock: true,\n        parse: true,\n      },\n      (err, res) => {\n        if (err) {\n          this.logger.trace({ name }, '[local-storage/_lockAndReadJSON] read new file: @{name} has failed');\n\n          return cb(err);\n        }\n\n        this.logger.trace({ name }, '[local-storage/_lockAndReadJSON] file: @{name} read');\n        return cb(null, res);\n      }\n    );\n  }\n\n  private _unlockJSON(name: string, cb: Function): void {\n    unlockFile(this._getStorage(name), cb);\n  }\n}\n"]} |
\ | No newline at end of file |