• Stars
    star
    1,503
  • Rank 31,212 (Top 0.7 %)
  • Language
    TypeScript
  • License
    MIT License
  • Created about 13 years ago
  • Updated 8 months ago

Reviews

There are no reviews yet. Be the first to send feedback to the community and the maintainers!

Repository Details

A generic rate limiter for node.js. Useful for API clients, web crawling, or other tasks that need to be throttled

limiter

Build Status NPM Downloads

Provides a generic rate limiter for the web and node.js. Useful for API clients, web crawling, or other tasks that need to be throttled. Two classes are exposed, RateLimiter and TokenBucket. TokenBucket provides a lower level interface to rate limiting with a configurable burst rate and drip rate. RateLimiter sits on top of the token bucket and adds a restriction on the maximum number of tokens that can be removed each interval to comply with common API restrictions such as "150 requests per hour maximum".

Installation

yarn add limiter

Usage

A simple example allowing 150 requests per hour:

import { RateLimiter } from "limiter";

// Allow 150 requests per hour (the Twitter search limit). Also understands
// 'second', 'minute', 'day', or a number of milliseconds
const limiter = new RateLimiter({ tokensPerInterval: 150, interval: "hour" });

async function sendRequest() {
  // This call will throw if we request more than the maximum number of requests
  // that were set in the constructor
  // remainingRequests tells us how many additional requests could be sent
  // right this moment
  const remainingRequests = await limiter.removeTokens(1);
  callMyRequestSendingFunction(...);
}

Another example allowing one message to be sent every 250ms:

import { RateLimiter } from "limiter";

const limiter = new RateLimiter({ tokensPerInterval: 1, interval: 250 });

async function sendMessage() {
  const remainingMessages = await limiter.removeTokens(1);
  callMyMessageSendingFunction(...);
}

The default behaviour is to wait for the duration of the rate limiting that's currently in effect before the promise is resolved, but if you pass in "fireImmediately": true, the promise will be resolved immediately with remainingRequests set to -1:

import { RateLimiter } from "limiter";

const limiter = new RateLimiter({
  tokensPerInterval: 150,
  interval: "hour",
  fireImmediately: true
});

async function requestHandler(request, response) {
  // Immediately send 429 header to client when rate limiting is in effect
  const remainingRequests = await limiter.removeTokens(1);
  if (remainingRequests < 0) {
    response.writeHead(429, {'Content-Type': 'text/plain;charset=UTF-8'});
    response.end('429 Too Many Requests - your IP is being rate limited');
  } else {
    callMyMessageSendingFunction(...);
  }
}

A synchronous method, tryRemoveTokens(), is available in both RateLimiter and TokenBucket. This will return immediately with a boolean value indicating if the token removal was successful.

import { RateLimiter } from "limiter";

const limiter = new RateLimiter({ tokensPerInterval: 10, interval: "second" });

if (limiter.tryRemoveTokens(5))
  console.log('Tokens removed');
else
  console.log('No tokens removed');

To get the number of remaining tokens outside the removeTokens promise, simply use the getTokensRemaining method.

import { RateLimiter } from "limiter";

const limiter = new RateLimiter({ tokensPerInterval: 1, interval: 250 });

// Prints 1 since we did not remove a token and our number of tokens per
// interval is 1
console.log(limiter.getTokensRemaining());

Using the token bucket directly to throttle at the byte level:

import { TokenBucket } from "limiter";

const BURST_RATE = 1024 * 1024 * 150; // 150KB/sec burst rate
const FILL_RATE = 1024 * 1024 * 50; // 50KB/sec sustained rate

// We could also pass a parent token bucket in to create a hierarchical token
// bucket
// bucketSize, tokensPerInterval, interval
const bucket = new TokenBucket({
  bucketSize: BURST_RATE,
  tokensPerInterval: FILL_RATE,
  interval: "second"
});

async function handleData(myData) {
  await bucket.removeTokens(myData.byteLength);
  sendMyData(myData);
}

Additional Notes

Both the token bucket and rate limiter should be used with a message queue or some way of preventing multiple simultaneous calls to removeTokens(). Otherwise, earlier messages may get held up for long periods of time if more recent messages are continually draining the token bucket. This can lead to out of order messages or the appearance of "lost" messages under heavy load.

License

MIT License

More Repositories

1

node-echoprint-server

A node.js implementation of the Echoprint music identification server
JavaScript
74
star
2

node-pcm

Extract PCM data out of media files and into node.js
JavaScript
54
star
3

node-streamcount

Provides implementations of "sketch" algorithms for real-time counting of stream data
JavaScript
45
star
4

node-phpass

Javascript/Node implementation of the portable PHP password hashing framework
JavaScript
41
star
5

node-mysql-simple

Provides connection pooling and a simplified interface on top of node-mysql and generic-pool
JavaScript
23
star
6

node-frontpoint

Unofficial FrontPoint Security API client.
JavaScript
12
star
7

ATracerDarkly

Swift + Metal real-time ray tracing
Metal
11
star
8

homebridge-frontpoint

Homebridge plugin for FrontPoint alarm systems.
JavaScript
9
star
9

node-gitdeploy

Listen for git web hooks and automatically update and deploy code
JavaScript
9
star
10

node-graph-suggestions

Provides "people you should follow" or other graph-based recommendations
JavaScript
8
star
11

homebridge-motionblinds

A homebridge plugin to control Motion Blinds by Coulisse B.V. (incl. OmniaBlinds)
TypeScript
8
star
12

music-interpolation

Mix between music tracks using machine learning
Python
7
star
13

node-motionblinds

A node.js library for interfacing with Motion Blinds from Coulisse B.V
TypeScript
7
star
14

apple-sales-bot

Post daily Apple sales report summaries to Slack
JavaScript
6
star
15

cordova-plugin-gyroscope

Cordova plugin for accessing Gyroscope data
JavaScript
5
star
16

homebridge-rfxcom

Homebridge plugin for RFXtrx433(E) transceivers.
JavaScript
3
star
17

d3research

Diablo III R&D
C#
3
star
18

binaryninja-functionmatcher

A Binary Ninja plugin to match functions and transplant symbols between similar binaries
C
3
star
19

portrayal

iOS app to transform photos into artistic renditions
Objective-C
3
star
20

kdtreepp

k-d tree for C++ and Eigen
CMake
2
star
21

callstackpp

Stack trace pretty printing in C++
C++
1
star
22

mcapextract

Extract images, videos, and point clouds from MCAP files (https://mcap.dev/)
1
star
23

eslint-config-standard-strict

ESLint config for ES7, following StandardJS with added checks
JavaScript
1
star
24

cordova-plugin-inkfilepicker

Plugin to the native Ink Filepicker SDK
Objective-C
1
star
25

NeuralSwift

A neural network implementation in Swift
Swift
1
star
26

dashbuild

Build custom self-hosted dashboards
JavaScript
1
star
27

git-promote

git extension to squash merge pull requests into master
Shell
1
star
28

foxsocketpp

C++ implementation of the Foxglove WebSocket protocol
C++
1
star
29

duckiebot

My Jetson Nano Duckiebot sandbox
Dockerfile
1
star
30

jetson-crosscompile-template

A starting point for C++/CUDA cross-compilation and debugging for NVIDIA Jetson
C++
1
star
31

jetson-ppa

Build scripts to publish Jetson-optimized software to a custom PPA
Shell
1
star
32

music-key-detection

Detects the musical key of an audio file
Python
1
star