Made requests object safer. Expire requests before sending error back

pull/306/head
Tim Robinson 2016-05-16 19:15:15 +00:00
rodzic 9d6c7e47d7
commit 1e04331410
1 zmienionych plików z 33 dodań i 25 usunięć

56
node_modules/c9/ratelimit.js wygenerowano vendored
Wyświetl plik

@ -7,34 +7,42 @@ var _ = require("lodash");
module.exports = ratelimit; module.exports = ratelimit;
function ratelimit(key, duration, max) { function ratelimit(key, duration, max) {
var requests = {}; var requests = Object.create(null); // in case there handles like 'constructor'
setInterval(function() { setInterval(function() {
_.forIn(requests, function (requestsForHandle, key) { Object.keys(requests).forEach(expireRequests);
var totalToSplice = 0; }, duration * 0.75);
var expireTime = Date.now() - duration;
/* Requests are already sorted by date as they are appended, so we just loop
until we find one that shouldn't have expired and splice them from the list */
for (var i = 0; i < requestsForHandle.length; i++) {
if (requestsForHandle[i] >= expireTime) break;
totalToSplice++;
}
requests[key].splice(0, totalToSplice);
if (requests[key].length == 0) {
delete requests[key];
}
});
}, 10);
return function(req, res, next) {
var handle = req.params[key];
requests[handle] = requests[handle] || []; function expireRequests(value) {
if (requests[handle].length >= max) { var requestsForHandle = requests[value];
var err = new error.TooManyRequests("Rate limit exceeded"); var totalToSplice = 0;
err.retryIn = Math.min(duration, 5000); var expireTime = Date.now() - duration;
return next(err); /* Requests are already sorted by date as they are appended, so we just loop
until we find one that shouldn't have expired and splice them from the list */
for (var i = 0; i < requestsForHandle.length; i++) {
if (requestsForHandle[i] >= expireTime) break;
totalToSplice++;
}
requests[value].splice(0, totalToSplice);
if (requests[value].length == 0) {
delete requests[value];
} }
requests[handle].push(Date.now()); return true;
}
return function(req, res, next) {
var value = req.params[key];
requests[value] = requests[value] || [];
if (requests[value].length >= max) {
if (expireRequests(value) && requests[value].length >= max) {
var err = new error.TooManyRequests("Rate limit exceeded");
err.retryIn = Math.min(duration, 5000);
return next(err);
}
}
requests[value].push(Date.now());
return next(); return next();
}; };
} }