FastGPT/packages/service/common/api/frequencyLimit.ts
Finley Ge 91156f13e7
Some checks failed
Build FastGPT images in Personal warehouse / get-vars (push) Has been cancelled
Build FastGPT images in Personal warehouse / build-fastgpt-images (map[arch:amd64 runs-on:ubuntu-24.04]) (push) Has been cancelled
Build FastGPT images in Personal warehouse / build-fastgpt-images (map[arch:arm64 runs-on:ubuntu-24.04-arm]) (push) Has been cancelled
Build FastGPT images in Personal warehouse / release-fastgpt-images (push) Has been cancelled
feat: team rate limitation (#5931)
2025-11-21 18:05:17 +08:00

50 lines
1.3 KiB
TypeScript

import { getGlobalRedisConnection } from '../../common/redis';
import { jsonRes } from '../../common/response';
import type { NextApiResponse } from 'next';
type FrequencyLimitOption = {
teamId: string;
seconds: number;
limit: number;
keyPrefix: string;
res: NextApiResponse;
};
export const teamFrequencyLimit = async ({
teamId,
seconds,
limit,
keyPrefix,
res
}: FrequencyLimitOption) => {
const redis = getGlobalRedisConnection();
const key = `${keyPrefix}:${teamId}`;
const result = await redis
.multi()
.incr(key)
.expire(key, seconds, 'NX') // 只在key不存在时设置过期时间
.exec();
if (!result) {
return Promise.reject(new Error('Redis connection error'));
}
const currentCount = result[0][1] as number;
if (currentCount > limit) {
const remainingTime = await redis.ttl(key);
jsonRes(res, {
code: 429,
error: `Rate limit exceeded. Maximum ${limit} requests per ${seconds} seconds for this team. Please try again in ${remainingTime} seconds.`
});
return false;
}
// 在响应头中添加限流信息
res.setHeader('X-RateLimit-Limit', limit);
res.setHeader('X-RateLimit-Remaining', Math.max(0, limit - currentCount));
res.setHeader('X-RateLimit-Reset', Date.now() + seconds * 1000);
return true;
};