Files
cloudron-box/src/storage/filesystem.js

207 lines
8.2 KiB
JavaScript
Raw Normal View History

'use strict';
exports = module.exports = {
upload: upload,
download: download,
copy: copy,
2018-07-27 14:29:07 -07:00
listDir: listDir,
remove: remove,
removeDir: removeDir,
2016-09-16 11:21:08 +02:00
testConfig: testConfig,
removePrivateFields: removePrivateFields,
injectPrivateFields: injectPrivateFields
};
var assert = require('assert'),
2019-10-22 20:36:20 -07:00
BoxError = require('../boxerror.js'),
debug = require('debug')('box:storage/filesystem'),
EventEmitter = require('events'),
fs = require('fs'),
mkdirp = require('mkdirp'),
2017-04-18 15:32:59 +02:00
path = require('path'),
readdirp = require('readdirp'),
safe = require('safetydance'),
shell = require('../shell.js');
2017-04-18 15:32:59 +02:00
// storage api
function upload(apiConfig, backupFilePath, sourceStream, callback) {
assert.strictEqual(typeof apiConfig, 'object');
2017-09-19 20:40:38 -07:00
assert.strictEqual(typeof backupFilePath, 'string');
assert.strictEqual(typeof sourceStream, 'object');
assert.strictEqual(typeof callback, 'function');
mkdirp(path.dirname(backupFilePath), function (error) {
2019-10-22 20:36:20 -07:00
if (error) return callback(new BoxError(BoxError.EXTERNAL_ERROR, error.message));
2016-09-16 11:21:08 +02:00
2017-09-22 14:40:37 -07:00
safe.fs.unlinkSync(backupFilePath); // remove any hardlink
var fileStream = fs.createWriteStream(backupFilePath);
2016-09-16 11:21:08 +02:00
// this pattern is required to ensure that the file got created before 'finish'
fileStream.on('open', function () {
sourceStream.pipe(fileStream);
});
2017-04-20 15:35:52 +02:00
fileStream.on('error', function (error) {
debug('[%s] upload: out stream error.', backupFilePath, error);
2019-10-22 20:36:20 -07:00
callback(new BoxError(BoxError.EXTERNAL_ERROR, error.message));
2017-04-20 15:35:52 +02:00
});
2017-09-26 16:42:54 -07:00
fileStream.on('finish', function () {
2017-09-27 14:44:48 -07:00
// in test, upload() may or may not be called via sudo script
const BACKUP_UID = parseInt(process.env.SUDO_UID, 10) || process.getuid();
2019-10-22 20:36:20 -07:00
if (!safe.fs.chownSync(backupFilePath, BACKUP_UID, BACKUP_UID)) return callback(new BoxError(BoxError.EXTERNAL_ERROR, 'Unable to chown:' + safe.error.message));
if (!safe.fs.chownSync(path.dirname(backupFilePath), BACKUP_UID, BACKUP_UID)) return callback(new BoxError(BoxError.EXTERNAL_ERROR, 'Unable to chown:' + safe.error.message));
2017-09-27 14:44:48 -07:00
debug('upload %s: done.', backupFilePath);
2017-09-27 14:44:48 -07:00
callback(null);
});
});
2016-09-16 11:21:08 +02:00
}
function download(apiConfig, sourceFilePath, callback) {
assert.strictEqual(typeof apiConfig, 'object');
2017-09-19 20:40:38 -07:00
assert.strictEqual(typeof sourceFilePath, 'string');
assert.strictEqual(typeof callback, 'function');
2018-07-30 07:39:34 -07:00
debug(`download: ${sourceFilePath}`);
2019-10-22 20:36:20 -07:00
if (!safe.fs.existsSync(sourceFilePath)) return callback(new BoxError(BoxError.NOT_FOUND, `File not found: ${sourceFilePath}`));
2017-09-28 14:26:39 -07:00
var fileStream = fs.createReadStream(sourceFilePath);
2017-09-28 14:26:39 -07:00
callback(null, fileStream);
}
function listDir(apiConfig, dir, batchSize, iteratorCallback, callback) {
assert.strictEqual(typeof apiConfig, 'object');
assert.strictEqual(typeof dir, 'string');
assert.strictEqual(typeof batchSize, 'number');
assert.strictEqual(typeof iteratorCallback, 'function');
assert.strictEqual(typeof callback, 'function');
var entries = [];
2019-04-24 10:40:33 -07:00
var entryStream = readdirp(dir, { type: 'files', alwaysStat: true, lstat: true });
entryStream.on('data', function (entryInfo) {
if (entryInfo.stats.isSymbolicLink()) return;
2018-08-02 14:59:50 -07:00
2019-04-24 10:40:33 -07:00
entries.push({ fullPath: entryInfo.fullPath });
if (entries.length < batchSize) return;
entryStream.pause();
iteratorCallback(entries, function (error) {
if (error) return callback(error);
entries = [];
entryStream.resume();
});
});
entryStream.on('warn', function (error) {
debug('listDir: warning ', error);
});
entryStream.on('end', function () {
iteratorCallback(entries, callback);
});
}
function copy(apiConfig, oldFilePath, newFilePath) {
assert.strictEqual(typeof apiConfig, 'object');
2017-09-19 20:40:38 -07:00
assert.strictEqual(typeof oldFilePath, 'string');
assert.strictEqual(typeof newFilePath, 'string');
var events = new EventEmitter();
mkdirp(path.dirname(newFilePath), function (error) {
2019-10-22 20:36:20 -07:00
if (error) return events.emit('done', new BoxError(BoxError.EXTERNAL_ERROR, error.message));
events.emit('progress', `Copying ${oldFilePath} to ${newFilePath}`);
2017-09-18 12:42:42 -07:00
// this will hardlink backups saving space
var cpOptions = apiConfig.noHardlinks ? '-a' : '-al';
2018-11-17 19:26:19 -08:00
shell.spawn('copy', '/bin/cp', [ cpOptions, oldFilePath, newFilePath ], { }, function (error) {
2019-10-22 20:36:20 -07:00
if (error) return events.emit('done', new BoxError(BoxError.EXTERNAL_ERROR, error.message));
events.emit('done', null);
});
2017-04-20 15:41:25 +02:00
});
return events;
}
function remove(apiConfig, filename, callback) {
assert.strictEqual(typeof apiConfig, 'object');
assert.strictEqual(typeof filename, 'string');
assert.strictEqual(typeof callback, 'function');
var stat = safe.fs.statSync(filename);
if (!stat) return callback();
if (stat.isFile()) {
2019-10-22 20:36:20 -07:00
if (!safe.fs.unlinkSync(filename)) return callback(new BoxError(BoxError.EXTERNAL_ERROR, safe.error.message));
} else if (stat.isDirectory()) {
2019-10-22 20:36:20 -07:00
if (!safe.fs.rmdirSync(filename)) return callback(new BoxError(BoxError.EXTERNAL_ERROR, safe.error.message));
}
2017-10-02 20:08:00 -07:00
callback(null);
}
2017-10-10 20:23:04 -07:00
function removeDir(apiConfig, pathPrefix) {
assert.strictEqual(typeof apiConfig, 'object');
assert.strictEqual(typeof pathPrefix, 'string');
2017-10-10 20:23:04 -07:00
var events = new EventEmitter();
process.nextTick(() => events.emit('progress', `Removing directory ${pathPrefix}`));
2018-11-17 19:26:19 -08:00
shell.spawn('removeDir', '/bin/rm', [ '-rf', pathPrefix ], { }, function (error) {
2019-10-22 20:36:20 -07:00
if (error) return events.emit('done', new BoxError(BoxError.EXTERNAL_ERROR, error.message));
2017-10-10 20:23:04 -07:00
events.emit('done', null);
});
2017-10-10 20:23:04 -07:00
return events;
}
function testConfig(apiConfig, callback) {
assert.strictEqual(typeof apiConfig, 'object');
assert.strictEqual(typeof callback, 'function');
2019-10-22 20:36:20 -07:00
if (typeof apiConfig.backupFolder !== 'string') return callback(new BoxError(BoxError.BAD_FIELD, 'backupFolder must be string', { field: 'backupFolder' }));
2019-10-22 20:36:20 -07:00
if (!apiConfig.backupFolder) return callback(new BoxError(BoxError.BAD_FIELD, 'backupFolder is required', { field: 'backupFolder' }));
2019-10-22 20:36:20 -07:00
if ('noHardlinks' in apiConfig && typeof apiConfig.noHardlinks !== 'boolean') return callback(new BoxError(BoxError.BAD_FIELD, 'noHardlinks must be boolean', { field: 'noHardLinks' }));
2019-10-22 20:36:20 -07:00
if ('externalDisk' in apiConfig && typeof apiConfig.externalDisk !== 'boolean') return callback(new BoxError(BoxError.BAD_FIELD, 'externalDisk must be boolean', { field: 'externalDisk' }));
2020-05-26 14:57:20 -07:00
const stat = safe.fs.statSync(apiConfig.backupFolder);
if (!stat) return callback(new BoxError(BoxError.BAD_FIELD, 'Directory does not exist or cannot be accessed: ' + safe.error.message), { field: 'backupFolder' });
if (!stat.isDirectory()) return callback(new BoxError(BoxError.BAD_FIELD, 'Backup location is not a directory', { field: 'backupFolder' }));
2020-05-26 14:57:20 -07:00
if (!safe.fs.mkdirSync(path.join(apiConfig.backupFolder, 'snapshot')) && safe.error.code !== 'EEXIST') {
if (safe.error && safe.error.code === 'EACCES') return callback(new BoxError(BoxError.BAD_FIELD, `Access denied. Run "chown yellowtent:yellowtent ${apiConfig.backupFolder}" on the server`, { field: 'backupFolder' }));
return callback(new BoxError(BoxError.BAD_FIELD, safe.error.message, { field: 'backupFolder' }));
}
2020-05-26 14:57:20 -07:00
if (!safe.fs.writeFileSync(path.join(apiConfig.backupFolder, 'cloudron-testfile'), 'testcontent')) {
return callback(new BoxError(BoxError.BAD_FIELD, `Unable to create test file as 'yellowtent' user in ${apiConfig.backupFolder}: ${safe.error.message}. Check dir/mount permissions`, { field: 'backupFolder' }));
}
if (!safe.fs.unlinkSync(path.join(apiConfig.backupFolder, 'cloudron-testfile'))) {
return callback(new BoxError(BoxError.BAD_FIELD, `Unable to remove test file as 'yellowtent' user in ${apiConfig.backupFolder}: ${safe.error.message}. Check dir/mount permissions`, { field: 'backupFolder' }));
}
callback(null);
}
2017-01-04 16:22:58 -08:00
function removePrivateFields(apiConfig) {
return apiConfig;
}
function injectPrivateFields(/* newConfig, currentConfig */) {
}