2020-08-26 04:54:58 +02:00
|
|
|
const winston = require('winston');
|
|
|
|
const Busboy = require('busboy');
|
2011-11-18 21:51:38 +01:00
|
|
|
|
|
|
|
// For handling serving stored documents
|
|
|
|
|
2020-08-28 04:39:03 +02:00
|
|
|
const DocumentHandler = function(options){
|
|
|
|
if (!options) options = new Object;
|
|
|
|
this.keyLength = options.keyLength || DocumentHandler.defaultKeyLength;
|
|
|
|
this.maxLength = options.maxLength; // none by default
|
|
|
|
this.store = options.store;
|
|
|
|
this.keyGenerator = options.keyGenerator;
|
2011-11-18 21:51:38 +01:00
|
|
|
};
|
|
|
|
|
2011-11-28 22:46:59 +01:00
|
|
|
DocumentHandler.defaultKeyLength = 10;
|
|
|
|
|
2011-11-18 22:45:48 +01:00
|
|
|
// Handle retrieving a document
|
2020-08-28 04:39:03 +02:00
|
|
|
DocumentHandler.prototype.handleGet = async function(key, res, skipExpire){
|
|
|
|
await this.store.get(key, function(ret){
|
|
|
|
if (ret){
|
|
|
|
winston.verbose('retrieved document', { key: key });
|
|
|
|
res.writeHead(200, { 'content-type': 'application/json' });
|
|
|
|
res.end(JSON.stringify({ data: ret, key: key }));
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
winston.warn('document not found', { key: key });
|
|
|
|
res.writeHead(404, { 'content-type': 'application/json' });
|
|
|
|
res.end(JSON.stringify({ message: 'Document not found.' }));
|
|
|
|
}
|
|
|
|
}, skipExpire);
|
2011-11-18 21:51:38 +01:00
|
|
|
};
|
|
|
|
|
2011-12-16 13:57:09 +01:00
|
|
|
// Handle retrieving the raw version of a document
|
2020-08-28 04:39:03 +02:00
|
|
|
DocumentHandler.prototype.handleGetRaw = async function(key, res, skipExpire){
|
|
|
|
await this.store.get(key, function(ret){
|
|
|
|
if (ret){
|
|
|
|
winston.verbose('retrieved raw document', { key: key });
|
|
|
|
res.writeHead(200, { 'content-type': 'text/plain; charset=UTF-8' });
|
|
|
|
res.end(ret);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
winston.warn('raw document not found', { key: key });
|
|
|
|
res.writeHead(404, { 'content-type': 'application/json' });
|
|
|
|
res.end(JSON.stringify({ message: 'Document not found.' }));
|
|
|
|
}
|
|
|
|
}, skipExpire);
|
2011-12-16 13:57:09 +01:00
|
|
|
};
|
|
|
|
|
2011-11-18 22:45:48 +01:00
|
|
|
// Handle adding a new Document
|
2020-08-28 04:39:03 +02:00
|
|
|
DocumentHandler.prototype.handlePost = function (req, res){
|
|
|
|
let _this = this;
|
|
|
|
let buffer = '';
|
|
|
|
let cancelled = false;
|
2014-04-21 20:16:23 +02:00
|
|
|
|
2020-08-28 04:39:03 +02:00
|
|
|
// What to do when done
|
|
|
|
let onSuccess = async function (){
|
|
|
|
// Check length
|
|
|
|
if (_this.maxLength && buffer.length > _this.maxLength){
|
|
|
|
cancelled = true;
|
|
|
|
winston.warn('document >maxLength', { maxLength: _this.maxLength });
|
|
|
|
res.writeHead(400, { 'content-type': 'application/json' });
|
|
|
|
res.end(
|
|
|
|
JSON.stringify({ message: 'Document exceeds maximum length.' })
|
|
|
|
);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
// And then save if we should
|
|
|
|
await _this.chooseKey(async function (key){
|
|
|
|
await _this.store.set(key, buffer, function (resp){
|
|
|
|
if (resp){
|
|
|
|
winston.verbose('added document', { key: key });
|
|
|
|
res.writeHead(200, { 'content-type': 'application/json' });
|
|
|
|
res.end(JSON.stringify({ key: key }));
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
winston.verbose('error adding document');
|
|
|
|
res.writeHead(500, { 'content-type': 'application/json' });
|
|
|
|
res.end(JSON.stringify({ message: 'Error adding document.' }));
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
};
|
2014-04-21 20:16:23 +02:00
|
|
|
|
2020-08-28 04:39:03 +02:00
|
|
|
// If we should, parse a form to grab the data
|
|
|
|
let ct = req.headers['content-type'];
|
|
|
|
if (ct && ct.split(';')[0] == 'multipart/form-data'){
|
|
|
|
let busboy = new Busboy({ headers: req.headers });
|
|
|
|
busboy.on('field', function (fieldname, val){
|
|
|
|
if (fieldname == 'data'){
|
|
|
|
buffer = val;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
busboy.on('finish', async function (){
|
|
|
|
await onSuccess();
|
|
|
|
});
|
|
|
|
req.pipe(busboy);
|
|
|
|
// Otherwise, use our own and just grab flat data from POST body
|
|
|
|
} else {
|
|
|
|
req.on('data', function (data){
|
|
|
|
buffer += data.toString();
|
|
|
|
});
|
|
|
|
req.on('end', async function (){
|
|
|
|
if (cancelled){ return; }
|
|
|
|
await onSuccess();
|
|
|
|
});
|
|
|
|
req.on('error', function (error){
|
|
|
|
winston.error('connection error: ' + error.message);
|
|
|
|
res.writeHead(500, { 'content-type': 'application/json' });
|
|
|
|
res.end(JSON.stringify({ message: 'Connection error.' }));
|
|
|
|
cancelled = true;
|
|
|
|
});
|
|
|
|
}
|
2011-11-18 21:51:38 +01:00
|
|
|
};
|
|
|
|
|
2012-01-13 17:16:42 +01:00
|
|
|
// Keep choosing keys until one isn't taken
|
2020-08-28 04:39:03 +02:00
|
|
|
DocumentHandler.prototype.chooseKey = async function(callback){
|
|
|
|
let key = this.acceptableKey();
|
|
|
|
let _this = this;
|
|
|
|
await this.store.get(key, function(ret){
|
|
|
|
if (ret){
|
|
|
|
_this.chooseKey(callback);
|
|
|
|
} else {
|
|
|
|
callback(key);
|
|
|
|
}
|
|
|
|
}, true); // Don't bump expirations when key searching
|
2011-11-22 03:48:09 +01:00
|
|
|
};
|
|
|
|
|
2020-08-28 04:39:03 +02:00
|
|
|
DocumentHandler.prototype.acceptableKey = function(){
|
|
|
|
return this.keyGenerator.createKey(this.keyLength);
|
2012-01-13 17:16:42 +01:00
|
|
|
};
|
|
|
|
|
2011-11-18 21:51:38 +01:00
|
|
|
module.exports = DocumentHandler;
|