@xyph3r/rate-limiter 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (91) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +310 -0
  3. package/dist/adapters/express.d.ts +26 -0
  4. package/dist/adapters/express.d.ts.map +1 -0
  5. package/dist/adapters/express.js +51 -0
  6. package/dist/adapters/express.js.map +1 -0
  7. package/dist/adapters/fastify.d.ts +24 -0
  8. package/dist/adapters/fastify.d.ts.map +1 -0
  9. package/dist/adapters/fastify.js +40 -0
  10. package/dist/adapters/fastify.js.map +1 -0
  11. package/dist/adapters/fetch.d.ts +17 -0
  12. package/dist/adapters/fetch.d.ts.map +1 -0
  13. package/dist/adapters/fetch.js +53 -0
  14. package/dist/adapters/fetch.js.map +1 -0
  15. package/dist/adapters/hono.d.ts +25 -0
  16. package/dist/adapters/hono.d.ts.map +1 -0
  17. package/dist/adapters/hono.js +40 -0
  18. package/dist/adapters/hono.js.map +1 -0
  19. package/dist/adapters/nest.d.ts +32 -0
  20. package/dist/adapters/nest.d.ts.map +1 -0
  21. package/dist/adapters/nest.js +47 -0
  22. package/dist/adapters/nest.js.map +1 -0
  23. package/dist/adapters/next.d.ts +12 -0
  24. package/dist/adapters/next.d.ts.map +1 -0
  25. package/dist/adapters/next.js +11 -0
  26. package/dist/adapters/next.js.map +1 -0
  27. package/dist/core/cached-rate-limiter-proxy.d.ts +17 -0
  28. package/dist/core/cached-rate-limiter-proxy.d.ts.map +1 -0
  29. package/dist/core/cached-rate-limiter-proxy.js +47 -0
  30. package/dist/core/cached-rate-limiter-proxy.js.map +1 -0
  31. package/dist/core/create-rate-limiter.d.ts +11 -0
  32. package/dist/core/create-rate-limiter.d.ts.map +1 -0
  33. package/dist/core/create-rate-limiter.js +34 -0
  34. package/dist/core/create-rate-limiter.js.map +1 -0
  35. package/dist/core/rate-limiter-builder.d.ts +27 -0
  36. package/dist/core/rate-limiter-builder.d.ts.map +1 -0
  37. package/dist/core/rate-limiter-builder.js +73 -0
  38. package/dist/core/rate-limiter-builder.js.map +1 -0
  39. package/dist/core/rate-limiter.d.ts +23 -0
  40. package/dist/core/rate-limiter.d.ts.map +1 -0
  41. package/dist/core/rate-limiter.js +59 -0
  42. package/dist/core/rate-limiter.js.map +1 -0
  43. package/dist/core/strategy-factory.d.ts +18 -0
  44. package/dist/core/strategy-factory.d.ts.map +1 -0
  45. package/dist/core/strategy-factory.js +17 -0
  46. package/dist/core/strategy-factory.js.map +1 -0
  47. package/dist/errors.d.ts +7 -0
  48. package/dist/errors.d.ts.map +1 -0
  49. package/dist/errors.js +13 -0
  50. package/dist/errors.js.map +1 -0
  51. package/dist/index.d.ts +19 -0
  52. package/dist/index.d.ts.map +1 -0
  53. package/dist/index.js +17 -0
  54. package/dist/index.js.map +1 -0
  55. package/dist/stores/memory-store.d.ts +9 -0
  56. package/dist/stores/memory-store.d.ts.map +1 -0
  57. package/dist/stores/memory-store.js +24 -0
  58. package/dist/stores/memory-store.js.map +1 -0
  59. package/dist/stores/rate-limit-store.d.ts +6 -0
  60. package/dist/stores/rate-limit-store.d.ts.map +1 -0
  61. package/dist/stores/rate-limit-store.js +2 -0
  62. package/dist/stores/rate-limit-store.js.map +1 -0
  63. package/dist/stores/redis-store.d.ts +26 -0
  64. package/dist/stores/redis-store.d.ts.map +1 -0
  65. package/dist/stores/redis-store.js +41 -0
  66. package/dist/stores/redis-store.js.map +1 -0
  67. package/dist/strategies/sliding-window-strategy.d.ts +31 -0
  68. package/dist/strategies/sliding-window-strategy.d.ts.map +1 -0
  69. package/dist/strategies/sliding-window-strategy.js +212 -0
  70. package/dist/strategies/sliding-window-strategy.js.map +1 -0
  71. package/dist/strategies/token-bucket-strategy.d.ts +30 -0
  72. package/dist/strategies/token-bucket-strategy.d.ts.map +1 -0
  73. package/dist/strategies/token-bucket-strategy.js +154 -0
  74. package/dist/strategies/token-bucket-strategy.js.map +1 -0
  75. package/dist/types.d.ts +48 -0
  76. package/dist/types.d.ts.map +1 -0
  77. package/dist/types.js +2 -0
  78. package/dist/types.js.map +1 -0
  79. package/dist/utils/headers.d.ts +4 -0
  80. package/dist/utils/headers.d.ts.map +1 -0
  81. package/dist/utils/headers.js +16 -0
  82. package/dist/utils/headers.js.map +1 -0
  83. package/dist/utils/http.d.ts +11 -0
  84. package/dist/utils/http.d.ts.map +1 -0
  85. package/dist/utils/http.js +41 -0
  86. package/dist/utils/http.js.map +1 -0
  87. package/dist/utils/math.d.ts +4 -0
  88. package/dist/utils/math.d.ts.map +1 -0
  89. package/dist/utils/math.js +21 -0
  90. package/dist/utils/math.js.map +1 -0
  91. package/package.json +94 -0
