wildduck/lib/attachment-storage.js

98 lines
2.6 KiB
JavaScript

'use strict';
const GridstoreStorage = require('./attachments/gridstore-storage.js');
const crypto = require('crypto');
const CHUNK_SIZE = 64 * 1024; // chunk size for calculating hashes
class AttachmentStorage {
constructor(options) {
this.options = options || {};
let type = (options.options && options.options.type) || 'gridstore';
switch (type) {
case 'gridstore':
default:
this.storage = new GridstoreStorage(this.options);
break;
}
}
async get(attachmentId) {
return await this.storage.get(attachmentId);
}
create(attachment, callback) {
this.calculateHash(attachment.body, (err, hash) => {
if (err) {
return callback(err);
}
return this.storage.create(attachment, hash, callback);
});
}
createReadStream(id, attachmentData, options) {
return this.storage.createReadStream(id, attachmentData, options);
}
deleteMany(ids, magic, callback) {
let pos = 0;
let deleteNext = () => {
if (pos >= ids.length) {
return callback(null, true);
}
let id = ids[pos++];
this.delete(id, magic, deleteNext);
};
deleteNext();
}
updateMany(ids, count, magic, callback) {
return this.storage.update(ids, count, magic, callback);
}
delete(id, magic, callback) {
return this.storage.delete(id, magic, callback);
}
deleteOrphaned(callback) {
return this.storage.deleteOrphaned(callback);
}
calculateHash(input, callback) {
let algo = 'sha256';
let hash = crypto.createHash(algo);
let chunkPos = 0;
let nextChunk = () => {
try {
if (chunkPos >= input.length) {
let result = hash.digest('hex');
return callback(null, result);
}
if (!chunkPos && CHUNK_SIZE >= input.length) {
// fits all
hash.update(input);
} else if (chunkPos + CHUNK_SIZE >= input.length) {
// final chunk
hash.update(input.slice(chunkPos));
} else {
// middle chunk
hash.update(input.slice(chunkPos, chunkPos + CHUNK_SIZE));
}
chunkPos += CHUNK_SIZE;
return setImmediate(nextChunk);
} catch (E) {
return callback(E);
}
};
setImmediate(nextChunk);
}
}
module.exports = AttachmentStorage;