c9-core/node_modules/c9/ratelimit.js

49 wiersze
1.6 KiB
JavaScript
Czysty Zwykły widok Historia

2015-02-10 19:41:24 +00:00
var error = require("http-error");
var _ = require("lodash");
2015-02-10 19:41:24 +00:00
/**
* In memory rate limiter as connect middleware
*/
module.exports = ratelimit;
function ratelimit(key, duration, max) {
var requests = Object.create(null); // in case there handles like 'constructor'
setInterval(function() {
Object.keys(requests).forEach(expireRequests);
}, duration * 0.75);
2016-05-16 19:24:49 +00:00
function expireRequests(handle) {
var requestsForHandle = requests[handle];
var totalToSplice = 0;
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++;
}
2016-05-16 19:24:49 +00:00
requests[handle].splice(0, totalToSplice);
if (requests[handle].length == 0) {
delete requests[handle];
}
return true;
}
2015-02-10 19:41:24 +00:00
return function(req, res, next) {
2016-05-16 19:24:49 +00:00
var handle = req.params[key];
2015-02-10 19:41:24 +00:00
2016-05-16 19:24:49 +00:00
requests[handle] = requests[handle] || [];
if (requests[handle].length >= max) {
if (expireRequests(handle) && requests[handle].length >= max) {
var err = new error.TooManyRequests("Rate limit exceeded");
err.retryIn = Math.min(duration, 5000);
return next(err);
}
2015-02-10 19:41:24 +00:00
}
2016-04-28 10:10:39 +00:00
2016-05-16 19:24:49 +00:00
requests[handle].push(Date.now());
2015-02-10 19:41:24 +00:00
return next();
};
}