mirror of
https://github.com/laurent22/joplin.git
synced 2024-12-15 09:04:04 +02:00
235 lines
4.8 KiB
JavaScript
235 lines
4.8 KiB
JavaScript
import fs from 'fs-extra';
|
|
import { promiseChain } from 'lib/promise-utils.js';
|
|
import moment from 'moment';
|
|
import { BaseItem } from 'lib/models/base-item.js';
|
|
import { time } from 'lib/time-utils.js';
|
|
|
|
class FileApiDriverLocal {
|
|
|
|
syncTargetId() {
|
|
return 2;
|
|
}
|
|
|
|
syncTargetName() {
|
|
return 'filesystem';
|
|
}
|
|
|
|
fsErrorToJsError_(error) {
|
|
let msg = error.toString();
|
|
let output = new Error(msg);
|
|
if (error.code) output.code = error.code;
|
|
return output;
|
|
}
|
|
|
|
supportsDelta() {
|
|
return false;
|
|
}
|
|
|
|
stat(path) {
|
|
return new Promise((resolve, reject) => {
|
|
fs.stat(path, (error, s) => {
|
|
if (error) {
|
|
if (error.code == 'ENOENT') {
|
|
resolve(null);
|
|
} else {
|
|
reject(this.fsErrorToJsError_(error));
|
|
}
|
|
return;
|
|
}
|
|
resolve(this.metadataFromStats_(path, s));
|
|
});
|
|
});
|
|
}
|
|
|
|
statTimeToTimestampMs_(time) {
|
|
let m = moment(time, 'YYYY-MM-DDTHH:mm:ss.SSSZ');
|
|
if (!m.isValid()) {
|
|
throw new Error('Invalid date: ' + time);
|
|
}
|
|
return m.toDate().getTime();
|
|
}
|
|
|
|
metadataFromStats_(path, stats) {
|
|
return {
|
|
path: path,
|
|
created_time: this.statTimeToTimestampMs_(stats.birthtime),
|
|
updated_time: this.statTimeToTimestampMs_(stats.mtime),
|
|
created_time_orig: stats.birthtime,
|
|
updated_time_orig: stats.mtime,
|
|
isDir: stats.isDirectory(),
|
|
};
|
|
}
|
|
|
|
setTimestamp(path, timestampMs) {
|
|
return new Promise((resolve, reject) => {
|
|
let t = Math.floor(timestampMs / 1000);
|
|
fs.utimes(path, t, t, (error) => {
|
|
if (error) {
|
|
reject(this.fsErrorToJsError_(error));
|
|
return;
|
|
}
|
|
resolve();
|
|
});
|
|
});
|
|
}
|
|
|
|
async delta(path, options) {
|
|
try {
|
|
let items = await fs.readdir(path);
|
|
let output = [];
|
|
for (let i = 0; i < items.length; i++) {
|
|
let stat = await this.stat(path + '/' + items[i]);
|
|
if (!stat) continue; // Has been deleted between the readdir() call and now
|
|
stat.path = items[i];
|
|
output.push(stat);
|
|
}
|
|
|
|
if (!Array.isArray(options.itemIds)) throw new Error('Delta API not supported - local IDs must be provided');
|
|
|
|
let deletedItems = [];
|
|
for (let i = 0; i < options.itemIds.length; i++) {
|
|
const itemId = options.itemIds[i];
|
|
let found = false;
|
|
for (let j = 0; j < output.length; j++) {
|
|
const item = output[j];
|
|
if (BaseItem.pathToId(item.path) == itemId) {
|
|
found = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!found) {
|
|
deletedItems.push({
|
|
path: BaseItem.systemPath(itemId),
|
|
isDeleted: true,
|
|
});
|
|
}
|
|
}
|
|
|
|
return {
|
|
hasMore: false,
|
|
context: null,
|
|
items: output,
|
|
};
|
|
} catch(error) {
|
|
throw this.fsErrorToJsError_(error);
|
|
}
|
|
}
|
|
|
|
async list(path, options) {
|
|
try {
|
|
let items = await fs.readdir(path);
|
|
let output = [];
|
|
for (let i = 0; i < items.length; i++) {
|
|
let stat = await this.stat(path + '/' + items[i]);
|
|
if (!stat) continue; // Has been deleted between the readdir() call and now
|
|
stat.path = items[i];
|
|
output.push(stat);
|
|
}
|
|
|
|
return {
|
|
items: output,
|
|
hasMore: false,
|
|
context: null,
|
|
};
|
|
} catch(error) {
|
|
throw this.fsErrorToJsError_(error);
|
|
}
|
|
}
|
|
|
|
async get(path, options) {
|
|
let output = null;
|
|
|
|
try {
|
|
if (options.encoding == 'binary') {
|
|
output = fs.readFile(path);
|
|
} else {
|
|
output = fs.readFile(path, options.encoding);
|
|
}
|
|
} catch (error) {
|
|
if (error.code == 'ENOENT') return null;
|
|
throw this.fsErrorToJsError_(error);
|
|
}
|
|
|
|
return output;
|
|
}
|
|
|
|
mkdir(path) {
|
|
return new Promise((resolve, reject) => {
|
|
fs.exists(path, (exists) => {
|
|
if (exists) {
|
|
resolve();
|
|
return;
|
|
}
|
|
|
|
const mkdirp = require('mkdirp');
|
|
|
|
mkdirp(path, (error) => {
|
|
if (error) {
|
|
reject(this.fsErrorToJsError_(error));
|
|
} else {
|
|
resolve();
|
|
}
|
|
});
|
|
});
|
|
});
|
|
}
|
|
|
|
put(path, content) {
|
|
return new Promise((resolve, reject) => {
|
|
fs.writeFile(path, content, function(error) {
|
|
if (error) {
|
|
reject(this.fsErrorToJsError_(error));
|
|
} else {
|
|
resolve();
|
|
}
|
|
});
|
|
});
|
|
}
|
|
|
|
delete(path) {
|
|
return new Promise((resolve, reject) => {
|
|
fs.unlink(path, function(error) {
|
|
if (error) {
|
|
if (error && error.code == 'ENOENT') {
|
|
// File doesn't exist - it's fine
|
|
resolve();
|
|
} else {
|
|
reject(this.fsErrorToJsError_(error));
|
|
}
|
|
} else {
|
|
resolve();
|
|
}
|
|
});
|
|
});
|
|
}
|
|
|
|
async move(oldPath, newPath) {
|
|
let lastError = null;
|
|
|
|
for (let i = 0; i < 5; i++) {
|
|
try {
|
|
let output = await fs.move(oldPath, newPath, { overwrite: true });
|
|
return output;
|
|
} catch (error) {
|
|
lastError = error;
|
|
// Normally cannot happen with the `overwrite` flag but sometime it still does.
|
|
// In this case, retry.
|
|
if (error.code == 'EEXIST') {
|
|
await time.sleep(1);
|
|
continue;
|
|
}
|
|
throw this.fsErrorToJsError_(error);
|
|
}
|
|
}
|
|
|
|
throw lastError;
|
|
}
|
|
|
|
format() {
|
|
throw new Error('Not supported');
|
|
}
|
|
|
|
}
|
|
|
|
export { FileApiDriverLocal }; |