diff --git a/README.md b/README.md index ddf3716deb..9b6880b958 100644 --- a/README.md +++ b/README.md @@ -170,6 +170,7 @@ pnpm dev 通用: - `AUTH_SECRET_KEY` 访问权限密钥,可选 +- `MAX_REQUEST_PER_HOUR` 每小时最大请求次数,可选,默认无限 - `TIMEOUT_MS` 超时,单位毫秒,可选 - `SOCKS_PROXY_HOST` 和 `SOCKS_PROXY_PORT` 一起时生效,可选 - `SOCKS_PROXY_PORT` 和 `SOCKS_PROXY_HOST` 一起时生效,可选 @@ -224,6 +225,8 @@ services: API_REVERSE_PROXY: xxx # 访问权限密钥,可选 AUTH_SECRET_KEY: xxx + # 每小时最大请求次数,可选,默认无限 + MAX_REQUEST_PER_HOUR: 0 # 超时,单位毫秒,可选 TIMEOUT_MS: 60000 # Socks代理,可选,和 SOCKS_PROXY_PORT 一起时生效 @@ -245,6 +248,7 @@ services: | --------------------- | ---------------------- | -------------------------------------------------------------------------------------------------- | | `PORT` | 必填 | 默认 `3002` | `AUTH_SECRET_KEY` | 可选 | 访问权限密钥 | +| `MAX_REQUEST_PER_HOUR` | 可选 | 每小时最大请求次数,可选,默认无限 | | `TIMEOUT_MS` | 可选 | 超时时间,单位毫秒 | | `OPENAI_API_KEY` | `OpenAI API` 二选一 | 使用 `OpenAI API` 所需的 `apiKey` [(获取 apiKey)](https://platform.openai.com/overview) | | `OPENAI_ACCESS_TOKEN` | `Web API` 二选一 | 使用 `Web API` 所需的 `accessToken` [(获取 accessToken)](https://chat.openai.com/api/auth/session) | diff --git a/docker-compose/docker-compose.yml b/docker-compose/docker-compose.yml index 247bf8b987..a353b33576 100644 --- a/docker-compose/docker-compose.yml +++ b/docker-compose/docker-compose.yml @@ -20,6 +20,8 @@ services: API_REVERSE_PROXY: xxx # 访问jwt加密参数,可选 不为空则允许登录 同时需要设置 MONGODB_URL AUTH_SECRET_KEY: xxx + # 每小时最大请求次数,可选,默认无限 + MAX_REQUEST_PER_HOUR: 0 # 超时,单位毫秒,可选 TIMEOUT_MS: 60000 # Socks代理,可选,和 SOCKS_PROXY_PORT 一起时生效 diff --git a/service/.env.example b/service/.env.example index ac2366f6e3..7c9139e913 100644 --- a/service/.env.example +++ b/service/.env.example @@ -59,4 +59,7 @@ SMTP_TSL=true SMTP_USERNAME=yourname@example.com SMTP_PASSWORD=yourpassword -# ----- Only valid after setting AUTH_SECRET_KEY end ---- \ No newline at end of file +# ----- Only valid after setting AUTH_SECRET_KEY end ---- +# Rate Limit +MAX_REQUEST_PER_HOUR= + diff --git a/service/package.json b/service/package.json index 4e5ad1f288..7b0841dee2 100644 --- a/service/package.json +++ b/service/package.json @@ -29,6 +29,7 @@ "dotenv": "^16.0.3", "esno": "^0.16.3", "express": "^4.18.2", + "express-rate-limit": "^6.7.0", "https-proxy-agent": "^5.0.1", "isomorphic-fetch": "^3.0.0", "mongodb": "^5.1.0", diff --git a/service/pnpm-lock.yaml b/service/pnpm-lock.yaml index 346511367f..f63462a192 100644 --- a/service/pnpm-lock.yaml +++ b/service/pnpm-lock.yaml @@ -11,6 +11,7 @@ specifiers: eslint: ^8.35.0 esno: ^0.16.3 express: ^4.18.2 + express-rate-limit: ^6.7.0 https-proxy-agent: ^5.0.1 isomorphic-fetch: ^3.0.0 jsonwebtoken: ^9.0.0 @@ -28,6 +29,7 @@ dependencies: dotenv: 16.0.3 esno: 0.16.3 express: 4.18.2 + express-rate-limit: 6.7.0_express@4.18.2 https-proxy-agent: 5.0.1 isomorphic-fetch: 3.0.0 mongodb: 5.1.0 @@ -1781,6 +1783,15 @@ packages: strip-final-newline: 2.0.0 dev: true + /express-rate-limit/6.7.0_express@4.18.2: + resolution: {integrity: sha512-vhwIdRoqcYB/72TK3tRZI+0ttS8Ytrk24GfmsxDXK9o9IhHNO5bXRiXQSExPQ4GbaE5tvIS7j1SGrxsuWs+sGA==} + engines: {node: '>= 12.9.0'} + peerDependencies: + express: ^4 || ^5 + dependencies: + express: 4.18.2 + dev: false + /express/4.18.2: resolution: {integrity: sha512-5/PsL6iGPdfQ/lKM1UuielYgv3BUoJfz1aUwU9vHZ+J7gyvwdQXFEBIEIaxeGf0GIcreATNyBExtalisDbuMqQ==} engines: {node: '>= 0.10.0'} diff --git a/service/src/index.ts b/service/src/index.ts index f2501c96fc..780e397cdb 100644 --- a/service/src/index.ts +++ b/service/src/index.ts @@ -8,6 +8,9 @@ import { Status } from './storage/model' import { clearChat, createChatRoom, createUser, deleteChat, deleteChatRoom, existsChatRoom, getChat, getChatRooms, getChats, getUser, insertChat, renameChatRoom, updateChat, verifyUser } from './storage/mongo' import { sendMail } from './utils/mail' import { checkUserVerify, getUserVerifyUrl, md5 } from './utils/security' +import { limiter } from './middleware/limiter' +import { isNotEmptyString } from './utils/is' + const app = express() const router = express.Router() @@ -147,7 +150,9 @@ router.post('/chat', auth, async (req, res) => { } }) -router.post('/chat-process', auth, async (req, res) => { + +router.post('/chat-process', [auth, limiter], async (req, res) => { + res.setHeader('Content-type', 'application/octet-stream') try { diff --git a/service/src/middleware/limiter.ts b/service/src/middleware/limiter.ts new file mode 100644 index 0000000000..d4df149395 --- /dev/null +++ b/service/src/middleware/limiter.ts @@ -0,0 +1,19 @@ +import { rateLimit } from 'express-rate-limit' +import { isNotEmptyString } from '../utils/is' + +const MAX_REQUEST_PER_HOUR = process.env.MAX_REQUEST_PER_HOUR + +const maxCount = (isNotEmptyString(MAX_REQUEST_PER_HOUR) && !isNaN(Number(MAX_REQUEST_PER_HOUR))) + ? parseInt(MAX_REQUEST_PER_HOUR) + : 0 // 0 means unlimited + +const limiter = rateLimit({ + windowMs: 60 * 60 * 1000, // Maximum number of accesses within an hour + max: maxCount, + statusCode: 200, // 200 means success,but the message is 'Too many request from this IP in 1 hour' + message: async (req, res) => { + res.send({ status: 'Fail', message: 'Too many request from this IP in 1 hour', data: null }) + }, +}) + +export { limiter }