0
Fork 0
mirror of https://github.com/verdaccio/verdaccio.git synced 2025-01-20 22:52:46 -05:00
verdaccio/packages/plugins/local-storage/src/local-fs.ts
Juan Picado a828271d63
refactor: fetch package endpoint and basic integration fastify (#2750)
* refactor: download manifest endpoint and integrate fastify

* fix file not found issue

* add temporary migration method to storage memory

* sanitize fs methods

* restore tests

* chore: restore sanitilize

will do later

* add tests

* add changeset

* lint

* trying something test

* chore: lint

* restore code

* fix e2e

* fix lint
2021-12-12 18:00:19 +01:00

400 lines
11 KiB
TypeScript

/* eslint-disable no-undef */
import buildDebug from 'debug';
import fs from 'fs';
import _ from 'lodash';
import path from 'path';
import { VerdaccioError, errorUtils } from '@verdaccio/core';
import { readFile, unlockFile } from '@verdaccio/file-locking';
import { ReadTarball, UploadTarball } from '@verdaccio/streams';
import { Callback, ILocalPackageManager, IUploadTarball, Logger, Package } from '@verdaccio/types';
import { readFilePromise, rmdirPromise, unlinkPromise } from './fs';
export const fileExist = 'EEXISTS';
export const noSuchFile = 'ENOENT';
export const resourceNotAvailable = 'EAGAIN';
export const packageJSONFileName = 'package.json';
const debug = buildDebug('verdaccio:plugin:local-storage:local-fs');
export const fSError = function (message: string, code = 409): VerdaccioError {
const err: VerdaccioError = errorUtils.getCode(code, message);
// FIXME: we should return http-status codes here instead, future improvement
// @ts-ignore
err.code = message;
return err;
};
const tempFile = function (str): string {
return `${str}.tmp${String(Math.random()).substr(2)}`;
};
const renameTmp = function (src, dst, _cb): void {
const cb = (err): void => {
if (err) {
fs.unlink(src, () => {});
}
_cb(err);
};
if (process.platform !== 'win32') {
return fs.rename(src, dst, cb);
}
// windows can't remove opened file,
// but it seem to be able to rename it
const tmp = tempFile(dst);
fs.rename(dst, tmp, function (err) {
fs.rename(src, dst, cb);
if (!err) {
fs.unlink(tmp, () => {});
}
});
};
export type ILocalFSPackageManager = ILocalPackageManager & { path: string };
export default class LocalFS implements ILocalFSPackageManager {
public path: string;
public logger: Logger;
public constructor(path: string, logger: Logger) {
this.path = path;
this.logger = logger;
}
/**
* This function allows to update the package thread-safely
Algorithm:
1. lock package.json for writing
2. read package.json
3. updateFn(pkg, cb), and wait for cb
4. write package.json.tmp
5. move package.json.tmp package.json
6. callback(err?)
* @param {*} name
* @param {*} updateHandler
* @param {*} onWrite
* @param {*} transformPackage
* @param {*} onEnd
*/
public updatePackage(
name: string,
updateHandler: Callback,
onWrite: Callback,
transformPackage: Function,
onEnd: Callback
): void {
this._lockAndReadJSON(packageJSONFileName, (err, json) => {
let locked = false;
const self = this;
// callback that cleans up lock first
const unLockCallback = function (lockError: Error): void {
// eslint-disable-next-line prefer-rest-params
const _args = arguments;
if (locked) {
debug('unlock %s', packageJSONFileName);
self._unlockJSON(packageJSONFileName, () => {
// ignore any error from the unlock
if (lockError !== null) {
debug('lock file: %o has failed with error %o', name, lockError);
}
onEnd.apply(lockError, _args);
});
} else {
debug('file: %o has been updated', name);
onEnd(..._args);
}
};
if (!err) {
locked = true;
debug('file: %o has been locked', name);
}
if (_.isNil(err) === false) {
if (err.code === resourceNotAvailable) {
return unLockCallback(errorUtils.getInternalError('resource temporarily unavailable'));
} else if (err.code === noSuchFile) {
return unLockCallback(errorUtils.getNotFound());
} else {
return unLockCallback(err);
}
}
updateHandler(json, (err) => {
if (err) {
return unLockCallback(err);
}
onWrite(name, transformPackage(json), unLockCallback);
});
});
}
public async deletePackage(packageName: string): Promise<void> {
debug('delete a file/package %o', packageName);
return await unlinkPromise(this._getStorage(packageName));
}
public async removePackage(): Promise<void> {
debug('remove a package folder %o', this.path);
await rmdirPromise(this._getStorage('.'));
}
public createPackage(name: string, value: Package, cb: Callback): void {
debug('create a package %o', name);
this._createFile(this._getStorage(packageJSONFileName), this._convertToString(value), cb);
}
public savePackage(name: string, value: Package, cb: Callback): void {
debug('save a package %o', name);
this._writeFile(this._getStorage(packageJSONFileName), this._convertToString(value), cb);
}
public async readPackageNext(name: string): Promise<Package> {
debug('read a package %o', name);
try {
const res = await this._readStorageFile(this._getStorage(packageJSONFileName));
const data: any = JSON.parse(res.toString('utf8'));
debug('read storage file %o has succeed', name);
return data;
} catch (err: any) {
debug('parse error');
this.logger.error({ err, name }, 'error @{err.message} on parse @{name}');
throw err;
}
}
public readPackage(name: string, cb: Callback): void {
debug('read a package %o', name);
this._readStorageFile(this._getStorage(packageJSONFileName))
.then((res) => {
try {
const data: any = JSON.parse(res.toString('utf8'));
debug('read storage file %o has succeed', name);
cb(null, data);
} catch (err: any) {
debug('parse error');
this.logger.error({ err, name }, 'error @{err.message} on parse @{name}');
throw err;
}
})
.catch((err) => {
debug('error on read storage file %o', err.message);
return cb(err);
});
}
public writeTarball(name: string): IUploadTarball {
const uploadStream = new UploadTarball({});
debug('write a tarball for a package %o', name);
let _ended = 0;
uploadStream.on('end', function () {
_ended = 1;
});
const pathName: string = this._getStorage(name);
fs.access(pathName, (fileNotFound) => {
const exists = !fileNotFound;
if (exists) {
uploadStream.emit('error', fSError(fileExist));
} else {
const temporalName = path.join(
this.path,
`${name}.tmp-${String(Math.random()).replace(/^0\./, '')}`
);
debug('write a temporal name %o', temporalName);
const file = fs.createWriteStream(temporalName);
const removeTempFile = (): void => fs.unlink(temporalName, () => {});
let opened = false;
uploadStream.pipe(file);
uploadStream.done = function (): void {
const onend = function (): void {
file.on('close', function () {
renameTmp(temporalName, pathName, function (err) {
if (err) {
uploadStream.emit('error', err);
} else {
uploadStream.emit('success');
}
});
});
file.end();
};
if (_ended) {
onend();
} else {
uploadStream.on('end', onend);
}
};
uploadStream.abort = function (): void {
if (opened) {
opened = false;
file.on('close', function () {
removeTempFile();
});
} else {
// if the file does not recieve any byte never is opened and has to be removed anyway.
removeTempFile();
}
file.end();
};
file.on('open', function () {
opened = true;
// re-emitting open because it's handled in storage.js
uploadStream.emit('open');
});
file.on('error', function (err) {
uploadStream.emit('error', err);
});
}
});
return uploadStream;
}
public readTarball(name: string): ReadTarball {
const pathName: string = this._getStorage(name);
debug('read a a tarball %o on path %o', name, pathName);
const readTarballStream = new ReadTarball({});
const readStream = fs.createReadStream(pathName);
readStream.on('error', function (err) {
debug('error on read a tarball %o with error %o', name, err);
readTarballStream.emit('error', err);
});
readStream.on('open', function (fd) {
fs.fstat(fd, function (err, stats) {
if (_.isNil(err) === false) {
debug('error on read a tarball %o with error %o', name, err);
return readTarballStream.emit('error', err);
}
readTarballStream.emit('content-length', stats.size);
readTarballStream.emit('open');
debug('open on read a tarball %o', name);
readStream.pipe(readTarballStream);
});
});
readTarballStream.abort = function (): void {
debug('abort on read a tarball %o', name);
readStream.close();
};
return readTarballStream;
}
private _createFile(name: string, contents: any, callback: Function): void {
debug(' create a new file: %o', name);
fs.open(name, 'wx', (err) => {
if (err) {
// native EEXIST used here to check exception on fs.open
if (err.code === 'EEXIST') {
debug('file %o cannot be created, it already exists: %o', name);
return callback(fSError(fileExist));
}
}
this._writeFile(name, contents, callback);
});
}
private async _readStorageFile(name: string): Promise<any> {
debug('reading the file: %o', name);
try {
debug('reading the file: %o', name);
return await readFilePromise(name);
} catch (err: any) {
debug('error reading the file: %o with error %o', name, err.message);
throw err;
}
}
private _convertToString(value: Package): string {
return JSON.stringify(value, null, '\t');
}
private _getStorage(fileName = ''): string {
const storagePath: string = path.join(this.path, fileName);
return storagePath;
}
private _writeFile(dest: string, data: string, cb: Callback): void {
const createTempFile = (cb): void => {
const tempFilePath = tempFile(dest);
fs.writeFile(tempFilePath, data, (err) => {
if (err) {
debug('error on write the file: %o', dest);
return cb(err);
}
debug('creating a new file:: %o', dest);
renameTmp(tempFilePath, dest, cb);
});
};
createTempFile((err) => {
if (err && err.code === noSuchFile) {
fs.mkdir(path.dirname(dest), { recursive: true }, function (err) {
if (err) {
return cb(err);
}
createTempFile(cb);
});
} else {
cb(err);
}
});
}
private _lockAndReadJSON(name: string, cb: Function): void {
const fileName: string = this._getStorage(name);
debug('lock and read a file %o', fileName);
readFile(
fileName,
{
lock: true,
parse: true,
},
(err, res) => {
if (err) {
this.logger.error({ err }, 'error on lock file @{err.message}');
debug('error on lock and read json for file: %o', name);
return cb(err);
}
debug('lock and read json for file: %o', name);
return cb(null, res);
}
);
}
private _unlockJSON(name: string, cb: Function): void {
unlockFile(this._getStorage(name), cb);
}
}