1
0
Fork 0
mirror of https://github.com/SunRed/haste-server.git synced 2024-11-01 01:30:21 +01:00
haste-server/lib/document_handler.js

132 lines
4.1 KiB
JavaScript
Raw Normal View History

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