2023-01-13 05:40:33 +01:00
|
|
|
import Limiter from "ratelimiter";
|
|
|
|
import Logger from "@/services/logger.js";
|
2023-10-30 09:58:37 +01:00
|
|
|
import { redisClient } from "@/db/redis.js";
|
2023-01-13 05:40:33 +01:00
|
|
|
import type { IEndpointMeta } from "./endpoints.js";
|
2023-06-23 02:14:27 +02:00
|
|
|
import { convertMilliseconds } from "@/misc/convert-milliseconds.js";
|
2016-12-28 23:49:51 +01:00
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
const logger = new Logger("limiter");
|
2017-01-23 10:25:52 +01:00
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
export const limiter = (
|
|
|
|
limitation: IEndpointMeta["limit"] & { key: NonNullable<string> },
|
|
|
|
actor: string,
|
|
|
|
) =>
|
|
|
|
new Promise<void>((ok, reject) => {
|
|
|
|
if (process.env.NODE_ENV === "test") ok();
|
2022-06-26 12:16:32 +02:00
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
const hasShortTermLimit = typeof limitation.minInterval === "number";
|
2016-12-28 23:49:51 +01:00
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
const hasLongTermLimit =
|
|
|
|
typeof limitation.duration === "number" &&
|
|
|
|
typeof limitation.max === "number";
|
2016-12-28 23:49:51 +01:00
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
if (hasShortTermLimit) {
|
|
|
|
min();
|
|
|
|
} else if (hasLongTermLimit) {
|
|
|
|
max();
|
|
|
|
} else {
|
|
|
|
ok();
|
|
|
|
}
|
2016-12-28 23:49:51 +01:00
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
// Short-term limit
|
|
|
|
function min(): void {
|
|
|
|
const minIntervalLimiter = new Limiter({
|
|
|
|
id: `${actor}:${limitation.key}:min`,
|
|
|
|
duration: limitation.minInterval,
|
|
|
|
max: 1,
|
|
|
|
db: redisClient,
|
|
|
|
});
|
2016-12-28 23:49:51 +01:00
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
minIntervalLimiter.get((err, info) => {
|
|
|
|
if (err) {
|
|
|
|
return reject("ERR");
|
|
|
|
}
|
2017-01-23 10:25:52 +01:00
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
logger.debug(
|
|
|
|
`${actor} ${limitation.key} min remaining: ${info.remaining}`,
|
|
|
|
);
|
2017-01-23 10:25:52 +01:00
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
if (info.remaining === 0) {
|
|
|
|
reject("BRIEF_REQUEST_INTERVAL");
|
2016-12-28 23:49:51 +01:00
|
|
|
} else {
|
2023-01-13 05:40:33 +01:00
|
|
|
if (hasLongTermLimit) {
|
|
|
|
max();
|
|
|
|
} else {
|
|
|
|
ok();
|
|
|
|
}
|
2016-12-28 23:49:51 +01:00
|
|
|
}
|
2023-01-13 05:40:33 +01:00
|
|
|
});
|
|
|
|
}
|
2016-12-28 23:49:51 +01:00
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
// Long term limit
|
|
|
|
function max(): void {
|
|
|
|
const limiter = new Limiter({
|
|
|
|
id: `${actor}:${limitation.key}`,
|
|
|
|
duration: limitation.duration,
|
|
|
|
max: limitation.max,
|
|
|
|
db: redisClient,
|
|
|
|
});
|
2016-12-28 23:49:51 +01:00
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
limiter.get((err, info) => {
|
|
|
|
if (err) {
|
|
|
|
return reject("ERR");
|
|
|
|
}
|
2017-01-23 10:25:52 +01:00
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
logger.debug(
|
|
|
|
`${actor} ${limitation.key} max remaining: ${info.remaining}`,
|
|
|
|
);
|
2017-01-23 10:25:52 +01:00
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
if (info.remaining === 0) {
|
2023-06-23 02:14:27 +02:00
|
|
|
reject({
|
|
|
|
message: "RATE_LIMIT_EXCEEDED",
|
2023-06-29 08:04:40 +02:00
|
|
|
remainingTime: convertMilliseconds(info.resetMs - Date.now()),
|
2023-06-23 02:14:27 +02:00
|
|
|
});
|
2023-01-13 05:40:33 +01:00
|
|
|
} else {
|
|
|
|
ok();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|