2022-04-28 18:43:14 -07:00
|
|
|
'use strict';
|
|
|
|
|
|
|
|
|
|
exports = module.exports = {
|
|
|
|
|
getBackupFilePath,
|
|
|
|
|
download,
|
|
|
|
|
upload
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const assert = require('assert'),
|
|
|
|
|
async = require('async'),
|
|
|
|
|
BoxError = require('../boxerror.js'),
|
|
|
|
|
DataLayout = require('../datalayout.js'),
|
|
|
|
|
debug = require('debug')('box:backupformat/tgz'),
|
|
|
|
|
{ DecryptStream, EncryptStream } = require('../hush.js'),
|
|
|
|
|
once = require('../once.js'),
|
|
|
|
|
path = require('path'),
|
|
|
|
|
progressStream = require('progress-stream'),
|
|
|
|
|
storage = require('../storage.js'),
|
|
|
|
|
tar = require('tar-fs'),
|
|
|
|
|
zlib = require('zlib');
|
|
|
|
|
|
|
|
|
|
function getBackupFilePath(backupConfig, remotePath) {
|
|
|
|
|
assert.strictEqual(typeof backupConfig, 'object');
|
|
|
|
|
assert.strictEqual(typeof remotePath, 'string');
|
|
|
|
|
|
2022-10-02 16:26:27 +02:00
|
|
|
const rootPath = storage.api(backupConfig.provider).getBackupRootPath(backupConfig);
|
2022-04-28 18:43:14 -07:00
|
|
|
|
|
|
|
|
const fileType = backupConfig.encryption ? '.tar.gz.enc' : '.tar.gz';
|
|
|
|
|
return path.join(rootPath, remotePath + fileType);
|
|
|
|
|
}
|
|
|
|
|
|
2022-04-28 21:58:00 -07:00
|
|
|
function tarPack(dataLayout, encryption) {
|
2022-04-28 18:43:14 -07:00
|
|
|
assert(dataLayout instanceof DataLayout, 'dataLayout must be a DataLayout');
|
|
|
|
|
assert.strictEqual(typeof encryption, 'object');
|
|
|
|
|
|
|
|
|
|
const pack = tar.pack('/', {
|
|
|
|
|
dereference: false, // pack the symlink and not what it points to
|
|
|
|
|
entries: dataLayout.localPaths(),
|
|
|
|
|
ignoreStatError: (path, err) => {
|
|
|
|
|
debug(`tarPack: error stat'ing ${path} - ${err.code}`);
|
|
|
|
|
return err.code === 'ENOENT'; // ignore if file or dir got removed (probably some temporary file)
|
|
|
|
|
},
|
|
|
|
|
map: function(header) {
|
|
|
|
|
header.name = dataLayout.toRemotePath(header.name);
|
|
|
|
|
// the tar pax format allows us to encode filenames > 100 and size > 8GB (see #640)
|
|
|
|
|
// https://www.systutorials.com/docs/linux/man/5-star/
|
|
|
|
|
if (header.size > 8589934590 || header.name > 99) header.pax = { size: header.size };
|
|
|
|
|
return header;
|
|
|
|
|
},
|
|
|
|
|
strict: false // do not error for unknown types (skip fifo, char/block devices)
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
const gzip = zlib.createGzip({});
|
|
|
|
|
const ps = progressStream({ time: 10000 }); // emit 'progress' every 10 seconds
|
|
|
|
|
|
|
|
|
|
pack.on('error', function (error) {
|
|
|
|
|
debug('tarPack: tar stream error.', error);
|
|
|
|
|
ps.emit('error', new BoxError(BoxError.EXTERNAL_ERROR, error.message));
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
gzip.on('error', function (error) {
|
|
|
|
|
debug('tarPack: gzip stream error.', error);
|
|
|
|
|
ps.emit('error', new BoxError(BoxError.EXTERNAL_ERROR, error.message));
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
if (encryption) {
|
|
|
|
|
const encryptStream = new EncryptStream(encryption);
|
|
|
|
|
encryptStream.on('error', function (error) {
|
|
|
|
|
debug('tarPack: encrypt stream error.', error);
|
|
|
|
|
ps.emit('error', new BoxError(BoxError.EXTERNAL_ERROR, error.message));
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
pack.pipe(gzip).pipe(encryptStream).pipe(ps);
|
|
|
|
|
} else {
|
|
|
|
|
pack.pipe(gzip).pipe(ps);
|
|
|
|
|
}
|
|
|
|
|
|
2022-04-28 21:58:00 -07:00
|
|
|
return ps;
|
2022-04-28 18:43:14 -07:00
|
|
|
}
|
|
|
|
|
|
2022-04-28 21:58:00 -07:00
|
|
|
function tarExtract(inStream, dataLayout, encryption) {
|
2022-04-28 18:43:14 -07:00
|
|
|
assert.strictEqual(typeof inStream, 'object');
|
|
|
|
|
assert(dataLayout instanceof DataLayout, 'dataLayout must be a DataLayout');
|
|
|
|
|
assert.strictEqual(typeof encryption, 'object');
|
|
|
|
|
|
|
|
|
|
const gunzip = zlib.createGunzip({});
|
|
|
|
|
const ps = progressStream({ time: 10000 }); // display a progress every 10 seconds
|
|
|
|
|
const extract = tar.extract('/', {
|
|
|
|
|
map: function (header) {
|
|
|
|
|
header.name = dataLayout.toLocalPath(header.name);
|
|
|
|
|
return header;
|
|
|
|
|
},
|
|
|
|
|
dmode: 500 // ensure directory is writable
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
const emitError = once((error) => {
|
|
|
|
|
inStream.destroy();
|
|
|
|
|
ps.emit('error', error);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
inStream.on('error', function (error) {
|
|
|
|
|
debug('tarExtract: input stream error.', error);
|
|
|
|
|
emitError(new BoxError(BoxError.EXTERNAL_ERROR, error.message));
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
gunzip.on('error', function (error) {
|
|
|
|
|
debug('tarExtract: gunzip stream error.', error);
|
|
|
|
|
emitError(new BoxError(BoxError.EXTERNAL_ERROR, error.message));
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
extract.on('error', function (error) {
|
|
|
|
|
debug('tarExtract: extract stream error.', error);
|
|
|
|
|
emitError(new BoxError(BoxError.EXTERNAL_ERROR, error.message));
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
extract.on('finish', function () {
|
|
|
|
|
debug('tarExtract: done.');
|
|
|
|
|
// we use a separate event because ps is a through2 stream which emits 'finish' event indicating end of inStream and not extract
|
|
|
|
|
ps.emit('done');
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
if (encryption) {
|
2022-04-28 21:58:00 -07:00
|
|
|
const decrypt = new DecryptStream(encryption);
|
2022-04-28 18:43:14 -07:00
|
|
|
decrypt.on('error', function (error) {
|
|
|
|
|
debug('tarExtract: decrypt stream error.', error);
|
|
|
|
|
emitError(new BoxError(BoxError.EXTERNAL_ERROR, `Failed to decrypt: ${error.message}`));
|
|
|
|
|
});
|
|
|
|
|
inStream.pipe(ps).pipe(decrypt).pipe(gunzip).pipe(extract);
|
|
|
|
|
} else {
|
|
|
|
|
inStream.pipe(ps).pipe(gunzip).pipe(extract);
|
|
|
|
|
}
|
|
|
|
|
|
2022-04-28 21:58:00 -07:00
|
|
|
return ps;
|
2022-04-28 18:43:14 -07:00
|
|
|
}
|
|
|
|
|
|
2022-04-30 16:42:14 -07:00
|
|
|
async function download(backupConfig, remotePath, dataLayout, progressCallback) {
|
2022-04-28 18:43:14 -07:00
|
|
|
assert.strictEqual(typeof backupConfig, 'object');
|
|
|
|
|
assert.strictEqual(typeof remotePath, 'string');
|
|
|
|
|
assert(dataLayout instanceof DataLayout, 'dataLayout must be a DataLayout');
|
|
|
|
|
assert.strictEqual(typeof progressCallback, 'function');
|
|
|
|
|
|
|
|
|
|
debug(`download: Downloading ${remotePath} to ${dataLayout.toString()}`);
|
|
|
|
|
|
|
|
|
|
const backupFilePath = getBackupFilePath(backupConfig, remotePath);
|
|
|
|
|
|
2022-04-30 16:42:14 -07:00
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
|
async.retry({ times: 5, interval: 20000 }, function (retryCallback) {
|
|
|
|
|
progressCallback({ message: `Downloading backup ${remotePath}` });
|
2022-04-28 18:43:14 -07:00
|
|
|
|
2022-04-30 16:42:14 -07:00
|
|
|
storage.api(backupConfig.provider).download(backupConfig, backupFilePath, function (error, sourceStream) {
|
|
|
|
|
if (error) return retryCallback(error);
|
2022-04-28 18:43:14 -07:00
|
|
|
|
2022-04-30 16:42:14 -07:00
|
|
|
const ps = tarExtract(sourceStream, dataLayout, backupConfig.encryption);
|
2022-04-28 18:43:14 -07:00
|
|
|
|
2022-04-30 16:42:14 -07:00
|
|
|
ps.on('progress', function (progress) {
|
|
|
|
|
const transferred = Math.round(progress.transferred/1024/1024), speed = Math.round(progress.speed/1024/1024);
|
|
|
|
|
if (!transferred && !speed) return progressCallback({ message: 'Downloading backup' }); // 0M@0MBps looks wrong
|
|
|
|
|
progressCallback({ message: `Downloading ${transferred}M@${speed}MBps` });
|
|
|
|
|
});
|
|
|
|
|
ps.on('error', retryCallback);
|
|
|
|
|
ps.on('done', retryCallback);
|
2022-04-28 18:43:14 -07:00
|
|
|
});
|
2022-04-30 16:42:14 -07:00
|
|
|
}, (error) => {
|
|
|
|
|
if (error) return reject(error);
|
|
|
|
|
resolve();
|
2022-04-28 18:43:14 -07:00
|
|
|
});
|
2022-04-30 16:42:14 -07:00
|
|
|
});
|
2022-04-28 18:43:14 -07:00
|
|
|
}
|
|
|
|
|
|
2022-04-30 16:42:14 -07:00
|
|
|
async function upload(backupConfig, remotePath, dataLayout, progressCallback) {
|
2022-04-28 18:43:14 -07:00
|
|
|
assert.strictEqual(typeof backupConfig, 'object');
|
|
|
|
|
assert.strictEqual(typeof remotePath, 'string');
|
|
|
|
|
assert.strictEqual(typeof dataLayout, 'object');
|
|
|
|
|
assert.strictEqual(typeof progressCallback, 'function');
|
|
|
|
|
|
2022-04-30 16:42:14 -07:00
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
|
async.retry({ times: 5, interval: 20000 }, function (retryCallback) {
|
|
|
|
|
retryCallback = once(retryCallback); // protect again upload() erroring much later after tar stream error
|
2022-04-28 18:43:14 -07:00
|
|
|
|
2022-04-30 16:42:14 -07:00
|
|
|
const tarStream = tarPack(dataLayout, backupConfig.encryption);
|
2022-04-28 18:43:14 -07:00
|
|
|
|
2022-04-30 16:42:14 -07:00
|
|
|
tarStream.on('progress', function (progress) {
|
|
|
|
|
const transferred = Math.round(progress.transferred/1024/1024), speed = Math.round(progress.speed/1024/1024);
|
|
|
|
|
if (!transferred && !speed) return progressCallback({ message: 'Uploading backup' }); // 0M@0MBps looks wrong
|
|
|
|
|
progressCallback({ message: `Uploading backup ${transferred}M@${speed}MBps` });
|
|
|
|
|
});
|
|
|
|
|
tarStream.on('error', retryCallback); // already returns BoxError
|
2022-04-28 21:58:00 -07:00
|
|
|
|
2022-04-30 16:42:14 -07:00
|
|
|
storage.api(backupConfig.provider).upload(backupConfig, getBackupFilePath(backupConfig, remotePath), tarStream, retryCallback);
|
|
|
|
|
}, (error) => {
|
|
|
|
|
if (error) return reject(error);
|
|
|
|
|
resolve();
|
|
|
|
|
});
|
|
|
|
|
});
|
2022-04-28 18:43:14 -07:00
|
|
|
}
|