2018-03-09 22:59:12 +02:00
|
|
|
const fs = require('fs-extra');
|
|
|
|
const { time } = require('lib/time-utils.js');
|
|
|
|
const FsDriverBase = require('lib/fs-driver-base');
|
2017-07-05 23:52:31 +02:00
|
|
|
|
2018-02-25 23:08:32 +02:00
|
|
|
class FsDriverNode extends FsDriverBase {
|
2018-01-28 19:36:36 +02:00
|
|
|
fsErrorToJsError_(error, path = null) {
|
|
|
|
let msg = error.toString();
|
2019-09-19 23:51:18 +02:00
|
|
|
if (path !== null) msg += `. Path: ${path}`;
|
2018-01-28 19:36:36 +02:00
|
|
|
let output = new Error(msg);
|
|
|
|
if (error.code) output.code = error.code;
|
|
|
|
return output;
|
|
|
|
}
|
|
|
|
|
2017-07-05 23:52:31 +02:00
|
|
|
appendFileSync(path, string) {
|
|
|
|
return fs.appendFileSync(path, string);
|
|
|
|
}
|
|
|
|
|
2018-03-09 22:59:12 +02:00
|
|
|
async appendFile(path, string, encoding = 'base64') {
|
2018-01-28 19:36:36 +02:00
|
|
|
try {
|
|
|
|
return await fs.appendFile(path, string, { encoding: encoding });
|
|
|
|
} catch (error) {
|
|
|
|
throw this.fsErrorToJsError_(error, path);
|
|
|
|
}
|
2017-12-12 01:52:42 +02:00
|
|
|
}
|
|
|
|
|
2018-01-28 19:36:36 +02:00
|
|
|
async writeBinaryFile(path, content) {
|
|
|
|
try {
|
2018-09-27 20:35:10 +02:00
|
|
|
// let buffer = new Buffer(content);
|
|
|
|
let buffer = Buffer.from(content);
|
2018-01-28 19:36:36 +02:00
|
|
|
return await fs.writeFile(path, buffer);
|
|
|
|
} catch (error) {
|
|
|
|
throw this.fsErrorToJsError_(error, path);
|
|
|
|
}
|
2017-07-05 23:52:31 +02:00
|
|
|
}
|
|
|
|
|
2018-03-09 22:59:12 +02:00
|
|
|
async writeFile(path, string, encoding = 'base64') {
|
2018-01-28 19:36:36 +02:00
|
|
|
try {
|
2018-03-12 20:01:47 +02:00
|
|
|
if (encoding === 'buffer') {
|
|
|
|
return await fs.writeFile(path, string);
|
|
|
|
} else {
|
|
|
|
return await fs.writeFile(path, string, { encoding: encoding });
|
|
|
|
}
|
2018-01-28 19:36:36 +02:00
|
|
|
} catch (error) {
|
|
|
|
throw this.fsErrorToJsError_(error, path);
|
|
|
|
}
|
2018-01-17 20:51:15 +02:00
|
|
|
}
|
|
|
|
|
2018-01-25 23:15:58 +02:00
|
|
|
// same as rm -rf
|
|
|
|
async remove(path) {
|
2018-01-28 19:36:36 +02:00
|
|
|
try {
|
2018-06-18 20:56:07 +02:00
|
|
|
const r = await fs.remove(path);
|
|
|
|
return r;
|
2018-01-28 19:36:36 +02:00
|
|
|
} catch (error) {
|
|
|
|
throw this.fsErrorToJsError_(error, path);
|
|
|
|
}
|
2018-01-25 23:15:58 +02:00
|
|
|
}
|
|
|
|
|
2018-01-17 20:51:15 +02:00
|
|
|
async move(source, dest) {
|
|
|
|
let lastError = null;
|
|
|
|
|
|
|
|
for (let i = 0; i < 5; i++) {
|
|
|
|
try {
|
|
|
|
const output = await fs.move(source, dest, { overwrite: true });
|
|
|
|
return output;
|
|
|
|
} catch (error) {
|
|
|
|
lastError = error;
|
|
|
|
// Normally cannot happen with the `overwrite` flag but sometime it still does.
|
|
|
|
// In this case, retry.
|
2018-03-09 22:59:12 +02:00
|
|
|
if (error.code == 'EEXIST') {
|
2018-01-17 20:51:15 +02:00
|
|
|
await time.sleep(1);
|
|
|
|
continue;
|
|
|
|
}
|
2018-01-28 19:36:36 +02:00
|
|
|
throw this.fsErrorToJsError_(error);
|
2018-01-17 20:51:15 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
throw lastError;
|
2017-12-19 21:01:29 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
exists(path) {
|
|
|
|
return fs.pathExists(path);
|
|
|
|
}
|
|
|
|
|
2018-01-17 20:51:15 +02:00
|
|
|
async mkdir(path) {
|
|
|
|
return fs.mkdirp(path);
|
|
|
|
}
|
|
|
|
|
|
|
|
async stat(path) {
|
2018-01-17 23:01:41 +02:00
|
|
|
try {
|
2018-02-25 23:08:32 +02:00
|
|
|
const stat = await fs.stat(path);
|
|
|
|
return {
|
|
|
|
birthtime: stat.birthtime,
|
|
|
|
mtime: stat.mtime,
|
|
|
|
isDirectory: () => stat.isDirectory(),
|
|
|
|
path: path,
|
|
|
|
size: stat.size,
|
|
|
|
};
|
2018-01-17 23:01:41 +02:00
|
|
|
} catch (error) {
|
2018-03-09 22:59:12 +02:00
|
|
|
if (error.code == 'ENOENT') return null;
|
2018-01-17 23:01:41 +02:00
|
|
|
throw error;
|
|
|
|
}
|
2018-01-17 20:51:15 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
async setTimestamp(path, timestampDate) {
|
|
|
|
return fs.utimes(path, timestampDate, timestampDate);
|
|
|
|
}
|
|
|
|
|
2018-02-25 23:08:32 +02:00
|
|
|
async readDirStats(path, options = null) {
|
|
|
|
if (!options) options = {};
|
2018-03-09 22:59:12 +02:00
|
|
|
if (!('recursive' in options)) options.recursive = false;
|
2018-02-25 23:08:32 +02:00
|
|
|
|
2018-02-26 20:43:50 +02:00
|
|
|
let items = [];
|
|
|
|
try {
|
|
|
|
items = await fs.readdir(path);
|
|
|
|
} catch (error) {
|
|
|
|
throw this.fsErrorToJsError_(error);
|
|
|
|
}
|
|
|
|
|
2018-01-17 20:51:15 +02:00
|
|
|
let output = [];
|
|
|
|
for (let i = 0; i < items.length; i++) {
|
2018-02-25 23:08:32 +02:00
|
|
|
const item = items[i];
|
2019-09-19 23:51:18 +02:00
|
|
|
let stat = await this.stat(`${path}/${item}`);
|
2018-01-17 20:51:15 +02:00
|
|
|
if (!stat) continue; // Has been deleted between the readdir() call and now
|
|
|
|
stat.path = stat.path.substr(path.length + 1);
|
|
|
|
output.push(stat);
|
2018-02-25 23:08:32 +02:00
|
|
|
|
|
|
|
output = await this.readDirStatsHandleRecursion_(path, stat, output, options);
|
2018-01-17 20:51:15 +02:00
|
|
|
}
|
|
|
|
return output;
|
|
|
|
}
|
|
|
|
|
2018-01-28 19:36:36 +02:00
|
|
|
async open(path, mode) {
|
|
|
|
try {
|
|
|
|
return await fs.open(path, mode);
|
|
|
|
} catch (error) {
|
|
|
|
throw this.fsErrorToJsError_(error, path);
|
|
|
|
}
|
2017-12-12 01:52:42 +02:00
|
|
|
}
|
|
|
|
|
2018-01-28 19:36:36 +02:00
|
|
|
async close(handle) {
|
|
|
|
try {
|
|
|
|
return await fs.close(handle);
|
|
|
|
} catch (error) {
|
2019-07-30 09:35:42 +02:00
|
|
|
throw this.fsErrorToJsError_(error, '');
|
2018-01-28 19:36:36 +02:00
|
|
|
}
|
2017-12-12 01:52:42 +02:00
|
|
|
}
|
|
|
|
|
2018-03-09 22:59:12 +02:00
|
|
|
async readFile(path, encoding = 'utf8') {
|
2018-02-26 21:25:54 +02:00
|
|
|
try {
|
2018-03-09 22:59:12 +02:00
|
|
|
if (encoding === 'Buffer') return await fs.readFile(path); // Returns the raw buffer
|
2018-02-26 21:25:54 +02:00
|
|
|
return await fs.readFile(path, encoding);
|
|
|
|
} catch (error) {
|
|
|
|
throw this.fsErrorToJsError_(error, path);
|
|
|
|
}
|
2018-01-17 20:51:15 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Always overwrite destination
|
|
|
|
async copy(source, dest) {
|
2018-02-26 21:25:54 +02:00
|
|
|
try {
|
|
|
|
return await fs.copy(source, dest, { overwrite: true });
|
|
|
|
} catch (error) {
|
|
|
|
throw this.fsErrorToJsError_(error, source);
|
|
|
|
}
|
2017-07-05 23:52:31 +02:00
|
|
|
}
|
|
|
|
|
2017-12-12 01:52:42 +02:00
|
|
|
async unlink(path) {
|
|
|
|
try {
|
|
|
|
await fs.unlink(path);
|
|
|
|
} catch (error) {
|
2018-03-09 22:59:12 +02:00
|
|
|
if (error.code === 'ENOENT') return; // Don't throw if the file does not exist
|
2017-12-12 01:52:42 +02:00
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-09 22:59:12 +02:00
|
|
|
async readFileChunk(handle, length, encoding = 'base64') {
|
2019-10-09 21:35:13 +02:00
|
|
|
// let buffer = new Buffer(length);
|
2018-09-27 20:35:10 +02:00
|
|
|
let buffer = Buffer.alloc(length);
|
2017-12-23 22:21:11 +02:00
|
|
|
const result = await fs.read(handle, buffer, 0, length, null);
|
2017-12-21 21:06:08 +02:00
|
|
|
if (!result.bytesRead) return null;
|
2017-12-23 22:21:11 +02:00
|
|
|
buffer = buffer.slice(0, result.bytesRead);
|
2018-03-09 22:59:12 +02:00
|
|
|
if (encoding === 'base64') return buffer.toString('base64');
|
|
|
|
if (encoding === 'ascii') return buffer.toString('ascii');
|
2019-09-19 23:51:18 +02:00
|
|
|
throw new Error(`Unsupported encoding: ${encoding}`);
|
2017-12-12 01:52:42 +02:00
|
|
|
}
|
2017-07-05 23:52:31 +02:00
|
|
|
}
|
|
|
|
|
2019-07-29 15:43:53 +02:00
|
|
|
module.exports.FsDriverNode = FsDriverNode;
|