@@ -0,0 +1,16 @@
1
+ export function buildRateLimitHeaders(decision) {
2
+ return {
3
+ "ratelimit-limit": String(decision.limit),
4
+ "ratelimit-remaining": String(decision.remaining),
5
+ "ratelimit-reset": String(Math.max(0, Math.ceil(decision.resetAfterMs / 1000))),
6
+ "ratelimit-policy": decision.policy,
7
+ "retry-after": String(decision.retryAfterSeconds),
8
+ };
9
+ }
10
+ export function applyHeaders(setHeader, decision) {
11
+ const headers = buildRateLimitHeaders(decision);
12
+ for (const [name, value] of Object.entries(headers)) {
13
+ setHeader(name, value);
14
+ }
15
+ }
16
+ //# sourceMappingURL=headers.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"headers.js","sourceRoot":"","sources":["../../src/utils/headers.ts"],"names":[],"mappings":"AAEA,MAAM,UAAU,qBAAqB,CACnC,QAA2B;IAE3B,OAAO;QACL,iBAAiB,EAAE,MAAM,CAAC,QAAQ,CAAC,KAAK,CAAC;QACzC,qBAAqB,EAAE,MAAM,CAAC,QAAQ,CAAC,SAAS,CAAC;QACjD,iBAAiB,EAAE,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,YAAY,GAAG,IAAI,CAAC,CAAC,CAAC;QAC/E,kBAAkB,EAAE,QAAQ,CAAC,MAAM;QACnC,aAAa,EAAE,MAAM,CAAC,QAAQ,CAAC,iBAAiB,CAAC;KAClD,CAAC;AACJ,CAAC;AAED,MAAM,UAAU,YAAY,CAC1B,SAAgD,EAChD,QAA2B;IAE3B,MAAM,OAAO,GAAG,qBAAqB,CAAC,QAAQ,CAAC,CAAC;IAChD,KAAK,MAAM,CAAC,IAAI,EAAE,KAAK,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC;QACpD,SAAS,CAAC,IAAI,EAAE,KAAK,CAAC,CAAC;IACzB,CAAC;AACH,CAAC"}
@@ -0,0 +1,11 @@
1
+ export interface HeaderCarrier {
2
+ headers?: Record<string, string | string[] | undefined>;
3
+ ip?: string;
4
+ socket?: {
5
+ remoteAddress?: string;
6
+ };
7
+ }
8
+ export declare function getDefaultKeyFromHeaders(getHeader: (name: string) => string | null | undefined): string;
9
+ export declare function getDefaultKeyFromRequest(request: HeaderCarrier): string;
10
+ export declare function getDefaultKeyFromFetchRequest(request: Request): string;
11
+ //# sourceMappingURL=http.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"http.d.ts","sourceRoot":"","sources":["../../src/utils/http.ts"],"names":[],"mappings":"AAAA,MAAM,WAAW,aAAa;IAC5B,OAAO,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,GAAG,MAAM,EAAE,GAAG,SAAS,CAAC,CAAC;IACxD,EAAE,CAAC,EAAE,MAAM,CAAC;IACZ,MAAM,CAAC,EAAE;QACP,aAAa,CAAC,EAAE,MAAM,CAAC;KACxB,CAAC;CACH;AAED,wBAAgB,wBAAwB,CACtC,SAAS,EAAE,CAAC,IAAI,EAAE,MAAM,KAAK,MAAM,GAAG,IAAI,GAAG,SAAS,GACrD,MAAM,CAiBR;AAED,wBAAgB,wBAAwB,CAAC,OAAO,EAAE,aAAa,GAAG,MAAM,CA2BvE;AAED,wBAAgB,6BAA6B,CAAC,OAAO,EAAE,OAAO,GAAG,MAAM,CAEtE"}
@@ -0,0 +1,41 @@
1
+ export function getDefaultKeyFromHeaders(getHeader) {
2
+ const forwarded = getHeader("x-forwarded-for");
3
+ if (forwarded && forwarded.length > 0) {
4
+ return forwarded.split(",")[0]?.trim() || "anonymous";
5
+ }
6
+ const realIp = getHeader("x-real-ip");
7
+ if (realIp && realIp.length > 0) {
8
+ return realIp.trim();
9
+ }
10
+ const cfConnectingIp = getHeader("cf-connecting-ip");
11
+ if (cfConnectingIp && cfConnectingIp.length > 0) {
12
+ return cfConnectingIp.trim();
13
+ }
14
+ return "anonymous";
15
+ }
16
+ export function getDefaultKeyFromRequest(request) {
17
+ const forwarded = request.headers?.["x-forwarded-for"];
18
+ const realIp = request.headers?.["x-real-ip"];
19
+ const cfConnectingIp = request.headers?.["cf-connecting-ip"];
20
+ const key = getDefaultKeyFromHeaders((name) => {
21
+ if (name === "x-forwarded-for") {
22
+ if (typeof forwarded === "string") {
23
+ return forwarded;
24
+ }
25
+ return forwarded?.[0];
26
+ }
27
+ const candidate = name === "x-real-ip" ? realIp : cfConnectingIp;
28
+ if (typeof candidate === "string") {
29
+ return candidate;
30
+ }
31
+ return candidate?.[0];
32
+ });
33
+ if (key !== "anonymous") {
34
+ return key;
35
+ }
36
+ return request.ip || request.socket?.remoteAddress || "anonymous";
37
+ }
38
+ export function getDefaultKeyFromFetchRequest(request) {
39
+ return getDefaultKeyFromHeaders((name) => request.headers.get(name));
40
+ }
41
+ //# sourceMappingURL=http.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"http.js","sourceRoot":"","sources":["../../src/utils/http.ts"],"names":[],"mappings":"AAQA,MAAM,UAAU,wBAAwB,CACtC,SAAsD;IAEtD,MAAM,SAAS,GAAG,SAAS,CAAC,iBAAiB,CAAC,CAAC;IAC/C,IAAI,SAAS,IAAI,SAAS,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;QACtC,OAAO,SAAS,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,EAAE,IAAI,EAAE,IAAI,WAAW,CAAC;IACxD,CAAC;IAED,MAAM,MAAM,GAAG,SAAS,CAAC,WAAW,CAAC,CAAC;IACtC,IAAI,MAAM,IAAI,MAAM,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;QAChC,OAAO,MAAM,CAAC,IAAI,EAAE,CAAC;IACvB,CAAC;IAED,MAAM,cAAc,GAAG,SAAS,CAAC,kBAAkB,CAAC,CAAC;IACrD,IAAI,cAAc,IAAI,cAAc,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;QAChD,OAAO,cAAc,CAAC,IAAI,EAAE,CAAC;IAC/B,CAAC;IAED,OAAO,WAAW,CAAC;AACrB,CAAC;AAED,MAAM,UAAU,wBAAwB,CAAC,OAAsB;IAC7D,MAAM,SAAS,GAAG,OAAO,CAAC,OAAO,EAAE,CAAC,iBAAiB,CAAC,CAAC;IACvD,MAAM,MAAM,GAAG,OAAO,CAAC,OAAO,EAAE,CAAC,WAAW,CAAC,CAAC;IAC9C,MAAM,cAAc,GAAG,OAAO,CAAC,OAAO,EAAE,CAAC,kBAAkB,CAAC,CAAC;IAE7D,MAAM,GAAG,GAAG,wBAAwB,CAAC,CAAC,IAAI,EAAE,EAAE;QAC5C,IAAI,IAAI,KAAK,iBAAiB,EAAE,CAAC;YAC/B,IAAI,OAAO,SAAS,KAAK,QAAQ,EAAE,CAAC;gBAClC,OAAO,SAAS,CAAC;YACnB,CAAC;YAED,OAAO,SAAS,EAAE,CAAC,CAAC,CAAC,CAAC;QACxB,CAAC;QAED,MAAM,SAAS,GAAG,IAAI,KAAK,WAAW,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,cAAc,CAAC;QACjE,IAAI,OAAO,SAAS,KAAK,QAAQ,EAAE,CAAC;YAClC,OAAO,SAAS,CAAC;QACnB,CAAC;QAED,OAAO,SAAS,EAAE,CAAC,CAAC,CAAC,CAAC;IACxB,CAAC,CAAC,CAAC;IAEH,IAAI,GAAG,KAAK,WAAW,EAAE,CAAC;QACxB,OAAO,GAAG,CAAC;IACb,CAAC;IAED,OAAO,OAAO,CAAC,EAAE,IAAI,OAAO,CAAC,MAAM,EAAE,aAAa,IAAI,WAAW,CAAC;AACpE,CAAC;AAED,MAAM,UAAU,6BAA6B,CAAC,OAAgB;IAC5D,OAAO,wBAAwB,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,OAAO,CAAC,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,CAAC;AACvE,CAAC"}
@@ -0,0 +1,4 @@
1
+ export declare function clamp(value: number, min: number, max: number): number;
2
+ export declare function toWholeNumber(value: number): number;
3
+ export declare function normalizeRawRedisResult(raw: unknown, strategyName: string): [string, string, string, string, string, string, string];
4
+ //# sourceMappingURL=math.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"math.d.ts","sourceRoot":"","sources":["../../src/utils/math.ts"],"names":[],"mappings":"AAAA,wBAAgB,KAAK,CAAC,KAAK,EAAE,MAAM,EAAE,GAAG,EAAE,MAAM,EAAE,GAAG,EAAE,MAAM,GAAG,MAAM,CAErE;AAED,wBAAgB,aAAa,CAAC,KAAK,EAAE,MAAM,GAAG,MAAM,CAEnD;AAED,wBAAgB,uBAAuB,CACrC,GAAG,EAAE,OAAO,EACZ,YAAY,EAAE,MAAM,GACnB,CAAC,MAAM,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,CAAC,CAgB1D"}
@@ -0,0 +1,21 @@
1
+ export function clamp(value, min, max) {
2
+ return Math.min(Math.max(value, min), max);
3
+ }
4
+ export function toWholeNumber(value) {
5
+ return Number.isFinite(value) ? Math.max(0, Math.floor(value + 1e-9)) : 0;
6
+ }
7
+ export function normalizeRawRedisResult(raw, strategyName) {
8
+ if (!Array.isArray(raw) || raw.length < 7) {
9
+ throw new TypeError(`Redis program for "${strategyName}" returned an unexpected result.`);
10
+ }
11
+ return [
12
+ String(raw[0]),
13
+ String(raw[1]),
14
+ String(raw[2]),
15
+ String(raw[3]),
16
+ String(raw[4]),
17
+ String(raw[5]),
18
+ String(raw[6]),
19
+ ];
20
+ }
21
+ //# sourceMappingURL=math.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"math.js","sourceRoot":"","sources":["../../src/utils/math.ts"],"names":[],"mappings":"AAAA,MAAM,UAAU,KAAK,CAAC,KAAa,EAAE,GAAW,EAAE,GAAW;IAC3D,OAAO,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,GAAG,CAAC,KAAK,EAAE,GAAG,CAAC,EAAE,GAAG,CAAC,CAAC;AAC7C,CAAC;AAED,MAAM,UAAU,aAAa,CAAC,KAAa;IACzC,OAAO,MAAM,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,CAAC,KAAK,CAAC,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;AAC5E,CAAC;AAED,MAAM,UAAU,uBAAuB,CACrC,GAAY,EACZ,YAAoB;IAEpB,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,GAAG,CAAC,IAAI,GAAG,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;QAC1C,MAAM,IAAI,SAAS,CACjB,sBAAsB,YAAY,kCAAkC,CACrE,CAAC;IACJ,CAAC;IAED,OAAO;QACL,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;QACd,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;QACd,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;QACd,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;QACd,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;QACd,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;QACd,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;KACf,CAAC;AACJ,CAAC"}
package/package.json ADDED
@@ -0,0 +1,94 @@
1
+ {
2
+ "name": "@xyph3r/rate-limiter",
3
+ "version": "0.1.0",
4
+ "description": "Framework-agnostic rate limiting for Node.js with sliding window and token bucket strategies.",
5
+ "author": "xyph3r",
6
+ "license": "MIT",
7
+ "type": "module",
8
+ "sideEffects": false,
9
+ "packageManager": "npm@10.8.2",
10
+ "files": [
11
+ "dist"
12
+ ],
13
+ "main": "./dist/index.js",
14
+ "types": "./dist/index.d.ts",
15
+ "exports": {
16
+ ".": {
17
+ "types": "./dist/index.d.ts",
18
+ "import": "./dist/index.js"
19
+ },
20
+ "./adapters/express": {
21
+ "types": "./dist/adapters/express.d.ts",
22
+ "import": "./dist/adapters/express.js"
23
+ },
24
+ "./adapters/fastify": {
25
+ "types": "./dist/adapters/fastify.d.ts",
26
+ "import": "./dist/adapters/fastify.js"
27
+ },
28
+ "./adapters/fetch": {
29
+ "types": "./dist/adapters/fetch.d.ts",
30
+ "import": "./dist/adapters/fetch.js"
31
+ },
32
+ "./adapters/hono": {
33
+ "types": "./dist/adapters/hono.d.ts",
34
+ "import": "./dist/adapters/hono.js"
35
+ },
36
+ "./adapters/next": {
37
+ "types": "./dist/adapters/next.d.ts",
38
+ "import": "./dist/adapters/next.js"
39
+ },
40
+ "./adapters/nest": {
41
+ "types": "./dist/adapters/nest.d.ts",
42
+ "import": "./dist/adapters/nest.js"
43
+ },
44
+ "./stores/redis": {
45
+ "types": "./dist/stores/redis-store.d.ts",
46
+ "import": "./dist/stores/redis-store.js"
47
+ },
48
+ "./stores/memory": {
49
+ "types": "./dist/stores/memory-store.d.ts",
50
+ "import": "./dist/stores/memory-store.js"
51
+ },
52
+ "./package.json": "./package.json"
53
+ },
54
+ "repository": {
55
+ "type": "git",
56
+ "url": "git+https://github.com/faisalahmedsifat/rate-limiter.git"
57
+ },
58
+ "bugs": {
59
+ "url": "https://github.com/faisalahmedsifat/rate-limiter/issues"
60
+ },
61
+ "homepage": "https://github.com/faisalahmedsifat/rate-limiter#readme",
62
+ "publishConfig": {
63
+ "access": "public"
64
+ },
65
+ "engines": {
66
+ "node": ">=20"
67
+ },
68
+ "scripts": {
69
+ "build": "tsc -p tsconfig.build.json",
70
+ "typecheck": "tsc -p tsconfig.build.json --noEmit",
71
+ "test": "bun test test/*.test.ts",
72
+ "test:node": "rm -rf .test-dist && tsc -p tsconfig.test.json && node --test .test-dist/test/*.test.js",
73
+ "prepack": "node ./node_modules/typescript/bin/tsc -p tsconfig.build.json",
74
+ "prepublishOnly": "rm -rf .test-dist && node ./node_modules/typescript/bin/tsc -p tsconfig.test.json && node --test .test-dist/test/*.test.js"
75
+ },
76
+ "devDependencies": {
77
+ "typescript": "^5.6.3"
78
+ },
79
+ "keywords": [
80
+ "rate-limit",
81
+ "rate-limiter",
82
+ "ratelimit",
83
+ "ratelimiter",
84
+ "redis",
85
+ "bun",
86
+ "hono",
87
+ "nextjs",
88
+ "nestjs",
89
+ "express",
90
+ "fastify",
91
+ "token-bucket",
92
+ "sliding-window"
93
+ ]
94
+ }