mirror of
				https://github.com/songquanpeng/one-api.git
				synced 2025-10-26 03:13:41 +08:00 
			
		
		
		
	Compare commits
	
		
			265 Commits
		
	
	
		
			v0.1.2-alp
			...
			v0.4.5-alp
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 1932c56ea8 | ||
|  | dc7bb78c74 | ||
|  | 853a288052 | ||
|  | 6536a7be62 | ||
|  | 1b5c628e66 | ||
|  | e398f470a1 | ||
|  | 634099e592 | ||
|  | 868f0474a9 | ||
|  | ced9f060c7 | ||
|  | 14b85318a6 | ||
|  | b179c2f208 | ||
|  | 3d76a974d1 | ||
|  | 4250064296 | ||
|  | 868d9a87d2 | ||
|  | 33846ce4f6 | ||
|  | e5ac80c15d | ||
|  | 9291b5fb20 | ||
|  | d97f1df3c9 | ||
|  | f0434c810c | ||
|  | f6fe34676f | ||
|  | 5c18c559c3 | ||
|  | 75545a1f47 | ||
|  | 72ea805f84 | ||
|  | 0e35050b8b | ||
|  | 24a4b323eb | ||
|  | aa0a9f2262 | ||
|  | 4010164db1 | ||
|  | dc94765d32 | ||
|  | 1cb1f727c0 | ||
|  | d97640374c | ||
|  | ba89abedf0 | ||
|  | a680b1b8b7 | ||
|  | b3b7d0a0ea | ||
|  | 8e805e23bc | ||
|  | bcbfacc04a | ||
|  | 5531e21526 | ||
|  | c5837c3bb7 | ||
|  | eb70b84665 | ||
|  | a909972313 | ||
|  | 6855d0dc39 | ||
|  | a43b1e2add | ||
|  | 46c43396d8 | ||
|  | 6dcffca065 | ||
|  | d754620ef7 | ||
|  | 21111126a2 | ||
|  | d91e7dcfdc | ||
|  | d79289ccdd | ||
|  | f89f6c7fa6 | ||
|  | b7d71b4f0a | ||
|  | 70ed126ccb | ||
|  | 57b213a035 | ||
|  | 549e944b95 | ||
|  | 0cdab80a6e | ||
|  | 760183a970 | ||
|  | 58fb18aace | ||
|  | 630156dc0a | ||
|  | 5f23f59d1c | ||
|  | 538a5d7a9b | ||
|  | 593e1926e9 | ||
|  | e87ad1f402 | ||
|  | 07cccdc8c0 | ||
|  | f71f01662c | ||
|  | 54d7a1c2e8 | ||
|  | f426f31bd7 | ||
|  | 2930577cd6 | ||
|  | e09512177a | ||
|  | d6dbaff3c2 | ||
|  | 7f9577a386 | ||
|  | 38668e7331 | ||
|  | 323f3d263a | ||
|  | 0c34ed4c61 | ||
|  | 7c7eb6b7ec | ||
|  | 8b2ef666ef | ||
|  | 955d5f8707 | ||
|  | 47ca449e32 | ||
|  | 39481eb6c0 | ||
|  | 69153e7231 | ||
|  | cdef10cad8 | ||
|  | 077853416d | ||
|  | 596446dba4 | ||
|  | 9d0bec83df | ||
|  | f97a9ce597 | ||
|  | 4339f45f74 | ||
|  | e398e0756b | ||
|  | 64db39320a | ||
|  | 0b4bf30908 | ||
|  | d29c273073 | ||
|  | 74f508e847 | ||
|  | 145bb14cb2 | ||
|  | 8901f03864 | ||
|  | 813bf0bd66 | ||
|  | 45e9fd66e7 | ||
|  | e0d0674f81 | ||
|  | 4b6adaec0b | ||
|  | 9301b3fed3 | ||
|  | c6edb78ac9 | ||
|  | 521ede2469 | ||
|  | 2c53424db8 | ||
|  | 2ad22e1425 | ||
|  | 502515bbbd | ||
|  | 1e1c6a828f | ||
|  | 2847a08852 | ||
|  | 98f1a627f0 | ||
|  | 333e4216d2 | ||
|  | 7e80e2da3a | ||
|  | 139624b8a4 | ||
|  | 2f44aaa645 | ||
|  | 0f6958c57a | ||
|  | 5f045f8cf5 | ||
|  | f19ee05351 | ||
|  | fa71daa8a7 | ||
|  | 54215dc303 | ||
|  | f9f42997b2 | ||
|  | 25eab0b224 | ||
|  | 34bce5b464 | ||
|  | d4794fc051 | ||
|  | 8b43e0dd3f | ||
|  | 92c88fa273 | ||
|  | 38191d55be | ||
|  | d9e39f5906 | ||
|  | 17b7646c12 | ||
|  | 171b818504 | ||
|  | bcca0cc0bc | ||
|  | b92ec5e54c | ||
|  | fa79e8b7a3 | ||
|  | 1cc7c20183 | ||
|  | 2eee97e9b6 | ||
|  | a3a1b612b0 | ||
|  | 61e682ca47 | ||
|  | b383983106 | ||
|  | cfd587117e | ||
|  | ef9dca28f5 | ||
|  | 741c0b9c18 | ||
|  | 3711f4a741 | ||
|  | 7c6bf3e97b | ||
|  | 481ba41fbd | ||
|  | 2779d6629c | ||
|  | e509899daf | ||
|  | b53cdbaf05 | ||
|  | ced89398a5 | ||
|  | 09c2e3bcec | ||
|  | 5cba800fa6 | ||
|  | 2d39a135f2 | ||
|  | 3c6834a79c | ||
|  | 6da3410823 | ||
|  | ceb289cb4d | ||
|  | 6f8cc712b0 | ||
|  | ad01e1f3b3 | ||
|  | cc1ef2ffd5 | ||
|  | 7201bd1c97 | ||
|  | 73d5e0f283 | ||
|  | efc744ca35 | ||
|  | e8da98139f | ||
|  | 519cb030f7 | ||
|  | 58fe923c85 | ||
|  | c9ac5e391f | ||
|  | 69cf1de7bd | ||
|  | 4d6172a242 | ||
|  | 8afdc56b11 | ||
|  | a9ea1d9d10 | ||
|  | ea8e7c517b | ||
|  | d1e9b86f05 | ||
|  | 6d1e5cb5dc | ||
|  | 01abed0a30 | ||
|  | 7c56a36a1c | ||
|  | c48327ff91 | ||
|  | a5406c6963 | ||
|  | a1f61384c5 | ||
|  | 44ebae1559 | ||
|  | aae92683d7 | ||
|  | cc3072c4df | ||
|  | bffee4e91d | ||
|  | 79dc53ff0d | ||
|  | 68e53d3e10 | ||
|  | d267211ee7 | ||
|  | 570b3bc71c | ||
|  | 225176aae9 | ||
|  | 443a22b75d | ||
|  | b44f0519a0 | ||
|  | 4a0e81fe83 | ||
|  | 976c29ea9f | ||
|  | 926951ee03 | ||
|  | 2cdc718fde | ||
|  | 57cb150177 | ||
|  | 6167e20b34 | ||
|  | 8835d8302e | ||
|  | 224bebe67a | ||
|  | cf6883778e | ||
|  | 246b981e23 | ||
|  | 2edd52e851 | ||
|  | e123c66bc7 | ||
|  | 9edc82bde0 | ||
|  | d84c2f5c70 | ||
|  | 46e77389a4 | ||
|  | f5f4e6fbc6 | ||
|  | dc4a6cb711 | ||
|  | 5798fdac50 | ||
|  | 3710688efd | ||
|  | 83e86b9f8a | ||
|  | 74c1ba7cbc | ||
|  | 73aa53f536 | ||
|  | da9ccb528d | ||
|  | 44729da277 | ||
|  | 7a3378b4b7 | ||
|  | fd19d7d246 | ||
|  | 5c694a1503 | ||
|  | 9edc54ca69 | ||
|  | e6af636fa0 | ||
|  | 6e1ef75009 | ||
|  | d9db16e999 | ||
|  | 241ade2fae | ||
|  | 80065de8a3 | ||
|  | 16f53b5afb | ||
|  | 3071300c0c | ||
|  | 8b056bf408 | ||
|  | e5640857b1 | ||
|  | 331177d97e | ||
|  | 4fed003f1a | ||
|  | a1ea1bf696 | ||
|  | 7c66fc6c21 | ||
|  | d93cb8f645 | ||
|  | b08cd7e104 | ||
|  | aea6c859e7 | ||
|  | 480e789cd8 | ||
|  | 23ec541ba6 | ||
|  | 053bb85a1c | ||
|  | 601fa5cea8 | ||
|  | 7a5057f02d | ||
|  | c76027a210 | ||
|  | f97c2b4c22 | ||
|  | 54b1e4adef | ||
|  | 9272884381 | ||
|  | 195e94a75d | ||
|  | 5bfc224669 | ||
|  | fd149c242f | ||
|  | b9cc5dfa3f | ||
|  | 8c305dc1bc | ||
|  | f62a671fbe | ||
|  | 9e2f2383b9 | ||
|  | e7a809b082 | ||
|  | 4f8cbd643d | ||
|  | 1dd92a3f92 | ||
|  | 34a3329f5f | ||
|  | 4fb07b6d6d | ||
|  | 8be7c9ae80 | ||
|  | 4e8dc8d0cf | ||
|  | 1e46b9d135 | ||
|  | f16a2a5645 | ||
|  | 03491029f2 | ||
|  | faf84d833d | ||
|  | 109736cc05 | ||
|  | eb8f43acb5 | ||
|  | 05dd7dfd2a | ||
|  | b874784058 | ||
|  | 284beed8dc | ||
|  | 69ee87c57f | ||
|  | b74a17c963 | ||
|  | 5d602e9b57 | ||
|  | f067f64a3a | ||
|  | 01c1b906b5 | ||
|  | f6194fa86c | ||
|  | abb2449b35 | ||
|  | cc5ef9871a | ||
|  | 9e30524e2a | ||
|  | 16271e7813 | 
							
								
								
									
										1
									
								
								.github/FUNDING.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								.github/FUNDING.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | ||||
| custom: ['https://iamazing.cn/page/reward'] | ||||
							
								
								
									
										24
									
								
								.github/ISSUE_TEMPLATE/bug_report.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								.github/ISSUE_TEMPLATE/bug_report.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,24 @@ | ||||
| --- | ||||
| name: 报告问题 | ||||
| about: 使用简练详细的语言描述你遇到的问题 | ||||
| title: '' | ||||
| labels: bug | ||||
| assignees: '' | ||||
|  | ||||
| --- | ||||
|  | ||||
| **例行检查** | ||||
| + [ ] 我已确认目前没有类似 issue | ||||
| + [ ] 我已确认我已升级到最新版本 | ||||
| + [ ] 我已完整查看过项目 README,尤其是常见问题部分 | ||||
| + [ ] 我理解并愿意跟进此 issue,协助测试和提供反馈  | ||||
| + [ ] 我理解并认可上述内容,并理解项目维护者精力有限,不遵循规则的 issue 可能会被无视或直接关闭 | ||||
|  | ||||
| **问题描述** | ||||
|  | ||||
| **复现步骤** | ||||
|  | ||||
| **预期结果** | ||||
|  | ||||
| **相关截图** | ||||
| 如果没有的话,请删除此节。 | ||||
							
								
								
									
										11
									
								
								.github/ISSUE_TEMPLATE/config.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								.github/ISSUE_TEMPLATE/config.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | ||||
| blank_issues_enabled: false | ||||
| contact_links: | ||||
|   - name: 项目群聊 | ||||
|     url: https://openai.justsong.cn/ | ||||
|     about: QQ 群:828520184,自动审核,备注 One API | ||||
|   - name: 赞赏支持 | ||||
|     url: https://iamazing.cn/page/reward | ||||
|     about: 请作者喝杯咖啡,以激励作者持续开发 | ||||
|   - name: 付费部署或定制功能 | ||||
|     url: https://openai.justsong.cn/ | ||||
|     about: 加群后联系群主 | ||||
							
								
								
									
										18
									
								
								.github/ISSUE_TEMPLATE/feature_request.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								.github/ISSUE_TEMPLATE/feature_request.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| --- | ||||
| name: 功能请求 | ||||
| about: 使用简练详细的语言描述希望加入的新功能 | ||||
| title: '' | ||||
| labels: enhancement | ||||
| assignees: '' | ||||
|  | ||||
| --- | ||||
|  | ||||
| **例行检查** | ||||
| + [ ] 我已确认目前没有类似 issue | ||||
| + [ ] 我已确认我已升级到最新版本 | ||||
| + [ ] 我理解并愿意跟进此 issue,协助测试和提供反馈 | ||||
| + [ ] 我理解并认可上述内容,并理解项目维护者精力有限,不遵循规则的 issue 可能会被无视或直接关闭 | ||||
|  | ||||
| **功能描述** | ||||
|  | ||||
| **应用场景** | ||||
							
								
								
									
										1
									
								
								.github/workflows/docker-image-arm64.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.github/workflows/docker-image-arm64.yml
									
									
									
									
										vendored
									
									
								
							| @@ -4,6 +4,7 @@ on: | ||||
|   push: | ||||
|     tags: | ||||
|       - '*' | ||||
|       - '!*-alpha*' | ||||
|   workflow_dispatch: | ||||
|     inputs: | ||||
|       name: | ||||
|   | ||||
							
								
								
									
										29
									
								
								.github/workflows/github-pages.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										29
									
								
								.github/workflows/github-pages.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,29 +0,0 @@ | ||||
| name: Build GitHub Pages | ||||
| on: | ||||
|   workflow_dispatch: | ||||
|     inputs: | ||||
|       name: | ||||
|         description: 'Reason' | ||||
|         required: false | ||||
| jobs: | ||||
|   build-and-deploy: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Checkout 🛎️ | ||||
|         uses: actions/checkout@v2 # If you're using actions/checkout@v2 you must set persist-credentials to false in most cases for the deployment to work correctly. | ||||
|         with: | ||||
|           persist-credentials: false | ||||
|       - name: Install and Build 🔧 # This example project is built using npm and outputs the result to the 'build' folder. Replace with the commands required to build your project, or remove this step entirely if your site is pre-built. | ||||
|         env: | ||||
|           CI: "" | ||||
|         run: | | ||||
|           cd web | ||||
|           npm install | ||||
|           npm run build | ||||
|  | ||||
|       - name: Deploy 🚀 | ||||
|         uses: JamesIves/github-pages-deploy-action@releases/v3 | ||||
|         with: | ||||
|           ACCESS_TOKEN: ${{ secrets.ACCESS_TOKEN }} | ||||
|           BRANCH: gh-pages # The branch the action should deploy to. | ||||
|           FOLDER: web/build # The folder the action should deploy. | ||||
							
								
								
									
										1
									
								
								.github/workflows/linux-release.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.github/workflows/linux-release.yml
									
									
									
									
										vendored
									
									
								
							| @@ -6,6 +6,7 @@ on: | ||||
|   push: | ||||
|     tags: | ||||
|       - '*' | ||||
|       - '!*-alpha*' | ||||
| jobs: | ||||
|   release: | ||||
|     runs-on: ubuntu-latest | ||||
|   | ||||
							
								
								
									
										1
									
								
								.github/workflows/macos-release.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.github/workflows/macos-release.yml
									
									
									
									
										vendored
									
									
								
							| @@ -6,6 +6,7 @@ on: | ||||
|   push: | ||||
|     tags: | ||||
|       - '*' | ||||
|       - '!*-alpha*' | ||||
| jobs: | ||||
|   release: | ||||
|     runs-on: macos-latest | ||||
|   | ||||
							
								
								
									
										1
									
								
								.github/workflows/windows-release.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.github/workflows/windows-release.yml
									
									
									
									
										vendored
									
									
								
							| @@ -6,6 +6,7 @@ on: | ||||
|   push: | ||||
|     tags: | ||||
|       - '*' | ||||
|       - '!*-alpha*' | ||||
| jobs: | ||||
|   release: | ||||
|     runs-on: windows-latest | ||||
|   | ||||
							
								
								
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -3,4 +3,5 @@ | ||||
| upload | ||||
| *.exe | ||||
| *.db | ||||
| build | ||||
| build | ||||
| *.db-journal | ||||
| @@ -24,7 +24,7 @@ RUN apk update \ | ||||
|     && apk upgrade \ | ||||
|     && apk add --no-cache ca-certificates tzdata \ | ||||
|     && update-ca-certificates 2>/dev/null || true | ||||
| ENV PORT=3000 | ||||
|  | ||||
| COPY --from=builder2 /build/one-api / | ||||
| EXPOSE 3000 | ||||
| WORKDIR /data | ||||
|   | ||||
							
								
								
									
										2
									
								
								LICENSE
									
									
									
									
									
								
							
							
						
						
									
										2
									
								
								LICENSE
									
									
									
									
									
								
							| @@ -1,6 +1,6 @@ | ||||
| MIT License | ||||
|  | ||||
| Copyright (c) 2022 JustSong | ||||
| Copyright (c) 2023 JustSong | ||||
|  | ||||
| Permission is hereby granted, free of charge, to any person obtaining a copy | ||||
| of this software and associated documentation files (the "Software"), to deal | ||||
|   | ||||
							
								
								
									
										201
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										201
									
								
								README.md
									
									
									
									
									
								
							| @@ -29,42 +29,101 @@ _✨ All in one 的 OpenAI 接口,整合各种 API 访问方式,开箱即用 | ||||
| </p> | ||||
|  | ||||
| <p align="center"> | ||||
|   <a href="https://github.com/songquanpeng/one-api/releases">程序下载</a> | ||||
|   · | ||||
|   <a href="https://github.com/songquanpeng/one-api#部署">部署教程</a> | ||||
|   · | ||||
|   <a href="https://github.com/songquanpeng/one-api#使用方法">使用方法</a> | ||||
|   · | ||||
|   <a href="https://github.com/songquanpeng/one-api/issues">意见反馈</a> | ||||
|   · | ||||
|   <a href="https://github.com/songquanpeng/one-api#截图展示">截图展示</a> | ||||
|   · | ||||
|   <a href="https://openai.justsong.cn/">在线演示</a> | ||||
|   · | ||||
|   <a href="https://github.com/songquanpeng/one-api#常见问题">常见问题</a> | ||||
|   · | ||||
|   <a href="https://iamazing.cn/page/reward">赞赏支持</a> | ||||
| </p> | ||||
|  | ||||
| > **Note**:使用 Docker 拉取的最新镜像可能是 `alpha` 版本,如果追求稳定性请手动指定版本。 | ||||
|  | ||||
| > **Warning**:从 `v0.3` 版本升级到 `v0.4` 版本需要手动迁移数据库,请手动执行[数据库迁移脚本](./bin/migration_v0.3-v0.4.sql)。 | ||||
|  | ||||
| ## 功能 | ||||
| 1. 支持多种 API 访问渠道,欢迎 PR 或提 issue 添加更多渠道: | ||||
|    + [x] One API 服务端中继 | ||||
|    + [x] [API2D](https://api2d.com/r/197971) | ||||
|    + [ ] Azure OpenAI API | ||||
|    + [x] [CloseAI](https://console.openai-asia.com) | ||||
|    + [x] OpenAI 官方通道(支持配置代理) | ||||
|    + [x] **Azure OpenAI API** | ||||
|    + [x] [OpenAI-SB](https://openai-sb.com) | ||||
|    + [x] [API2D](https://api2d.com/r/197971) | ||||
|    + [x] [OhMyGPT](https://aigptx.top?aff=uFpUl2Kf) | ||||
|    + [x] [AI Proxy](https://aiproxy.io/?i=OneAPI) (邀请码:`OneAPI`) | ||||
|    + [x] [API2GPT](http://console.api2gpt.com/m/00002S) | ||||
|    + [x] [CloseAI](https://console.closeai-asia.com/r/2412) | ||||
|    + [x] [AI.LS](https://ai.ls) | ||||
|    + [x] [OpenAI Max](https://openaimax.com) | ||||
|    + [x] [OhMyGPT](https://www.ohmygpt.com) | ||||
|    + [x] 自定义渠道 | ||||
| 2. 支持通过负载均衡的方式访问多个渠道。 | ||||
| 3. 支持单个访问渠道设置多个 API Key,利用起来你的多个 API Key。 | ||||
| 4. 支持设置令牌的过期时间和使用次数。 | ||||
| 5. 支持 HTTP SSE。 | ||||
| 6. 多种用户登录注册方式: | ||||
|    + 邮箱登录注册以及通过邮箱进行密码重置。 | ||||
|    + [GitHub 开放授权](https://github.com/settings/applications/new)。 | ||||
|    + 微信公众号授权(需要额外部署 [WeChat Server](https://github.com/songquanpeng/wechat-server))。 | ||||
| 7. 支持用户管理。 | ||||
|    + [x] 自定义渠道:例如各种未收录的第三方代理服务 | ||||
| 2. 支持通过**负载均衡**的方式访问多个渠道。 | ||||
| 3. 支持 **stream 模式**,可以通过流式传输实现打字机效果。 | ||||
| 4. 支持**多机部署**,[详见此处](#多机部署)。 | ||||
| 5. 支持**令牌管理**,设置令牌的过期时间和使用次数。 | ||||
| 6. 支持**兑换码管理**,支持批量生成和导出兑换码,可使用兑换码为账户进行充值。 | ||||
| 7. 支持**通道管理**,批量创建通道。 | ||||
| 8. 支持**用户分组**以及**渠道分组**,支持为不同分组设置不同的倍率。 | ||||
| 9. 支持渠道**设置模型列表**。 | ||||
| 10. 支持**查看额度明细**。 | ||||
| 11. 支持**用户邀请奖励**。 | ||||
| 12. 支持以美元为单位显示额度。 | ||||
| 13. 支持发布公告,设置充值链接,设置新用户初始额度。 | ||||
| 14. 支持丰富的**自定义**设置, | ||||
|     1. 支持自定义系统名称,logo 以及页脚。 | ||||
|     2. 支持自定义首页和关于页面,可以选择使用 HTML & Markdown 代码进行自定义,或者使用一个单独的网页通过 iframe 嵌入。 | ||||
| 15. 支持通过系统访问令牌访问管理 API。 | ||||
| 16. 支持 Cloudflare Turnstile 用户校验。 | ||||
| 17. 支持用户管理,支持**多种用户登录注册方式**: | ||||
|     + 邮箱登录注册以及通过邮箱进行密码重置。 | ||||
|     + [GitHub 开放授权](https://github.com/settings/applications/new)。 | ||||
|     + 微信公众号授权(需要额外部署 [WeChat Server](https://github.com/songquanpeng/wechat-server))。 | ||||
| 18. 未来其他大模型开放 API 后,将第一时间支持,并将其封装成同样的 API 访问方式。 | ||||
|  | ||||
| ## 部署 | ||||
| ### 基于 Docker 进行部署 | ||||
| 执行:`docker run -d --restart always -p 3000:3000 -v /home/ubuntu/data/one-api:/data -v /etc/ssl/certs:/etc/ssl/certs:ro justsong/one-api` | ||||
| 部署命令:`docker run --name one-api -d --restart always -p 3000:3000 -v /home/ubuntu/data/one-api:/data justsong/one-api` | ||||
|  | ||||
| 数据将会保存在宿主机的 `/home/ubuntu/data/one-api` 目录。 | ||||
| 更新命令:`docker run --rm -v /var/run/docker.sock:/var/run/docker.sock containrrr/watchtower -cR` | ||||
|  | ||||
| `-p 3000:3000` 中的第一个 `3000` 是宿主机的端口,可以根据需要进行修改。 | ||||
|  | ||||
| 数据将会保存在宿主机的 `/home/ubuntu/data/one-api` 目录,请确保该目录存在且具有写入权限,或者更改为合适的目录。 | ||||
|  | ||||
| Nginx 的参考配置: | ||||
| ``` | ||||
| server{ | ||||
|    server_name openai.justsong.cn;  # 请根据实际情况修改你的域名 | ||||
|     | ||||
|    location / { | ||||
|           client_max_body_size  64m; | ||||
|           proxy_http_version 1.1; | ||||
|           proxy_pass http://localhost:3000;  # 请根据实际情况修改你的端口 | ||||
|           proxy_set_header Host $host; | ||||
|           proxy_set_header X-Forwarded-For $remote_addr; | ||||
|           proxy_cache_bypass $http_upgrade; | ||||
|           proxy_set_header Accept-Encoding gzip; | ||||
|    } | ||||
| } | ||||
| ``` | ||||
|  | ||||
| 之后使用 Let's Encrypt 的 certbot 配置 HTTPS: | ||||
| ```bash | ||||
| # Ubuntu 安装 certbot: | ||||
| sudo snap install --classic certbot | ||||
| sudo ln -s /snap/bin/certbot /usr/bin/certbot | ||||
| # 生成证书 & 修改 Nginx 配置 | ||||
| sudo certbot --nginx | ||||
| # 根据指示进行操作 | ||||
| # 重启 Nginx | ||||
| sudo service nginx restart | ||||
| ``` | ||||
|  | ||||
| 初始账号用户名为 `root`,密码为 `123456`。 | ||||
|  | ||||
| ### 手动部署 | ||||
| 1. 从 [GitHub Releases](https://github.com/songquanpeng/one-api/releases/latest) 下载可执行文件或者从源码编译: | ||||
| @@ -90,6 +149,61 @@ _✨ All in one 的 OpenAI 接口,整合各种 API 访问方式,开箱即用 | ||||
|  | ||||
| 更加详细的部署教程[参见此处](https://iamazing.cn/page/how-to-deploy-a-website)。 | ||||
|  | ||||
| ### 多机部署 | ||||
| 1. 所有服务器 `SESSION_SECRET` 设置一样的值。 | ||||
| 2. 必须设置 `SQL_DSN`,使用 MySQL 数据库而非 SQLite,请自行配置主备数据库同步。 | ||||
| 3. 所有从服务器必须设置 `SYNC_FREQUENCY`,以定期从数据库同步配置。 | ||||
| 4. 从服务器可以选择设置 `FRONTEND_BASE_URL`,以重定向页面请求到主服务器。 | ||||
|  | ||||
| 环境变量的具体使用方法详见[此处](#环境变量)。 | ||||
|  | ||||
| ### 宝塔部署教程 | ||||
|  | ||||
| 详见 [#175](https://github.com/songquanpeng/one-api/issues/175)。 | ||||
|  | ||||
| 如果部署后访问出现空白页面,详见 [#97](https://github.com/songquanpeng/one-api/issues/97)。 | ||||
|  | ||||
| ### 部署第三方服务配合 One API 使用 | ||||
| > 欢迎 PR 添加更多示例。 | ||||
|  | ||||
| #### ChatGPT Next Web | ||||
| 项目主页:https://github.com/Yidadaa/ChatGPT-Next-Web | ||||
|  | ||||
| ```bash | ||||
| docker run --name chat-next-web -d -p 3001:3000 -e BASE_URL=https://openai.justsong.cn yidadaa/chatgpt-next-web | ||||
| ``` | ||||
|  | ||||
| 注意修改端口号和 `BASE_URL`。 | ||||
|  | ||||
| #### ChatGPT Web | ||||
| 项目主页:https://github.com/Chanzhaoyu/chatgpt-web | ||||
|  | ||||
| ```bash | ||||
| docker run --name chatgpt-web -d -p 3002:3002 -e OPENAI_API_BASE_URL=https://openai.justsong.cn -e OPENAI_API_KEY=sk-xxx chenzhaoyu94/chatgpt-web | ||||
| ``` | ||||
|  | ||||
| 注意修改端口号、`OPENAI_API_BASE_URL` 和 `OPENAI_API_KEY`。 | ||||
|  | ||||
| ### 部署到第三方平台 | ||||
| <details> | ||||
| <summary><strong>部署到 Zeabur</strong></summary> | ||||
| <div> | ||||
|  | ||||
| > Zeabur 的服务器在国外,自动解决了网络的问题,同时免费的额度也足够个人使用。 | ||||
|  | ||||
| 1. 首先 fork 一份代码。 | ||||
| 2. 进入 [Zeabur](https://zeabur.com/),登录,进入控制台。 | ||||
| 3. 新建一个 Project,在 Service -> Add Service 选择 Marketplace,选择 MySQL,并记下连接参数(用户名、密码、地址、端口)。 | ||||
| 4. 复制链接参数,运行 ```create database `one-api` ``` 创建数据库。 | ||||
| 5. 然后在 Service -> Add Service,选择 Git(第一次使用需要先授权),选择你 fork 的仓库。 | ||||
| 6. Deploy 会自动开始,先取消。进入下方 Variable,添加一个 `PORT`,值为 `3000`,再添加一个 `SQL_DSN`,值为 `<username>:<password>@tcp(<addr>:<port>)/one-api` ,然后保存。 注意如果不填写 `SQL_DSN`,数据将无法持久化,重新部署后数据会丢失。 | ||||
| 7. 选择 Redeploy。 | ||||
| 8. 进入下方 Domains,选择一个合适的域名前缀,如 "my-one-api",最终域名为 "my-one-api.zeabur.app",也可以 CNAME 自己的域名。 | ||||
| 9. 等待部署完成,点击生成的域名进入 One API。 | ||||
|  | ||||
| </div> | ||||
| </details> | ||||
|  | ||||
| ## 配置 | ||||
| 系统本身开箱即用。 | ||||
|  | ||||
| @@ -97,12 +211,26 @@ _✨ All in one 的 OpenAI 接口,整合各种 API 访问方式,开箱即用 | ||||
|  | ||||
| 等到系统启动后,使用 `root` 用户登录系统并做进一步的配置。 | ||||
|  | ||||
| ## 使用方式 | ||||
| 在`渠道`页面中添加你的 API Key,之后在`令牌`页面中新增一个访问令牌。 | ||||
| ## 使用方法 | ||||
| 在`渠道`页面中添加你的 API Key,之后在`令牌`页面中新增访问令牌。 | ||||
|  | ||||
| 之后就可以使用你的令牌访问 One API 了,使用方式与 [OpenAI API](https://platform.openai.com/docs/api-reference/introduction) 一致。 | ||||
|  | ||||
| 你需要在各种用到 OpenAI API 的地方设置 API Base 为你的 One API 的部署地址,例如:`https://openai.justsong.cn`,API Key 则为你在 One API 中生成的令牌。 | ||||
|  | ||||
| 注意,具体的 API Base 的格式取决于你所使用的客户端。 | ||||
|  | ||||
| ```mermaid | ||||
| graph LR | ||||
|     A(用户) | ||||
|     A --->|请求| B(One API) | ||||
|     B -->|中继请求| C(OpenAI) | ||||
|     B -->|中继请求| D(Azure) | ||||
|     B -->|中继请求| E(其他下游渠道) | ||||
| ``` | ||||
|  | ||||
| 可以通过在令牌后面添加渠道 ID 的方式指定使用哪一个渠道处理本次请求,例如:`Authorization: Bearer ONE_API_KEY-CHANNEL_ID`。 | ||||
| 注意,需要是管理员用户创建的令牌才能指定渠道 ID。 | ||||
|  | ||||
| 不加的话将会使用负载均衡的方式使用多个渠道。 | ||||
|  | ||||
| @@ -111,8 +239,12 @@ _✨ All in one 的 OpenAI 接口,整合各种 API 访问方式,开箱即用 | ||||
|    + 例子:`REDIS_CONN_STRING=redis://default:redispw@localhost:49153` | ||||
| 2. `SESSION_SECRET`:设置之后将使用固定的会话密钥,这样系统重新启动后已登录用户的 cookie 将依旧有效。 | ||||
|    + 例子:`SESSION_SECRET=random_string` | ||||
| 3. `SQL_DSN`:设置之后将使用指定数据库而非 SQLite。 | ||||
|    + 例子:`SQL_DSN=root:123456@tcp(localhost:3306)/one-api` | ||||
| 3. `SQL_DSN`:设置之后将使用指定数据库而非 SQLite,请使用 MySQL 8.0 版本。 | ||||
|    + 例子:`SQL_DSN=root:123456@tcp(localhost:3306)/oneapi` | ||||
| 4. `FRONTEND_BASE_URL`:设置之后将使用指定的前端地址,而非后端地址。 | ||||
|    + 例子:`FRONTEND_BASE_URL=https://openai.justsong.cn` | ||||
| 5. `SYNC_FREQUENCY`:设置之后将定期与数据库同步配置,单位为秒,未设置则不进行同步。 | ||||
|    + 例子:`SYNC_FREQUENCY=60` | ||||
|  | ||||
| ### 命令行参数 | ||||
| 1. `--port <port_number>`: 指定服务器监听的端口号,默认为 `3000`。 | ||||
| @@ -120,6 +252,7 @@ _✨ All in one 的 OpenAI 接口,整合各种 API 访问方式,开箱即用 | ||||
| 2. `--log-dir <log_dir>`: 指定日志文件夹,如果没有设置,日志将不会被保存。 | ||||
|    + 例子:`--log-dir ./logs` | ||||
| 3. `--version`: 打印系统版本号并退出。 | ||||
| 4. `--help`: 查看命令的使用帮助和参数说明。 | ||||
|  | ||||
| ## 演示 | ||||
| ### 在线演示 | ||||
| @@ -129,3 +262,25 @@ https://openai.justsong.cn | ||||
| ### 截图展示 | ||||
|  | ||||
|  | ||||
|  | ||||
| ## 常见问题 | ||||
| 1. 额度是什么?怎么计算的?One API 的额度计算有问题? | ||||
|    + 额度 = token * 倍率 | ||||
|    + 倍率包括分组的倍率,以及补全的倍率。 | ||||
|    + 如果是非流模式,官方接口会返回消耗的总 token,但是你要注意提示和补全的消耗额度不一样。 | ||||
| 2. 账户额度足够为什么提示额度不足? | ||||
|    + 请检查你的令牌额度是否足够,这个和账户额度是分开的。 | ||||
|    + 令牌额度仅供用户设置最大使用量,用户可自由设置。 | ||||
| 3. 提示无可用渠道? | ||||
|    + 请检查的用户分组和渠道分组设置。 | ||||
|    + 以及渠道的模型设置。 | ||||
| 4. 渠道测试报错:`invalid character '<' looking for beginning of value` | ||||
|    + 这是因为返回值不是合法的 JSON,而是一个 HTML 页面。 | ||||
|    + 大概率是你的部署站的 IP 或代理的节点被 CloudFlare 封禁了。 | ||||
|  | ||||
| ## 注意 | ||||
| 本项目为开源项目,请在遵循 OpenAI 的[使用条款](https://openai.com/policies/terms-of-use)以及**法律法规**的情况下使用,不得用于非法用途。 | ||||
|  | ||||
| 本项目使用 MIT 协议进行开源,请以某种方式保留 One API 的版权信息。 | ||||
|  | ||||
| 依据 MIT 协议,使用者需自行承担使用本项目的风险与责任,本开源项目开发者与此无关。 | ||||
							
								
								
									
										6
									
								
								bin/migration_v0.2-v0.3.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								bin/migration_v0.2-v0.3.sql
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | ||||
| UPDATE users | ||||
| SET quota = quota + ( | ||||
|     SELECT SUM(remain_quota) | ||||
|     FROM tokens | ||||
|     WHERE tokens.user_id = users.id | ||||
| ) | ||||
							
								
								
									
										17
									
								
								bin/migration_v0.3-v0.4.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								bin/migration_v0.3-v0.4.sql
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,17 @@ | ||||
| INSERT INTO abilities (`group`, model, channel_id, enabled) | ||||
| SELECT c.`group`, m.model, c.id, 1 | ||||
| FROM channels c | ||||
| CROSS JOIN ( | ||||
|     SELECT 'gpt-3.5-turbo' AS model UNION ALL | ||||
|     SELECT 'gpt-3.5-turbo-0301' AS model UNION ALL | ||||
|     SELECT 'gpt-4' AS model UNION ALL | ||||
|     SELECT 'gpt-4-0314' AS model | ||||
| ) AS m | ||||
| WHERE c.status = 1 | ||||
|   AND NOT EXISTS ( | ||||
|     SELECT 1 | ||||
|     FROM abilities a | ||||
|     WHERE a.`group` = c.`group` | ||||
|       AND a.model = m.model | ||||
|       AND a.channel_id = c.id | ||||
| ); | ||||
| @@ -1,9 +1,10 @@ | ||||
| package common | ||||
|  | ||||
| import ( | ||||
| 	"github.com/google/uuid" | ||||
| 	"sync" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/google/uuid" | ||||
| ) | ||||
|  | ||||
| var StartTime = time.Now().Unix() // unit: second | ||||
| @@ -11,6 +12,11 @@ var Version = "v0.0.0"            // this hard coding will be replaced automatic | ||||
| var SystemName = "One API" | ||||
| var ServerAddress = "http://localhost:3000" | ||||
| var Footer = "" | ||||
| var Logo = "" | ||||
| var TopUpLink = "" | ||||
| var ChatLink = "" | ||||
| var QuotaPerUnit = 500 * 1000.0 // $0.002 / 1K tokens | ||||
| var DisplayInCurrencyEnabled = false | ||||
|  | ||||
| var UsingSQLite = false | ||||
|  | ||||
| @@ -23,6 +29,7 @@ var OptionMap map[string]string | ||||
| var OptionMapRWMutex sync.RWMutex | ||||
|  | ||||
| var ItemsPerPage = 10 | ||||
| var MaxRecentItems = 100 | ||||
|  | ||||
| var PasswordLoginEnabled = true | ||||
| var PasswordRegisterEnabled = true | ||||
| @@ -32,8 +39,12 @@ var WeChatAuthEnabled = false | ||||
| var TurnstileCheckEnabled = false | ||||
| var RegisterEnabled = true | ||||
|  | ||||
| var LogConsumeEnabled = true | ||||
|  | ||||
| var SMTPServer = "" | ||||
| var SMTPPort = 587 | ||||
| var SMTPAccount = "" | ||||
| var SMTPFrom = "" | ||||
| var SMTPToken = "" | ||||
|  | ||||
| var GitHubClientId = "" | ||||
| @@ -46,6 +57,16 @@ var WeChatAccountQRCodeImageURL = "" | ||||
| var TurnstileSiteKey = "" | ||||
| var TurnstileSecretKey = "" | ||||
|  | ||||
| var QuotaForNewUser = 0 | ||||
| var QuotaForInviter = 0 | ||||
| var QuotaForInvitee = 0 | ||||
| var ChannelDisableThreshold = 5.0 | ||||
| var AutomaticDisableChannelEnabled = false | ||||
| var QuotaRemindThreshold = 1000 | ||||
| var PreConsumedQuota = 500 | ||||
|  | ||||
| var RootUserEmail = "" | ||||
|  | ||||
| const ( | ||||
| 	RoleGuestUser  = 0 | ||||
| 	RoleCommonUser = 1 | ||||
| @@ -63,7 +84,7 @@ var ( | ||||
| // All duration's unit is seconds | ||||
| // Shouldn't larger then RateLimitKeyExpirationDuration | ||||
| var ( | ||||
| 	GlobalApiRateLimitNum            = 60000 // TODO: temporary set to 60000 | ||||
| 	GlobalApiRateLimitNum            = 180 | ||||
| 	GlobalApiRateLimitDuration int64 = 3 * 60 | ||||
|  | ||||
| 	GlobalWebRateLimitNum            = 60 | ||||
| @@ -93,6 +114,12 @@ const ( | ||||
| 	TokenStatusExhausted = 4 | ||||
| ) | ||||
|  | ||||
| const ( | ||||
| 	RedemptionCodeStatusEnabled  = 1 // don't use 0, 0 is the default value! | ||||
| 	RedemptionCodeStatusDisabled = 2 // also don't use 0 | ||||
| 	RedemptionCodeStatusUsed     = 3 // also don't use 0 | ||||
| ) | ||||
|  | ||||
| const ( | ||||
| 	ChannelStatusUnknown  = 0 | ||||
| 	ChannelStatusEnabled  = 1 // don't use 0, 0 is the default value! | ||||
| @@ -109,16 +136,24 @@ const ( | ||||
| 	ChannelTypeOpenAIMax = 6 | ||||
| 	ChannelTypeOhMyGPT   = 7 | ||||
| 	ChannelTypeCustom    = 8 | ||||
| 	ChannelTypeAILS      = 9 | ||||
| 	ChannelTypeAIProxy   = 10 | ||||
| 	ChannelTypePaLM      = 11 | ||||
| 	ChannelTypeAPI2GPT   = 12 | ||||
| ) | ||||
|  | ||||
| var ChannelBaseURLs = []string{ | ||||
| 	"",                            // 0 | ||||
| 	"https://api.openai.com",      // 1 | ||||
| 	"https://openai.api2d.net",    // 2 | ||||
| 	"",                            // 3 | ||||
| 	"https://api.openai-asia.com", // 4 | ||||
| 	"https://api.openai-sb.com",   // 5 | ||||
| 	"https://api.openaimax.com",   // 6 | ||||
| 	"https://api.ohmygpt.com",     // 7 | ||||
| 	"",                            // 8 | ||||
| 	"",                             // 0 | ||||
| 	"https://api.openai.com",       // 1 | ||||
| 	"https://oa.api2d.net",         // 2 | ||||
| 	"",                             // 3 | ||||
| 	"https://api.openai-proxy.org", // 4 | ||||
| 	"https://api.openai-sb.com",    // 5 | ||||
| 	"https://api.openaimax.com",    // 6 | ||||
| 	"https://api.ohmygpt.com",      // 7 | ||||
| 	"",                             // 8 | ||||
| 	"https://api.caipacity.com",    // 9 | ||||
| 	"https://api.aiproxy.io",       // 10 | ||||
| 	"",                             // 11 | ||||
| 	"https://api.api2gpt.com",      // 12 | ||||
| } | ||||
|   | ||||
							
								
								
									
										82
									
								
								common/custom-event.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										82
									
								
								common/custom-event.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,82 @@ | ||||
| // Copyright 2014 Manu Martinez-Almeida.  All rights reserved. | ||||
| // Use of this source code is governed by a MIT style | ||||
| // license that can be found in the LICENSE file. | ||||
|  | ||||
| package common | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"io" | ||||
| 	"net/http" | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| type stringWriter interface { | ||||
| 	io.Writer | ||||
| 	writeString(string) (int, error) | ||||
| } | ||||
|  | ||||
| type stringWrapper struct { | ||||
| 	io.Writer | ||||
| } | ||||
|  | ||||
| func (w stringWrapper) writeString(str string) (int, error) { | ||||
| 	return w.Writer.Write([]byte(str)) | ||||
| } | ||||
|  | ||||
| func checkWriter(writer io.Writer) stringWriter { | ||||
| 	if w, ok := writer.(stringWriter); ok { | ||||
| 		return w | ||||
| 	} else { | ||||
| 		return stringWrapper{writer} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // Server-Sent Events | ||||
| // W3C Working Draft 29 October 2009 | ||||
| // http://www.w3.org/TR/2009/WD-eventsource-20091029/ | ||||
|  | ||||
| var contentType = []string{"text/event-stream"} | ||||
| var noCache = []string{"no-cache"} | ||||
|  | ||||
| var fieldReplacer = strings.NewReplacer( | ||||
| 	"\n", "\\n", | ||||
| 	"\r", "\\r") | ||||
|  | ||||
| var dataReplacer = strings.NewReplacer( | ||||
| 	"\n", "\ndata:", | ||||
| 	"\r", "\\r") | ||||
|  | ||||
| type CustomEvent struct { | ||||
| 	Event string | ||||
| 	Id    string | ||||
| 	Retry uint | ||||
| 	Data  interface{} | ||||
| } | ||||
|  | ||||
| func encode(writer io.Writer, event CustomEvent) error { | ||||
| 	w := checkWriter(writer) | ||||
| 	return writeData(w, event.Data) | ||||
| } | ||||
|  | ||||
| func writeData(w stringWriter, data interface{}) error { | ||||
| 	dataReplacer.WriteString(w, fmt.Sprint(data)) | ||||
| 	if strings.HasPrefix(data.(string), "data") { | ||||
| 		w.writeString("\n\n") | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (r CustomEvent) Render(w http.ResponseWriter) error { | ||||
| 	r.WriteContentType(w) | ||||
| 	return encode(w, r) | ||||
| } | ||||
|  | ||||
| func (r CustomEvent) WriteContentType(w http.ResponseWriter) { | ||||
| 	header := w.Header() | ||||
| 	header["Content-Type"] = contentType | ||||
|  | ||||
| 	if _, exist := header["Cache-Control"]; !exist { | ||||
| 		header["Cache-Control"] = noCache | ||||
| 	} | ||||
| } | ||||
| @@ -1,14 +1,67 @@ | ||||
| package common | ||||
|  | ||||
| import "gopkg.in/gomail.v2" | ||||
| import ( | ||||
| 	"crypto/tls" | ||||
| 	"encoding/base64" | ||||
| 	"fmt" | ||||
| 	"net/smtp" | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| func SendEmail(subject string, receiver string, content string) error { | ||||
| 	m := gomail.NewMessage() | ||||
| 	m.SetHeader("From", SMTPAccount) | ||||
| 	m.SetHeader("To", receiver) | ||||
| 	m.SetHeader("Subject", subject) | ||||
| 	m.SetBody("text/html", content) | ||||
| 	d := gomail.NewDialer(SMTPServer, 587, SMTPAccount, SMTPToken) | ||||
| 	err := d.DialAndSend(m) | ||||
| 	if SMTPFrom == "" { // for compatibility | ||||
| 		SMTPFrom = SMTPAccount | ||||
| 	} | ||||
| 	encodedSubject := fmt.Sprintf("=?UTF-8?B?%s?=", base64.StdEncoding.EncodeToString([]byte(subject))) | ||||
| 	mail := []byte(fmt.Sprintf("To: %s\r\n"+ | ||||
| 		"From: %s<%s>\r\n"+ | ||||
| 		"Subject: %s\r\n"+ | ||||
| 		"Content-Type: text/html; charset=UTF-8\r\n\r\n%s\r\n", | ||||
| 		receiver, SystemName, SMTPFrom, encodedSubject, content)) | ||||
| 	auth := smtp.PlainAuth("", SMTPAccount, SMTPToken, SMTPServer) | ||||
| 	addr := fmt.Sprintf("%s:%d", SMTPServer, SMTPPort) | ||||
| 	to := strings.Split(receiver, ";") | ||||
| 	var err error | ||||
| 	if SMTPPort == 465 { | ||||
| 		tlsConfig := &tls.Config{ | ||||
| 			InsecureSkipVerify: true, | ||||
| 			ServerName:         SMTPServer, | ||||
| 		} | ||||
| 		conn, err := tls.Dial("tcp", fmt.Sprintf("%s:%d", SMTPServer, SMTPPort), tlsConfig) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		client, err := smtp.NewClient(conn, SMTPServer) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		defer client.Close() | ||||
| 		if err = client.Auth(auth); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		if err = client.Mail(SMTPFrom); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		receiverEmails := strings.Split(receiver, ";") | ||||
| 		for _, receiver := range receiverEmails { | ||||
| 			if err = client.Rcpt(receiver); err != nil { | ||||
| 				return err | ||||
| 			} | ||||
| 		} | ||||
| 		w, err := client.Data() | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		_, err = w.Write(mail) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		err = w.Close() | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 	} else { | ||||
| 		err = smtp.SendMail(addr, auth, SMTPAccount, to, mail) | ||||
| 	} | ||||
| 	return err | ||||
| } | ||||
|   | ||||
							
								
								
									
										26
									
								
								common/gin.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								common/gin.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,26 @@ | ||||
| package common | ||||
|  | ||||
| import ( | ||||
| 	"bytes" | ||||
| 	"encoding/json" | ||||
| 	"github.com/gin-gonic/gin" | ||||
| 	"io" | ||||
| ) | ||||
|  | ||||
| func UnmarshalBodyReusable(c *gin.Context, v any) error { | ||||
| 	requestBody, err := io.ReadAll(c.Request.Body) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	err = c.Request.Body.Close() | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	err = json.Unmarshal(requestBody, &v) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	// Reset request body | ||||
| 	c.Request.Body = io.NopCloser(bytes.NewBuffer(requestBody)) | ||||
| 	return nil | ||||
| } | ||||
							
								
								
									
										31
									
								
								common/group-ratio.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								common/group-ratio.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,31 @@ | ||||
| package common | ||||
|  | ||||
| import "encoding/json" | ||||
|  | ||||
| var GroupRatio = map[string]float64{ | ||||
| 	"default": 1, | ||||
| 	"vip":     1, | ||||
| 	"svip":    1, | ||||
| } | ||||
|  | ||||
| func GroupRatio2JSONString() string { | ||||
| 	jsonBytes, err := json.Marshal(GroupRatio) | ||||
| 	if err != nil { | ||||
| 		SysError("Error marshalling model ratio: " + err.Error()) | ||||
| 	} | ||||
| 	return string(jsonBytes) | ||||
| } | ||||
|  | ||||
| func UpdateGroupRatioByJSONString(jsonStr string) error { | ||||
| 	GroupRatio = make(map[string]float64) | ||||
| 	return json.Unmarshal([]byte(jsonStr), &GroupRatio) | ||||
| } | ||||
|  | ||||
| func GetGroupRatio(name string) float64 { | ||||
| 	ratio, ok := GroupRatio[name] | ||||
| 	if !ok { | ||||
| 		SysError("Group ratio not found: " + name) | ||||
| 		return 1 | ||||
| 	} | ||||
| 	return ratio | ||||
| } | ||||
| @@ -42,3 +42,11 @@ func FatalLog(v ...any) { | ||||
| 	_, _ = fmt.Fprintf(gin.DefaultErrorWriter, "[FATAL] %v | %v \n", t.Format("2006/01/02 - 15:04:05"), v) | ||||
| 	os.Exit(1) | ||||
| } | ||||
|  | ||||
| func LogQuota(quota int) string { | ||||
| 	if DisplayInCurrencyEnabled { | ||||
| 		return fmt.Sprintf("$%.6f 额度", float64(quota)/QuotaPerUnit) | ||||
| 	} else { | ||||
| 		return fmt.Sprintf("%d 点额度", quota) | ||||
| 	} | ||||
| } | ||||
|   | ||||
							
								
								
									
										60
									
								
								common/model-ratio.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										60
									
								
								common/model-ratio.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,60 @@ | ||||
| package common | ||||
|  | ||||
| import "encoding/json" | ||||
|  | ||||
| // ModelRatio | ||||
| // https://platform.openai.com/docs/models/model-endpoint-compatibility | ||||
| // https://openai.com/pricing | ||||
| // TODO: when a new api is enabled, check the pricing here | ||||
| // 1 === $0.002 / 1K tokens | ||||
| var ModelRatio = map[string]float64{ | ||||
| 	"gpt-4":                   15, | ||||
| 	"gpt-4-0314":              15, | ||||
| 	"gpt-4-0613":              15, | ||||
| 	"gpt-4-32k":               30, | ||||
| 	"gpt-4-32k-0314":          30, | ||||
| 	"gpt-4-32k-0613":          30, | ||||
| 	"gpt-3.5-turbo":           0.75, // $0.0015 / 1K tokens | ||||
| 	"gpt-3.5-turbo-0301":      0.75, | ||||
| 	"gpt-3.5-turbo-0613":      0.75, | ||||
| 	"gpt-3.5-turbo-16k":       1.5, // $0.003 / 1K tokens | ||||
| 	"gpt-3.5-turbo-16k-0613":  1.5, | ||||
| 	"text-ada-001":            0.2, | ||||
| 	"text-babbage-001":        0.25, | ||||
| 	"text-curie-001":          1, | ||||
| 	"text-davinci-002":        10, | ||||
| 	"text-davinci-003":        10, | ||||
| 	"text-davinci-edit-001":   10, | ||||
| 	"code-davinci-edit-001":   10, | ||||
| 	"whisper-1":               10, | ||||
| 	"davinci":                 10, | ||||
| 	"curie":                   10, | ||||
| 	"babbage":                 10, | ||||
| 	"ada":                     10, | ||||
| 	"text-embedding-ada-002":  0.2, | ||||
| 	"text-search-ada-doc-001": 10, | ||||
| 	"text-moderation-stable":  0.1, | ||||
| 	"text-moderation-latest":  0.1, | ||||
| } | ||||
|  | ||||
| func ModelRatio2JSONString() string { | ||||
| 	jsonBytes, err := json.Marshal(ModelRatio) | ||||
| 	if err != nil { | ||||
| 		SysError("Error marshalling model ratio: " + err.Error()) | ||||
| 	} | ||||
| 	return string(jsonBytes) | ||||
| } | ||||
|  | ||||
| func UpdateModelRatioByJSONString(jsonStr string) error { | ||||
| 	ModelRatio = make(map[string]float64) | ||||
| 	return json.Unmarshal([]byte(jsonStr), &ModelRatio) | ||||
| } | ||||
|  | ||||
| func GetModelRatio(name string) float64 { | ||||
| 	ratio, ok := ModelRatio[name] | ||||
| 	if !ok { | ||||
| 		SysError("Model ratio not found: " + name) | ||||
| 		return 30 | ||||
| 	} | ||||
| 	return ratio | ||||
| } | ||||
| @@ -37,3 +37,18 @@ func ParseRedisOption() *redis.Options { | ||||
| 	} | ||||
| 	return opt | ||||
| } | ||||
|  | ||||
| func RedisSet(key string, value string, expiration time.Duration) error { | ||||
| 	ctx := context.Background() | ||||
| 	return RDB.Set(ctx, key, value, expiration).Err() | ||||
| } | ||||
|  | ||||
| func RedisGet(key string) (string, error) { | ||||
| 	ctx := context.Background() | ||||
| 	return RDB.Get(ctx, key).Result() | ||||
| } | ||||
|  | ||||
| func RedisDel(key string) error { | ||||
| 	ctx := context.Background() | ||||
| 	return RDB.Del(ctx, key).Err() | ||||
| } | ||||
|   | ||||
| @@ -5,6 +5,7 @@ import ( | ||||
| 	"github.com/google/uuid" | ||||
| 	"html/template" | ||||
| 	"log" | ||||
| 	"math/rand" | ||||
| 	"net" | ||||
| 	"os/exec" | ||||
| 	"runtime" | ||||
| @@ -133,6 +134,38 @@ func GetUUID() string { | ||||
| 	return code | ||||
| } | ||||
|  | ||||
| const keyChars = "0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ" | ||||
|  | ||||
| func init() { | ||||
| 	rand.Seed(time.Now().UnixNano()) | ||||
| } | ||||
|  | ||||
| func GenerateKey() string { | ||||
| 	rand.Seed(time.Now().UnixNano()) | ||||
| 	key := make([]byte, 48) | ||||
| 	for i := 0; i < 16; i++ { | ||||
| 		key[i] = keyChars[rand.Intn(len(keyChars))] | ||||
| 	} | ||||
| 	uuid_ := GetUUID() | ||||
| 	for i := 0; i < 32; i++ { | ||||
| 		c := uuid_[i] | ||||
| 		if i%2 == 0 && c >= 'a' && c <= 'z' { | ||||
| 			c = c - 'a' + 'A' | ||||
| 		} | ||||
| 		key[i+16] = c | ||||
| 	} | ||||
| 	return string(key) | ||||
| } | ||||
|  | ||||
| func GetRandomString(length int) string { | ||||
| 	rand.Seed(time.Now().UnixNano()) | ||||
| 	key := make([]byte, length) | ||||
| 	for i := 0; i < length; i++ { | ||||
| 		key[i] = keyChars[rand.Intn(len(keyChars))] | ||||
| 	} | ||||
| 	return string(key) | ||||
| } | ||||
|  | ||||
| func GetTimestamp() int64 { | ||||
| 	return time.Now().Unix() | ||||
| } | ||||
|   | ||||
							
								
								
									
										60
									
								
								controller/billing.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										60
									
								
								controller/billing.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,60 @@ | ||||
| package controller | ||||
|  | ||||
| import ( | ||||
| 	"github.com/gin-gonic/gin" | ||||
| 	"one-api/common" | ||||
| 	"one-api/model" | ||||
| ) | ||||
|  | ||||
| func GetSubscription(c *gin.Context) { | ||||
| 	userId := c.GetInt("id") | ||||
| 	quota, err := model.GetUserQuota(userId) | ||||
| 	if err != nil { | ||||
| 		openAIError := OpenAIError{ | ||||
| 			Message: err.Error(), | ||||
| 			Type:    "one_api_error", | ||||
| 		} | ||||
| 		c.JSON(200, gin.H{ | ||||
| 			"error": openAIError, | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	amount := float64(quota) | ||||
| 	if common.DisplayInCurrencyEnabled { | ||||
| 		amount /= common.QuotaPerUnit | ||||
| 	} | ||||
| 	subscription := OpenAISubscriptionResponse{ | ||||
| 		Object:             "billing_subscription", | ||||
| 		HasPaymentMethod:   true, | ||||
| 		SoftLimitUSD:       amount, | ||||
| 		HardLimitUSD:       amount, | ||||
| 		SystemHardLimitUSD: amount, | ||||
| 	} | ||||
| 	c.JSON(200, subscription) | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func GetUsage(c *gin.Context) { | ||||
| 	userId := c.GetInt("id") | ||||
| 	quota, err := model.GetUserUsedQuota(userId) | ||||
| 	if err != nil { | ||||
| 		openAIError := OpenAIError{ | ||||
| 			Message: err.Error(), | ||||
| 			Type:    "one_api_error", | ||||
| 		} | ||||
| 		c.JSON(200, gin.H{ | ||||
| 			"error": openAIError, | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	amount := float64(quota) | ||||
| 	if common.DisplayInCurrencyEnabled { | ||||
| 		amount /= common.QuotaPerUnit | ||||
| 	} | ||||
| 	usage := OpenAIUsageResponse{ | ||||
| 		Object:     "list", | ||||
| 		TotalUsage: amount, | ||||
| 	} | ||||
| 	c.JSON(200, usage) | ||||
| 	return | ||||
| } | ||||
							
								
								
									
										279
									
								
								controller/channel-billing.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										279
									
								
								controller/channel-billing.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,279 @@ | ||||
| package controller | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"io" | ||||
| 	"net/http" | ||||
| 	"one-api/common" | ||||
| 	"one-api/model" | ||||
| 	"strconv" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/gin-gonic/gin" | ||||
| ) | ||||
|  | ||||
| // https://github.com/songquanpeng/one-api/issues/79 | ||||
|  | ||||
| type OpenAISubscriptionResponse struct { | ||||
| 	Object             string  `json:"object"` | ||||
| 	HasPaymentMethod   bool    `json:"has_payment_method"` | ||||
| 	SoftLimitUSD       float64 `json:"soft_limit_usd"` | ||||
| 	HardLimitUSD       float64 `json:"hard_limit_usd"` | ||||
| 	SystemHardLimitUSD float64 `json:"system_hard_limit_usd"` | ||||
| } | ||||
|  | ||||
| type OpenAIUsageDailyCost struct { | ||||
| 	Timestamp float64 `json:"timestamp"` | ||||
| 	LineItems []struct { | ||||
| 		Name string  `json:"name"` | ||||
| 		Cost float64 `json:"cost"` | ||||
| 	} | ||||
| } | ||||
|  | ||||
| type OpenAIUsageResponse struct { | ||||
| 	Object string `json:"object"` | ||||
| 	//DailyCosts []OpenAIUsageDailyCost `json:"daily_costs"` | ||||
| 	TotalUsage float64 `json:"total_usage"` // unit: 0.01 dollar | ||||
| } | ||||
|  | ||||
| type OpenAISBUsageResponse struct { | ||||
| 	Msg  string `json:"msg"` | ||||
| 	Data *struct { | ||||
| 		Credit string `json:"credit"` | ||||
| 	} `json:"data"` | ||||
| } | ||||
|  | ||||
| type AIProxyUserOverviewResponse struct { | ||||
| 	Success   bool   `json:"success"` | ||||
| 	Message   string `json:"message"` | ||||
| 	ErrorCode int    `json:"error_code"` | ||||
| 	Data      struct { | ||||
| 		TotalPoints float64 `json:"totalPoints"` | ||||
| 	} `json:"data"` | ||||
| } | ||||
|  | ||||
| type API2GPTUsageResponse struct { | ||||
| 	Object         string  `json:"object"` | ||||
| 	TotalGranted   float64 `json:"total_granted"` | ||||
| 	TotalUsed      float64 `json:"total_used"` | ||||
| 	TotalRemaining float64 `json:"total_remaining"` | ||||
| } | ||||
|  | ||||
| // GetAuthHeader get auth header | ||||
| func GetAuthHeader(token string) http.Header { | ||||
| 	h := http.Header{} | ||||
| 	h.Add("Authorization", fmt.Sprintf("Bearer %s", token)) | ||||
| 	return h | ||||
| } | ||||
|  | ||||
| func GetResponseBody(method, url string, channel *model.Channel, headers http.Header) ([]byte, error) { | ||||
| 	client := &http.Client{} | ||||
| 	req, err := http.NewRequest(method, url, nil) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	for k := range headers { | ||||
| 		req.Header.Add(k, headers.Get(k)) | ||||
| 	} | ||||
| 	res, err := client.Do(req) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	body, err := io.ReadAll(res.Body) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	err = res.Body.Close() | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return body, nil | ||||
| } | ||||
|  | ||||
| func updateChannelOpenAISBBalance(channel *model.Channel) (float64, error) { | ||||
| 	url := fmt.Sprintf("https://api.openai-sb.com/sb-api/user/status?api_key=%s", channel.Key) | ||||
| 	body, err := GetResponseBody("GET", url, channel, GetAuthHeader(channel.Key)) | ||||
| 	if err != nil { | ||||
| 		return 0, err | ||||
| 	} | ||||
| 	response := OpenAISBUsageResponse{} | ||||
| 	err = json.Unmarshal(body, &response) | ||||
| 	if err != nil { | ||||
| 		return 0, err | ||||
| 	} | ||||
| 	if response.Data == nil { | ||||
| 		return 0, errors.New(response.Msg) | ||||
| 	} | ||||
| 	balance, err := strconv.ParseFloat(response.Data.Credit, 64) | ||||
| 	if err != nil { | ||||
| 		return 0, err | ||||
| 	} | ||||
| 	channel.UpdateBalance(balance) | ||||
| 	return balance, nil | ||||
| } | ||||
|  | ||||
| func updateChannelAIProxyBalance(channel *model.Channel) (float64, error) { | ||||
| 	url := "https://aiproxy.io/api/report/getUserOverview" | ||||
| 	headers := http.Header{} | ||||
| 	headers.Add("Api-Key", channel.Key) | ||||
| 	body, err := GetResponseBody("GET", url, channel, headers) | ||||
| 	if err != nil { | ||||
| 		return 0, err | ||||
| 	} | ||||
| 	response := AIProxyUserOverviewResponse{} | ||||
| 	err = json.Unmarshal(body, &response) | ||||
| 	if err != nil { | ||||
| 		return 0, err | ||||
| 	} | ||||
| 	if !response.Success { | ||||
| 		return 0, fmt.Errorf("code: %d, message: %s", response.ErrorCode, response.Message) | ||||
| 	} | ||||
| 	channel.UpdateBalance(response.Data.TotalPoints) | ||||
| 	return response.Data.TotalPoints, nil | ||||
| } | ||||
|  | ||||
| func updateChannelAPI2GPTBalance(channel *model.Channel) (float64, error) { | ||||
| 	url := "https://api.api2gpt.com/dashboard/billing/credit_grants" | ||||
| 	body, err := GetResponseBody("GET", url, channel, GetAuthHeader(channel.Key)) | ||||
|  | ||||
| 	if err != nil { | ||||
| 		return 0, err | ||||
| 	} | ||||
| 	response := API2GPTUsageResponse{} | ||||
| 	err = json.Unmarshal(body, &response) | ||||
| 	if err != nil { | ||||
| 		return 0, err | ||||
| 	} | ||||
| 	channel.UpdateBalance(response.TotalRemaining) | ||||
| 	return response.TotalRemaining, nil | ||||
| } | ||||
|  | ||||
| func updateChannelBalance(channel *model.Channel) (float64, error) { | ||||
| 	baseURL := common.ChannelBaseURLs[channel.Type] | ||||
| 	switch channel.Type { | ||||
| 	case common.ChannelTypeOpenAI: | ||||
| 		if channel.BaseURL != "" { | ||||
| 			baseURL = channel.BaseURL | ||||
| 		} | ||||
| 	case common.ChannelTypeAzure: | ||||
| 		return 0, errors.New("尚未实现") | ||||
| 	case common.ChannelTypeCustom: | ||||
| 		baseURL = channel.BaseURL | ||||
| 	case common.ChannelTypeOpenAISB: | ||||
| 		return updateChannelOpenAISBBalance(channel) | ||||
| 	case common.ChannelTypeAIProxy: | ||||
| 		return updateChannelAIProxyBalance(channel) | ||||
| 	case common.ChannelTypeAPI2GPT: | ||||
| 		return updateChannelAPI2GPTBalance(channel) | ||||
| 	default: | ||||
| 		return 0, errors.New("尚未实现") | ||||
| 	} | ||||
| 	url := fmt.Sprintf("%s/v1/dashboard/billing/subscription", baseURL) | ||||
|  | ||||
| 	body, err := GetResponseBody("GET", url, channel, GetAuthHeader(channel.Key)) | ||||
| 	if err != nil { | ||||
| 		return 0, err | ||||
| 	} | ||||
| 	subscription := OpenAISubscriptionResponse{} | ||||
| 	err = json.Unmarshal(body, &subscription) | ||||
| 	if err != nil { | ||||
| 		return 0, err | ||||
| 	} | ||||
| 	now := time.Now() | ||||
| 	startDate := fmt.Sprintf("%s-01", now.Format("2006-01")) | ||||
| 	endDate := now.Format("2006-01-02") | ||||
| 	if !subscription.HasPaymentMethod { | ||||
| 		startDate = now.AddDate(0, 0, -100).Format("2006-01-02") | ||||
| 	} | ||||
| 	url = fmt.Sprintf("%s/v1/dashboard/billing/usage?start_date=%s&end_date=%s", baseURL, startDate, endDate) | ||||
| 	body, err = GetResponseBody("GET", url, channel, GetAuthHeader(channel.Key)) | ||||
| 	if err != nil { | ||||
| 		return 0, err | ||||
| 	} | ||||
| 	usage := OpenAIUsageResponse{} | ||||
| 	err = json.Unmarshal(body, &usage) | ||||
| 	if err != nil { | ||||
| 		return 0, err | ||||
| 	} | ||||
| 	balance := subscription.HardLimitUSD - usage.TotalUsage/100 | ||||
| 	channel.UpdateBalance(balance) | ||||
| 	return balance, nil | ||||
| } | ||||
|  | ||||
| func UpdateChannelBalance(c *gin.Context) { | ||||
| 	id, err := strconv.Atoi(c.Param("id")) | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	channel, err := model.GetChannelById(id, true) | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	balance, err := updateChannelBalance(channel) | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	c.JSON(http.StatusOK, gin.H{ | ||||
| 		"success": true, | ||||
| 		"message": "", | ||||
| 		"balance": balance, | ||||
| 	}) | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func updateAllChannelsBalance() error { | ||||
| 	channels, err := model.GetAllChannels(0, 0, true) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	for _, channel := range channels { | ||||
| 		if channel.Status != common.ChannelStatusEnabled { | ||||
| 			continue | ||||
| 		} | ||||
| 		// TODO: support Azure | ||||
| 		if channel.Type != common.ChannelTypeOpenAI && channel.Type != common.ChannelTypeCustom { | ||||
| 			continue | ||||
| 		} | ||||
| 		balance, err := updateChannelBalance(channel) | ||||
| 		if err != nil { | ||||
| 			continue | ||||
| 		} else { | ||||
| 			// err is nil & balance <= 0 means quota is used up | ||||
| 			if balance <= 0 { | ||||
| 				disableChannel(channel.Id, channel.Name, "余额不足") | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func UpdateAllChannelsBalance(c *gin.Context) { | ||||
| 	// TODO: make it async | ||||
| 	err := updateAllChannelsBalance() | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	c.JSON(http.StatusOK, gin.H{ | ||||
| 		"success": true, | ||||
| 		"message": "", | ||||
| 	}) | ||||
| 	return | ||||
| } | ||||
							
								
								
									
										202
									
								
								controller/channel-test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										202
									
								
								controller/channel-test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,202 @@ | ||||
| package controller | ||||
|  | ||||
| import ( | ||||
| 	"bytes" | ||||
| 	"encoding/json" | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"github.com/gin-gonic/gin" | ||||
| 	"net/http" | ||||
| 	"one-api/common" | ||||
| 	"one-api/model" | ||||
| 	"strconv" | ||||
| 	"sync" | ||||
| 	"time" | ||||
| ) | ||||
|  | ||||
| func testChannel(channel *model.Channel, request ChatRequest) error { | ||||
| 	switch channel.Type { | ||||
| 	case common.ChannelTypeAzure: | ||||
| 		request.Model = "gpt-35-turbo" | ||||
| 	default: | ||||
| 		request.Model = "gpt-3.5-turbo" | ||||
| 	} | ||||
| 	requestURL := common.ChannelBaseURLs[channel.Type] | ||||
| 	if channel.Type == common.ChannelTypeAzure { | ||||
| 		requestURL = fmt.Sprintf("%s/openai/deployments/%s/chat/completions?api-version=2023-03-15-preview", channel.BaseURL, request.Model) | ||||
| 	} else { | ||||
| 		if channel.BaseURL != "" { | ||||
| 			requestURL = channel.BaseURL | ||||
| 		} | ||||
| 		requestURL += "/v1/chat/completions" | ||||
| 	} | ||||
|  | ||||
| 	jsonData, err := json.Marshal(request) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	req, err := http.NewRequest("POST", requestURL, bytes.NewBuffer(jsonData)) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	if channel.Type == common.ChannelTypeAzure { | ||||
| 		req.Header.Set("api-key", channel.Key) | ||||
| 	} else { | ||||
| 		req.Header.Set("Authorization", "Bearer "+channel.Key) | ||||
| 	} | ||||
| 	req.Header.Set("Content-Type", "application/json") | ||||
| 	client := &http.Client{} | ||||
| 	resp, err := client.Do(req) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	defer resp.Body.Close() | ||||
| 	var response TextResponse | ||||
| 	err = json.NewDecoder(resp.Body).Decode(&response) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	if response.Usage.CompletionTokens == 0 { | ||||
| 		return errors.New(fmt.Sprintf("type %s, code %v, message %s", response.Error.Type, response.Error.Code, response.Error.Message)) | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func buildTestRequest(c *gin.Context) *ChatRequest { | ||||
| 	model_ := c.Query("model") | ||||
| 	testRequest := &ChatRequest{ | ||||
| 		Model:     model_, | ||||
| 		MaxTokens: 1, | ||||
| 	} | ||||
| 	testMessage := Message{ | ||||
| 		Role:    "user", | ||||
| 		Content: "hi", | ||||
| 	} | ||||
| 	testRequest.Messages = append(testRequest.Messages, testMessage) | ||||
| 	return testRequest | ||||
| } | ||||
|  | ||||
| func TestChannel(c *gin.Context) { | ||||
| 	id, err := strconv.Atoi(c.Param("id")) | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	channel, err := model.GetChannelById(id, true) | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	testRequest := buildTestRequest(c) | ||||
| 	tik := time.Now() | ||||
| 	err = testChannel(channel, *testRequest) | ||||
| 	tok := time.Now() | ||||
| 	milliseconds := tok.Sub(tik).Milliseconds() | ||||
| 	go channel.UpdateResponseTime(milliseconds) | ||||
| 	consumedTime := float64(milliseconds) / 1000.0 | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| 			"time":    consumedTime, | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	c.JSON(http.StatusOK, gin.H{ | ||||
| 		"success": true, | ||||
| 		"message": "", | ||||
| 		"time":    consumedTime, | ||||
| 	}) | ||||
| 	return | ||||
| } | ||||
|  | ||||
| var testAllChannelsLock sync.Mutex | ||||
| var testAllChannelsRunning bool = false | ||||
|  | ||||
| // disable & notify | ||||
| func disableChannel(channelId int, channelName string, reason string) { | ||||
| 	if common.RootUserEmail == "" { | ||||
| 		common.RootUserEmail = model.GetRootUserEmail() | ||||
| 	} | ||||
| 	model.UpdateChannelStatusById(channelId, common.ChannelStatusDisabled) | ||||
| 	subject := fmt.Sprintf("通道「%s」(#%d)已被禁用", channelName, channelId) | ||||
| 	content := fmt.Sprintf("通道「%s」(#%d)已被禁用,原因:%s", channelName, channelId, reason) | ||||
| 	err := common.SendEmail(subject, common.RootUserEmail, content) | ||||
| 	if err != nil { | ||||
| 		common.SysError(fmt.Sprintf("发送邮件失败:%s", err.Error())) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func testAllChannels(c *gin.Context) error { | ||||
| 	if common.RootUserEmail == "" { | ||||
| 		common.RootUserEmail = model.GetRootUserEmail() | ||||
| 	} | ||||
| 	testAllChannelsLock.Lock() | ||||
| 	if testAllChannelsRunning { | ||||
| 		testAllChannelsLock.Unlock() | ||||
| 		return errors.New("测试已在运行中") | ||||
| 	} | ||||
| 	testAllChannelsRunning = true | ||||
| 	testAllChannelsLock.Unlock() | ||||
| 	channels, err := model.GetAllChannels(0, 0, true) | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| 		}) | ||||
| 		return err | ||||
| 	} | ||||
| 	testRequest := buildTestRequest(c) | ||||
| 	var disableThreshold = int64(common.ChannelDisableThreshold * 1000) | ||||
| 	if disableThreshold == 0 { | ||||
| 		disableThreshold = 10000000 // a impossible value | ||||
| 	} | ||||
| 	go func() { | ||||
| 		for _, channel := range channels { | ||||
| 			if channel.Status != common.ChannelStatusEnabled { | ||||
| 				continue | ||||
| 			} | ||||
| 			tik := time.Now() | ||||
| 			err := testChannel(channel, *testRequest) | ||||
| 			tok := time.Now() | ||||
| 			milliseconds := tok.Sub(tik).Milliseconds() | ||||
| 			if err != nil || milliseconds > disableThreshold { | ||||
| 				if milliseconds > disableThreshold { | ||||
| 					err = errors.New(fmt.Sprintf("响应时间 %.2fs 超过阈值 %.2fs", float64(milliseconds)/1000.0, float64(disableThreshold)/1000.0)) | ||||
| 				} | ||||
| 				disableChannel(channel.Id, channel.Name, err.Error()) | ||||
| 			} | ||||
| 			channel.UpdateResponseTime(milliseconds) | ||||
| 		} | ||||
| 		err := common.SendEmail("通道测试完成", common.RootUserEmail, "通道测试完成,如果没有收到禁用通知,说明所有通道都正常") | ||||
| 		if err != nil { | ||||
| 			common.SysError(fmt.Sprintf("发送邮件失败:%s", err.Error())) | ||||
| 		} | ||||
| 		testAllChannelsLock.Lock() | ||||
| 		testAllChannelsRunning = false | ||||
| 		testAllChannelsLock.Unlock() | ||||
| 	}() | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func TestAllChannels(c *gin.Context) { | ||||
| 	err := testAllChannels(c) | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	c.JSON(http.StatusOK, gin.H{ | ||||
| 		"success": true, | ||||
| 		"message": "", | ||||
| 	}) | ||||
| 	return | ||||
| } | ||||
| @@ -6,6 +6,7 @@ import ( | ||||
| 	"one-api/common" | ||||
| 	"one-api/model" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| func GetAllChannels(c *gin.Context) { | ||||
| @@ -13,7 +14,7 @@ func GetAllChannels(c *gin.Context) { | ||||
| 	if p < 0 { | ||||
| 		p = 0 | ||||
| 	} | ||||
| 	channels, err := model.GetAllChannels(p*common.ItemsPerPage, common.ItemsPerPage) | ||||
| 	channels, err := model.GetAllChannels(p*common.ItemsPerPage, common.ItemsPerPage, false) | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| @@ -83,8 +84,17 @@ func AddChannel(c *gin.Context) { | ||||
| 		return | ||||
| 	} | ||||
| 	channel.CreatedTime = common.GetTimestamp() | ||||
| 	channel.AccessedTime = common.GetTimestamp() | ||||
| 	err = channel.Insert() | ||||
| 	keys := strings.Split(channel.Key, "\n") | ||||
| 	channels := make([]model.Channel, 0) | ||||
| 	for _, key := range keys { | ||||
| 		if key == "" { | ||||
| 			continue | ||||
| 		} | ||||
| 		localChannel := channel | ||||
| 		localChannel.Key = key | ||||
| 		channels = append(channels, localChannel) | ||||
| 	} | ||||
| 	err = model.BatchInsertChannels(channels) | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
|   | ||||
| @@ -125,7 +125,7 @@ func GitHubOAuth(c *gin.Context) { | ||||
| 			user.Role = common.RoleCommonUser | ||||
| 			user.Status = common.UserStatusEnabled | ||||
|  | ||||
| 			if err := user.Insert(); err != nil { | ||||
| 			if err := user.Insert(0); err != nil { | ||||
| 				c.JSON(http.StatusOK, gin.H{ | ||||
| 					"success": false, | ||||
| 					"message": err.Error(), | ||||
|   | ||||
							
								
								
									
										19
									
								
								controller/group.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								controller/group.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | ||||
| package controller | ||||
|  | ||||
| import ( | ||||
| 	"github.com/gin-gonic/gin" | ||||
| 	"net/http" | ||||
| 	"one-api/common" | ||||
| ) | ||||
|  | ||||
| func GetGroups(c *gin.Context) { | ||||
| 	groupNames := make([]string, 0) | ||||
| 	for groupName, _ := range common.GroupRatio { | ||||
| 		groupNames = append(groupNames, groupName) | ||||
| 	} | ||||
| 	c.JSON(http.StatusOK, gin.H{ | ||||
| 		"success": true, | ||||
| 		"message": "", | ||||
| 		"data":    groupNames, | ||||
| 	}) | ||||
| } | ||||
							
								
								
									
										86
									
								
								controller/log.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										86
									
								
								controller/log.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,86 @@ | ||||
| package controller | ||||
|  | ||||
| import ( | ||||
| 	"github.com/gin-gonic/gin" | ||||
| 	"one-api/common" | ||||
| 	"one-api/model" | ||||
| 	"strconv" | ||||
| ) | ||||
|  | ||||
| func GetAllLogs(c *gin.Context) { | ||||
| 	p, _ := strconv.Atoi(c.Query("p")) | ||||
| 	if p < 0 { | ||||
| 		p = 0 | ||||
| 	} | ||||
| 	logType, _ := strconv.Atoi(c.Query("type")) | ||||
| 	logs, err := model.GetAllLogs(logType, p*common.ItemsPerPage, common.ItemsPerPage) | ||||
| 	if err != nil { | ||||
| 		c.JSON(200, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	c.JSON(200, gin.H{ | ||||
| 		"success": true, | ||||
| 		"message": "", | ||||
| 		"data":    logs, | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func GetUserLogs(c *gin.Context) { | ||||
| 	p, _ := strconv.Atoi(c.Query("p")) | ||||
| 	if p < 0 { | ||||
| 		p = 0 | ||||
| 	} | ||||
| 	userId := c.GetInt("id") | ||||
| 	logType, _ := strconv.Atoi(c.Query("type")) | ||||
| 	logs, err := model.GetUserLogs(userId, logType, p*common.ItemsPerPage, common.ItemsPerPage) | ||||
| 	if err != nil { | ||||
| 		c.JSON(200, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	c.JSON(200, gin.H{ | ||||
| 		"success": true, | ||||
| 		"message": "", | ||||
| 		"data":    logs, | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func SearchAllLogs(c *gin.Context) { | ||||
| 	keyword := c.Query("keyword") | ||||
| 	logs, err := model.SearchAllLogs(keyword) | ||||
| 	if err != nil { | ||||
| 		c.JSON(200, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	c.JSON(200, gin.H{ | ||||
| 		"success": true, | ||||
| 		"message": "", | ||||
| 		"data":    logs, | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func SearchUserLogs(c *gin.Context) { | ||||
| 	keyword := c.Query("keyword") | ||||
| 	userId := c.GetInt("id") | ||||
| 	logs, err := model.SearchUserLogs(userId, keyword) | ||||
| 	if err != nil { | ||||
| 		c.JSON(200, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	c.JSON(200, gin.H{ | ||||
| 		"success": true, | ||||
| 		"message": "", | ||||
| 		"data":    logs, | ||||
| 	}) | ||||
| } | ||||
| @@ -14,18 +14,23 @@ func GetStatus(c *gin.Context) { | ||||
| 		"success": true, | ||||
| 		"message": "", | ||||
| 		"data": gin.H{ | ||||
| 			"version":            common.Version, | ||||
| 			"start_time":         common.StartTime, | ||||
| 			"email_verification": common.EmailVerificationEnabled, | ||||
| 			"github_oauth":       common.GitHubOAuthEnabled, | ||||
| 			"github_client_id":   common.GitHubClientId, | ||||
| 			"system_name":        common.SystemName, | ||||
| 			"footer_html":        common.Footer, | ||||
| 			"wechat_qrcode":      common.WeChatAccountQRCodeImageURL, | ||||
| 			"wechat_login":       common.WeChatAuthEnabled, | ||||
| 			"server_address":     common.ServerAddress, | ||||
| 			"turnstile_check":    common.TurnstileCheckEnabled, | ||||
| 			"turnstile_site_key": common.TurnstileSiteKey, | ||||
| 			"version":             common.Version, | ||||
| 			"start_time":          common.StartTime, | ||||
| 			"email_verification":  common.EmailVerificationEnabled, | ||||
| 			"github_oauth":        common.GitHubOAuthEnabled, | ||||
| 			"github_client_id":    common.GitHubClientId, | ||||
| 			"system_name":         common.SystemName, | ||||
| 			"logo":                common.Logo, | ||||
| 			"footer_html":         common.Footer, | ||||
| 			"wechat_qrcode":       common.WeChatAccountQRCodeImageURL, | ||||
| 			"wechat_login":        common.WeChatAuthEnabled, | ||||
| 			"server_address":      common.ServerAddress, | ||||
| 			"turnstile_check":     common.TurnstileCheckEnabled, | ||||
| 			"turnstile_site_key":  common.TurnstileSiteKey, | ||||
| 			"top_up_link":         common.TopUpLink, | ||||
| 			"chat_link":           common.ChatLink, | ||||
| 			"quota_per_unit":      common.QuotaPerUnit, | ||||
| 			"display_in_currency": common.DisplayInCurrencyEnabled, | ||||
| 		}, | ||||
| 	}) | ||||
| 	return | ||||
| @@ -53,6 +58,17 @@ func GetAbout(c *gin.Context) { | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func GetHomePageContent(c *gin.Context) { | ||||
| 	common.OptionMapRWMutex.RLock() | ||||
| 	defer common.OptionMapRWMutex.RUnlock() | ||||
| 	c.JSON(http.StatusOK, gin.H{ | ||||
| 		"success": true, | ||||
| 		"message": "", | ||||
| 		"data":    common.OptionMap["HomePageContent"], | ||||
| 	}) | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func SendEmailVerification(c *gin.Context) { | ||||
| 	email := c.Query("email") | ||||
| 	if err := common.Validate.Var(email, "required,email"); err != nil { | ||||
|   | ||||
							
								
								
									
										256
									
								
								controller/model.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										256
									
								
								controller/model.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,256 @@ | ||||
| package controller | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"github.com/gin-gonic/gin" | ||||
| ) | ||||
|  | ||||
| // https://platform.openai.com/docs/api-reference/models/list | ||||
|  | ||||
| type OpenAIModelPermission struct { | ||||
| 	Id                 string  `json:"id"` | ||||
| 	Object             string  `json:"object"` | ||||
| 	Created            int     `json:"created"` | ||||
| 	AllowCreateEngine  bool    `json:"allow_create_engine"` | ||||
| 	AllowSampling      bool    `json:"allow_sampling"` | ||||
| 	AllowLogprobs      bool    `json:"allow_logprobs"` | ||||
| 	AllowSearchIndices bool    `json:"allow_search_indices"` | ||||
| 	AllowView          bool    `json:"allow_view"` | ||||
| 	AllowFineTuning    bool    `json:"allow_fine_tuning"` | ||||
| 	Organization       string  `json:"organization"` | ||||
| 	Group              *string `json:"group"` | ||||
| 	IsBlocking         bool    `json:"is_blocking"` | ||||
| } | ||||
|  | ||||
| type OpenAIModels struct { | ||||
| 	Id         string                  `json:"id"` | ||||
| 	Object     string                  `json:"object"` | ||||
| 	Created    int                     `json:"created"` | ||||
| 	OwnedBy    string                  `json:"owned_by"` | ||||
| 	Permission []OpenAIModelPermission `json:"permission"` | ||||
| 	Root       string                  `json:"root"` | ||||
| 	Parent     *string                 `json:"parent"` | ||||
| } | ||||
|  | ||||
| var openAIModels []OpenAIModels | ||||
| var openAIModelsMap map[string]OpenAIModels | ||||
|  | ||||
| func init() { | ||||
| 	var permission []OpenAIModelPermission | ||||
| 	permission = append(permission, OpenAIModelPermission{ | ||||
| 		Id:                 "modelperm-LwHkVFn8AcMItP432fKKDIKJ", | ||||
| 		Object:             "model_permission", | ||||
| 		Created:            1626777600, | ||||
| 		AllowCreateEngine:  true, | ||||
| 		AllowSampling:      true, | ||||
| 		AllowLogprobs:      true, | ||||
| 		AllowSearchIndices: false, | ||||
| 		AllowView:          true, | ||||
| 		AllowFineTuning:    false, | ||||
| 		Organization:       "*", | ||||
| 		Group:              nil, | ||||
| 		IsBlocking:         false, | ||||
| 	}) | ||||
| 	// https://platform.openai.com/docs/models/model-endpoint-compatibility | ||||
| 	openAIModels = []OpenAIModels{ | ||||
| 		{ | ||||
| 			Id:         "gpt-3.5-turbo", | ||||
| 			Object:     "model", | ||||
| 			Created:    1677649963, | ||||
| 			OwnedBy:    "openai", | ||||
| 			Permission: permission, | ||||
| 			Root:       "gpt-3.5-turbo", | ||||
| 			Parent:     nil, | ||||
| 		}, | ||||
| 		{ | ||||
| 			Id:         "gpt-3.5-turbo-0301", | ||||
| 			Object:     "model", | ||||
| 			Created:    1677649963, | ||||
| 			OwnedBy:    "openai", | ||||
| 			Permission: permission, | ||||
| 			Root:       "gpt-3.5-turbo-0301", | ||||
| 			Parent:     nil, | ||||
| 		}, | ||||
| 		{ | ||||
| 			Id:         "gpt-3.5-turbo-0613", | ||||
| 			Object:     "model", | ||||
| 			Created:    1677649963, | ||||
| 			OwnedBy:    "openai", | ||||
| 			Permission: permission, | ||||
| 			Root:       "gpt-3.5-turbo-0613", | ||||
| 			Parent:     nil, | ||||
| 		}, | ||||
| 		{ | ||||
| 			Id:         "gpt-3.5-turbo-16k", | ||||
| 			Object:     "model", | ||||
| 			Created:    1677649963, | ||||
| 			OwnedBy:    "openai", | ||||
| 			Permission: permission, | ||||
| 			Root:       "gpt-3.5-turbo-16k", | ||||
| 			Parent:     nil, | ||||
| 		}, | ||||
| 		{ | ||||
| 			Id:         "gpt-3.5-turbo-16k-0613", | ||||
| 			Object:     "model", | ||||
| 			Created:    1677649963, | ||||
| 			OwnedBy:    "openai", | ||||
| 			Permission: permission, | ||||
| 			Root:       "gpt-3.5-turbo-16k-0613", | ||||
| 			Parent:     nil, | ||||
| 		}, | ||||
| 		{ | ||||
| 			Id:         "gpt-4", | ||||
| 			Object:     "model", | ||||
| 			Created:    1677649963, | ||||
| 			OwnedBy:    "openai", | ||||
| 			Permission: permission, | ||||
| 			Root:       "gpt-4", | ||||
| 			Parent:     nil, | ||||
| 		}, | ||||
| 		{ | ||||
| 			Id:         "gpt-4-0314", | ||||
| 			Object:     "model", | ||||
| 			Created:    1677649963, | ||||
| 			OwnedBy:    "openai", | ||||
| 			Permission: permission, | ||||
| 			Root:       "gpt-4-0314", | ||||
| 			Parent:     nil, | ||||
| 		}, | ||||
| 		{ | ||||
| 			Id:         "gpt-4-0613", | ||||
| 			Object:     "model", | ||||
| 			Created:    1677649963, | ||||
| 			OwnedBy:    "openai", | ||||
| 			Permission: permission, | ||||
| 			Root:       "gpt-4-0613", | ||||
| 			Parent:     nil, | ||||
| 		}, | ||||
| 		{ | ||||
| 			Id:         "gpt-4-32k", | ||||
| 			Object:     "model", | ||||
| 			Created:    1677649963, | ||||
| 			OwnedBy:    "openai", | ||||
| 			Permission: permission, | ||||
| 			Root:       "gpt-4-32k", | ||||
| 			Parent:     nil, | ||||
| 		}, | ||||
| 		{ | ||||
| 			Id:         "gpt-4-32k-0314", | ||||
| 			Object:     "model", | ||||
| 			Created:    1677649963, | ||||
| 			OwnedBy:    "openai", | ||||
| 			Permission: permission, | ||||
| 			Root:       "gpt-4-32k-0314", | ||||
| 			Parent:     nil, | ||||
| 		}, | ||||
| 		{ | ||||
| 			Id:         "gpt-4-32k-0613", | ||||
| 			Object:     "model", | ||||
| 			Created:    1677649963, | ||||
| 			OwnedBy:    "openai", | ||||
| 			Permission: permission, | ||||
| 			Root:       "gpt-4-32k-0613", | ||||
| 			Parent:     nil, | ||||
| 		}, | ||||
| 		{ | ||||
| 			Id:         "text-embedding-ada-002", | ||||
| 			Object:     "model", | ||||
| 			Created:    1677649963, | ||||
| 			OwnedBy:    "openai", | ||||
| 			Permission: permission, | ||||
| 			Root:       "text-embedding-ada-002", | ||||
| 			Parent:     nil, | ||||
| 		}, | ||||
| 		{ | ||||
| 			Id:         "text-davinci-003", | ||||
| 			Object:     "model", | ||||
| 			Created:    1677649963, | ||||
| 			OwnedBy:    "openai", | ||||
| 			Permission: permission, | ||||
| 			Root:       "text-davinci-003", | ||||
| 			Parent:     nil, | ||||
| 		}, | ||||
| 		{ | ||||
| 			Id:         "text-davinci-002", | ||||
| 			Object:     "model", | ||||
| 			Created:    1677649963, | ||||
| 			OwnedBy:    "openai", | ||||
| 			Permission: permission, | ||||
| 			Root:       "text-davinci-002", | ||||
| 			Parent:     nil, | ||||
| 		}, | ||||
| 		{ | ||||
| 			Id:         "text-curie-001", | ||||
| 			Object:     "model", | ||||
| 			Created:    1677649963, | ||||
| 			OwnedBy:    "openai", | ||||
| 			Permission: permission, | ||||
| 			Root:       "text-curie-001", | ||||
| 			Parent:     nil, | ||||
| 		}, | ||||
| 		{ | ||||
| 			Id:         "text-babbage-001", | ||||
| 			Object:     "model", | ||||
| 			Created:    1677649963, | ||||
| 			OwnedBy:    "openai", | ||||
| 			Permission: permission, | ||||
| 			Root:       "text-babbage-001", | ||||
| 			Parent:     nil, | ||||
| 		}, | ||||
| 		{ | ||||
| 			Id:         "text-ada-001", | ||||
| 			Object:     "model", | ||||
| 			Created:    1677649963, | ||||
| 			OwnedBy:    "openai", | ||||
| 			Permission: permission, | ||||
| 			Root:       "text-ada-001", | ||||
| 			Parent:     nil, | ||||
| 		}, | ||||
| 		{ | ||||
| 			Id:         "text-moderation-latest", | ||||
| 			Object:     "model", | ||||
| 			Created:    1677649963, | ||||
| 			OwnedBy:    "openai", | ||||
| 			Permission: permission, | ||||
| 			Root:       "text-moderation-latest", | ||||
| 			Parent:     nil, | ||||
| 		}, | ||||
| 		{ | ||||
| 			Id:         "text-moderation-stable", | ||||
| 			Object:     "model", | ||||
| 			Created:    1677649963, | ||||
| 			OwnedBy:    "openai", | ||||
| 			Permission: permission, | ||||
| 			Root:       "text-moderation-stable", | ||||
| 			Parent:     nil, | ||||
| 		}, | ||||
| 	} | ||||
| 	openAIModelsMap = make(map[string]OpenAIModels) | ||||
| 	for _, model := range openAIModels { | ||||
| 		openAIModelsMap[model.Id] = model | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func ListModels(c *gin.Context) { | ||||
| 	c.JSON(200, gin.H{ | ||||
| 		"object": "list", | ||||
| 		"data":   openAIModels, | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func RetrieveModel(c *gin.Context) { | ||||
| 	modelId := c.Param("model") | ||||
| 	if model, ok := openAIModelsMap[modelId]; ok { | ||||
| 		c.JSON(200, model) | ||||
| 	} else { | ||||
| 		openAIError := OpenAIError{ | ||||
| 			Message: fmt.Sprintf("The model '%s' does not exist", modelId), | ||||
| 			Type:    "invalid_request_error", | ||||
| 			Param:   "model", | ||||
| 			Code:    "model_not_found", | ||||
| 		} | ||||
| 		c.JSON(200, gin.H{ | ||||
| 			"error": openAIError, | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										192
									
								
								controller/redemption.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										192
									
								
								controller/redemption.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,192 @@ | ||||
| package controller | ||||
|  | ||||
| import ( | ||||
| 	"github.com/gin-gonic/gin" | ||||
| 	"net/http" | ||||
| 	"one-api/common" | ||||
| 	"one-api/model" | ||||
| 	"strconv" | ||||
| ) | ||||
|  | ||||
| func GetAllRedemptions(c *gin.Context) { | ||||
| 	p, _ := strconv.Atoi(c.Query("p")) | ||||
| 	if p < 0 { | ||||
| 		p = 0 | ||||
| 	} | ||||
| 	redemptions, err := model.GetAllRedemptions(p*common.ItemsPerPage, common.ItemsPerPage) | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	c.JSON(http.StatusOK, gin.H{ | ||||
| 		"success": true, | ||||
| 		"message": "", | ||||
| 		"data":    redemptions, | ||||
| 	}) | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func SearchRedemptions(c *gin.Context) { | ||||
| 	keyword := c.Query("keyword") | ||||
| 	redemptions, err := model.SearchRedemptions(keyword) | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	c.JSON(http.StatusOK, gin.H{ | ||||
| 		"success": true, | ||||
| 		"message": "", | ||||
| 		"data":    redemptions, | ||||
| 	}) | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func GetRedemption(c *gin.Context) { | ||||
| 	id, err := strconv.Atoi(c.Param("id")) | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	redemption, err := model.GetRedemptionById(id) | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	c.JSON(http.StatusOK, gin.H{ | ||||
| 		"success": true, | ||||
| 		"message": "", | ||||
| 		"data":    redemption, | ||||
| 	}) | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func AddRedemption(c *gin.Context) { | ||||
| 	redemption := model.Redemption{} | ||||
| 	err := c.ShouldBindJSON(&redemption) | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	if len(redemption.Name) == 0 || len(redemption.Name) > 20 { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": "兑换码名称长度必须在1-20之间", | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	if redemption.Count <= 0 { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": "兑换码个数必须大于0", | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	if redemption.Count > 100 { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": "一次兑换码批量生成的个数不能大于 100", | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	var keys []string | ||||
| 	for i := 0; i < redemption.Count; i++ { | ||||
| 		key := common.GetUUID() | ||||
| 		cleanRedemption := model.Redemption{ | ||||
| 			UserId:      c.GetInt("id"), | ||||
| 			Name:        redemption.Name, | ||||
| 			Key:         key, | ||||
| 			CreatedTime: common.GetTimestamp(), | ||||
| 			Quota:       redemption.Quota, | ||||
| 		} | ||||
| 		err = cleanRedemption.Insert() | ||||
| 		if err != nil { | ||||
| 			c.JSON(http.StatusOK, gin.H{ | ||||
| 				"success": false, | ||||
| 				"message": err.Error(), | ||||
| 				"data":    keys, | ||||
| 			}) | ||||
| 			return | ||||
| 		} | ||||
| 		keys = append(keys, key) | ||||
| 	} | ||||
| 	c.JSON(http.StatusOK, gin.H{ | ||||
| 		"success": true, | ||||
| 		"message": "", | ||||
| 		"data":    keys, | ||||
| 	}) | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func DeleteRedemption(c *gin.Context) { | ||||
| 	id, _ := strconv.Atoi(c.Param("id")) | ||||
| 	err := model.DeleteRedemptionById(id) | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	c.JSON(http.StatusOK, gin.H{ | ||||
| 		"success": true, | ||||
| 		"message": "", | ||||
| 	}) | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func UpdateRedemption(c *gin.Context) { | ||||
| 	statusOnly := c.Query("status_only") | ||||
| 	redemption := model.Redemption{} | ||||
| 	err := c.ShouldBindJSON(&redemption) | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	cleanRedemption, err := model.GetRedemptionById(redemption.Id) | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	if statusOnly != "" { | ||||
| 		cleanRedemption.Status = redemption.Status | ||||
| 	} else { | ||||
| 		// If you add more fields, please also update redemption.Update() | ||||
| 		cleanRedemption.Name = redemption.Name | ||||
| 		cleanRedemption.Quota = redemption.Quota | ||||
| 	} | ||||
| 	err = cleanRedemption.Update() | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	c.JSON(http.StatusOK, gin.H{ | ||||
| 		"success": true, | ||||
| 		"message": "", | ||||
| 		"data":    cleanRedemption, | ||||
| 	}) | ||||
| 	return | ||||
| } | ||||
							
								
								
									
										34
									
								
								controller/relay-image.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								controller/relay-image.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,34 @@ | ||||
| package controller | ||||
|  | ||||
| import ( | ||||
| 	"github.com/gin-gonic/gin" | ||||
| 	"io" | ||||
| 	"net/http" | ||||
| ) | ||||
|  | ||||
| func relayImageHelper(c *gin.Context, relayMode int) *OpenAIErrorWithStatusCode { | ||||
| 	// TODO: this part is not finished | ||||
| 	req, err := http.NewRequest(c.Request.Method, c.Request.RequestURI, c.Request.Body) | ||||
| 	client := &http.Client{} | ||||
| 	resp, err := client.Do(req) | ||||
| 	if err != nil { | ||||
| 		return errorWrapper(err, "do_request_failed", http.StatusOK) | ||||
| 	} | ||||
| 	err = req.Body.Close() | ||||
| 	if err != nil { | ||||
| 		return errorWrapper(err, "close_request_body_failed", http.StatusOK) | ||||
| 	} | ||||
| 	for k, v := range resp.Header { | ||||
| 		c.Writer.Header().Set(k, v[0]) | ||||
| 	} | ||||
| 	c.Writer.WriteHeader(resp.StatusCode) | ||||
| 	_, err = io.Copy(c.Writer, resp.Body) | ||||
| 	if err != nil { | ||||
| 		return errorWrapper(err, "copy_response_body_failed", http.StatusOK) | ||||
| 	} | ||||
| 	err = resp.Body.Close() | ||||
| 	if err != nil { | ||||
| 		return errorWrapper(err, "close_response_body_failed", http.StatusOK) | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
							
								
								
									
										59
									
								
								controller/relay-palm.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										59
									
								
								controller/relay-palm.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,59 @@ | ||||
| package controller | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"github.com/gin-gonic/gin" | ||||
| ) | ||||
|  | ||||
| type PaLMChatMessage struct { | ||||
| 	Author  string `json:"author"` | ||||
| 	Content string `json:"content"` | ||||
| } | ||||
|  | ||||
| type PaLMFilter struct { | ||||
| 	Reason  string `json:"reason"` | ||||
| 	Message string `json:"message"` | ||||
| } | ||||
|  | ||||
| // https://developers.generativeai.google/api/rest/generativelanguage/models/generateMessage#request-body | ||||
| type PaLMChatRequest struct { | ||||
| 	Prompt         []Message `json:"prompt"` | ||||
| 	Temperature    float64   `json:"temperature"` | ||||
| 	CandidateCount int       `json:"candidateCount"` | ||||
| 	TopP           float64   `json:"topP"` | ||||
| 	TopK           int       `json:"topK"` | ||||
| } | ||||
|  | ||||
| // https://developers.generativeai.google/api/rest/generativelanguage/models/generateMessage#response-body | ||||
| type PaLMChatResponse struct { | ||||
| 	Candidates []Message    `json:"candidates"` | ||||
| 	Messages   []Message    `json:"messages"` | ||||
| 	Filters    []PaLMFilter `json:"filters"` | ||||
| } | ||||
|  | ||||
| func relayPaLM(openAIRequest GeneralOpenAIRequest, c *gin.Context) *OpenAIErrorWithStatusCode { | ||||
| 	// https://developers.generativeai.google/api/rest/generativelanguage/models/generateMessage | ||||
| 	messages := make([]PaLMChatMessage, 0, len(openAIRequest.Messages)) | ||||
| 	for _, message := range openAIRequest.Messages { | ||||
| 		var author string | ||||
| 		if message.Role == "user" { | ||||
| 			author = "0" | ||||
| 		} else { | ||||
| 			author = "1" | ||||
| 		} | ||||
| 		messages = append(messages, PaLMChatMessage{ | ||||
| 			Author:  author, | ||||
| 			Content: message.Content, | ||||
| 		}) | ||||
| 	} | ||||
| 	request := PaLMChatRequest{ | ||||
| 		Prompt:         nil, | ||||
| 		Temperature:    openAIRequest.Temperature, | ||||
| 		CandidateCount: openAIRequest.N, | ||||
| 		TopP:           openAIRequest.TopP, | ||||
| 		TopK:           openAIRequest.MaxTokens, | ||||
| 	} | ||||
| 	// TODO: forward request to PaLM & convert response | ||||
| 	fmt.Print(request) | ||||
| 	return nil | ||||
| } | ||||
							
								
								
									
										262
									
								
								controller/relay-text.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										262
									
								
								controller/relay-text.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,262 @@ | ||||
| package controller | ||||
|  | ||||
| import ( | ||||
| 	"bufio" | ||||
| 	"bytes" | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"github.com/gin-gonic/gin" | ||||
| 	"io" | ||||
| 	"net/http" | ||||
| 	"one-api/common" | ||||
| 	"one-api/model" | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| func relayTextHelper(c *gin.Context, relayMode int) *OpenAIErrorWithStatusCode { | ||||
| 	channelType := c.GetInt("channel") | ||||
| 	tokenId := c.GetInt("token_id") | ||||
| 	consumeQuota := c.GetBool("consume_quota") | ||||
| 	group := c.GetString("group") | ||||
| 	var textRequest GeneralOpenAIRequest | ||||
| 	if consumeQuota || channelType == common.ChannelTypeAzure || channelType == common.ChannelTypePaLM { | ||||
| 		err := common.UnmarshalBodyReusable(c, &textRequest) | ||||
| 		if err != nil { | ||||
| 			return errorWrapper(err, "bind_request_body_failed", http.StatusBadRequest) | ||||
| 		} | ||||
| 	} | ||||
| 	if relayMode == RelayModeModeration && textRequest.Model == "" { | ||||
| 		textRequest.Model = "text-moderation-latest" | ||||
| 	} | ||||
| 	baseURL := common.ChannelBaseURLs[channelType] | ||||
| 	requestURL := c.Request.URL.String() | ||||
| 	if c.GetString("base_url") != "" { | ||||
| 		baseURL = c.GetString("base_url") | ||||
| 	} | ||||
| 	fullRequestURL := fmt.Sprintf("%s%s", baseURL, requestURL) | ||||
| 	if channelType == common.ChannelTypeAzure { | ||||
| 		// https://learn.microsoft.com/en-us/azure/cognitive-services/openai/chatgpt-quickstart?pivots=rest-api&tabs=command-line#rest-api | ||||
| 		query := c.Request.URL.Query() | ||||
| 		apiVersion := query.Get("api-version") | ||||
| 		if apiVersion == "" { | ||||
| 			apiVersion = c.GetString("api_version") | ||||
| 		} | ||||
| 		requestURL := strings.Split(requestURL, "?")[0] | ||||
| 		requestURL = fmt.Sprintf("%s?api-version=%s", requestURL, apiVersion) | ||||
| 		baseURL = c.GetString("base_url") | ||||
| 		task := strings.TrimPrefix(requestURL, "/v1/") | ||||
| 		model_ := textRequest.Model | ||||
| 		model_ = strings.Replace(model_, ".", "", -1) | ||||
| 		// https://github.com/songquanpeng/one-api/issues/67 | ||||
| 		model_ = strings.TrimSuffix(model_, "-0301") | ||||
| 		model_ = strings.TrimSuffix(model_, "-0314") | ||||
| 		model_ = strings.TrimSuffix(model_, "-0613") | ||||
| 		fullRequestURL = fmt.Sprintf("%s/openai/deployments/%s/%s", baseURL, model_, task) | ||||
| 	} else if channelType == common.ChannelTypePaLM { | ||||
| 		err := relayPaLM(textRequest, c) | ||||
| 		return err | ||||
| 	} | ||||
| 	var promptTokens int | ||||
| 	switch relayMode { | ||||
| 	case RelayModeChatCompletions: | ||||
| 		promptTokens = countTokenMessages(textRequest.Messages, textRequest.Model) | ||||
| 	case RelayModeCompletions: | ||||
| 		promptTokens = countTokenInput(textRequest.Prompt, textRequest.Model) | ||||
| 	case RelayModeModeration: | ||||
| 		promptTokens = countTokenInput(textRequest.Input, textRequest.Model) | ||||
| 	} | ||||
| 	preConsumedTokens := common.PreConsumedQuota | ||||
| 	if textRequest.MaxTokens != 0 { | ||||
| 		preConsumedTokens = promptTokens + textRequest.MaxTokens | ||||
| 	} | ||||
| 	modelRatio := common.GetModelRatio(textRequest.Model) | ||||
| 	groupRatio := common.GetGroupRatio(group) | ||||
| 	ratio := modelRatio * groupRatio | ||||
| 	preConsumedQuota := int(float64(preConsumedTokens) * ratio) | ||||
| 	if consumeQuota { | ||||
| 		err := model.PreConsumeTokenQuota(tokenId, preConsumedQuota) | ||||
| 		if err != nil { | ||||
| 			return errorWrapper(err, "pre_consume_token_quota_failed", http.StatusOK) | ||||
| 		} | ||||
| 	} | ||||
| 	req, err := http.NewRequest(c.Request.Method, fullRequestURL, c.Request.Body) | ||||
| 	if err != nil { | ||||
| 		return errorWrapper(err, "new_request_failed", http.StatusOK) | ||||
| 	} | ||||
| 	if channelType == common.ChannelTypeAzure { | ||||
| 		key := c.Request.Header.Get("Authorization") | ||||
| 		key = strings.TrimPrefix(key, "Bearer ") | ||||
| 		req.Header.Set("api-key", key) | ||||
| 	} else { | ||||
| 		req.Header.Set("Authorization", c.Request.Header.Get("Authorization")) | ||||
| 	} | ||||
| 	req.Header.Set("Content-Type", c.Request.Header.Get("Content-Type")) | ||||
| 	req.Header.Set("Accept", c.Request.Header.Get("Accept")) | ||||
| 	req.Header.Set("Connection", c.Request.Header.Get("Connection")) | ||||
| 	client := &http.Client{} | ||||
| 	resp, err := client.Do(req) | ||||
| 	if err != nil { | ||||
| 		return errorWrapper(err, "do_request_failed", http.StatusOK) | ||||
| 	} | ||||
| 	err = req.Body.Close() | ||||
| 	if err != nil { | ||||
| 		return errorWrapper(err, "close_request_body_failed", http.StatusOK) | ||||
| 	} | ||||
| 	err = c.Request.Body.Close() | ||||
| 	if err != nil { | ||||
| 		return errorWrapper(err, "close_request_body_failed", http.StatusOK) | ||||
| 	} | ||||
| 	var textResponse TextResponse | ||||
| 	isStream := strings.HasPrefix(resp.Header.Get("Content-Type"), "text/event-stream") | ||||
| 	var streamResponseText string | ||||
|  | ||||
| 	defer func() { | ||||
| 		if consumeQuota { | ||||
| 			quota := 0 | ||||
| 			completionRatio := 1.34 // default for gpt-3 | ||||
| 			if strings.HasPrefix(textRequest.Model, "gpt-4") { | ||||
| 				completionRatio = 2 | ||||
| 			} | ||||
| 			if isStream { | ||||
| 				responseTokens := countTokenText(streamResponseText, textRequest.Model) | ||||
| 				quota = promptTokens + int(float64(responseTokens)*completionRatio) | ||||
| 			} else { | ||||
| 				quota = textResponse.Usage.PromptTokens + int(float64(textResponse.Usage.CompletionTokens)*completionRatio) | ||||
| 			} | ||||
| 			quota = int(float64(quota) * ratio) | ||||
| 			if ratio != 0 && quota <= 0 { | ||||
| 				quota = 1 | ||||
| 			} | ||||
| 			quotaDelta := quota - preConsumedQuota | ||||
| 			err := model.PostConsumeTokenQuota(tokenId, quotaDelta) | ||||
| 			if err != nil { | ||||
| 				common.SysError("Error consuming token remain quota: " + err.Error()) | ||||
| 			} | ||||
| 			tokenName := c.GetString("token_name") | ||||
| 			userId := c.GetInt("id") | ||||
| 			model.RecordLog(userId, model.LogTypeConsume, fmt.Sprintf("通过令牌「%s」使用模型 %s 消耗 %s(模型倍率 %.2f,分组倍率 %.2f)", tokenName, textRequest.Model, common.LogQuota(quota), modelRatio, groupRatio)) | ||||
| 			model.UpdateUserUsedQuotaAndRequestCount(userId, quota) | ||||
| 			channelId := c.GetInt("channel_id") | ||||
| 			model.UpdateChannelUsedQuota(channelId, quota) | ||||
| 		} | ||||
| 	}() | ||||
|  | ||||
| 	if isStream { | ||||
| 		scanner := bufio.NewScanner(resp.Body) | ||||
| 		scanner.Split(func(data []byte, atEOF bool) (advance int, token []byte, err error) { | ||||
| 			if atEOF && len(data) == 0 { | ||||
| 				return 0, nil, nil | ||||
| 			} | ||||
|  | ||||
| 			if i := strings.Index(string(data), "\n\n"); i >= 0 { | ||||
| 				return i + 2, data[0:i], nil | ||||
| 			} | ||||
|  | ||||
| 			if atEOF { | ||||
| 				return len(data), data, nil | ||||
| 			} | ||||
|  | ||||
| 			return 0, nil, nil | ||||
| 		}) | ||||
| 		dataChan := make(chan string) | ||||
| 		stopChan := make(chan bool) | ||||
| 		go func() { | ||||
| 			for scanner.Scan() { | ||||
| 				data := scanner.Text() | ||||
| 				if len(data) < 6 { // must be something wrong! | ||||
| 					common.SysError("Invalid stream response: " + data) | ||||
| 					continue | ||||
| 				} | ||||
| 				dataChan <- data | ||||
| 				data = data[6:] | ||||
| 				if !strings.HasPrefix(data, "[DONE]") { | ||||
| 					switch relayMode { | ||||
| 					case RelayModeChatCompletions: | ||||
| 						var streamResponse ChatCompletionsStreamResponse | ||||
| 						err = json.Unmarshal([]byte(data), &streamResponse) | ||||
| 						if err != nil { | ||||
| 							common.SysError("Error unmarshalling stream response: " + err.Error()) | ||||
| 							return | ||||
| 						} | ||||
| 						for _, choice := range streamResponse.Choices { | ||||
| 							streamResponseText += choice.Delta.Content | ||||
| 						} | ||||
| 					case RelayModeCompletions: | ||||
| 						var streamResponse CompletionsStreamResponse | ||||
| 						err = json.Unmarshal([]byte(data), &streamResponse) | ||||
| 						if err != nil { | ||||
| 							common.SysError("Error unmarshalling stream response: " + err.Error()) | ||||
| 							return | ||||
| 						} | ||||
| 						for _, choice := range streamResponse.Choices { | ||||
| 							streamResponseText += choice.Text | ||||
| 						} | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 			stopChan <- true | ||||
| 		}() | ||||
| 		c.Writer.Header().Set("Content-Type", "text/event-stream") | ||||
| 		c.Writer.Header().Set("Cache-Control", "no-cache") | ||||
| 		c.Writer.Header().Set("Connection", "keep-alive") | ||||
| 		c.Writer.Header().Set("Transfer-Encoding", "chunked") | ||||
| 		c.Writer.Header().Set("X-Accel-Buffering", "no") | ||||
| 		c.Stream(func(w io.Writer) bool { | ||||
| 			select { | ||||
| 			case data := <-dataChan: | ||||
| 				if strings.HasPrefix(data, "data: [DONE]") { | ||||
| 					data = data[:12] | ||||
| 				} | ||||
| 				c.Render(-1, common.CustomEvent{Data: data}) | ||||
| 				return true | ||||
| 			case <-stopChan: | ||||
| 				return false | ||||
| 			} | ||||
| 		}) | ||||
| 		err = resp.Body.Close() | ||||
| 		if err != nil { | ||||
| 			return errorWrapper(err, "close_response_body_failed", http.StatusOK) | ||||
| 		} | ||||
| 		return nil | ||||
| 	} else { | ||||
| 		if consumeQuota { | ||||
| 			responseBody, err := io.ReadAll(resp.Body) | ||||
| 			if err != nil { | ||||
| 				return errorWrapper(err, "read_response_body_failed", http.StatusOK) | ||||
| 			} | ||||
| 			err = resp.Body.Close() | ||||
| 			if err != nil { | ||||
| 				return errorWrapper(err, "close_response_body_failed", http.StatusOK) | ||||
| 			} | ||||
| 			err = json.Unmarshal(responseBody, &textResponse) | ||||
| 			if err != nil { | ||||
| 				return errorWrapper(err, "unmarshal_response_body_failed", http.StatusOK) | ||||
| 			} | ||||
| 			if textResponse.Error.Type != "" { | ||||
| 				return &OpenAIErrorWithStatusCode{ | ||||
| 					OpenAIError: textResponse.Error, | ||||
| 					StatusCode:  resp.StatusCode, | ||||
| 				} | ||||
| 			} | ||||
| 			// Reset response body | ||||
| 			resp.Body = io.NopCloser(bytes.NewBuffer(responseBody)) | ||||
| 		} | ||||
| 		// We shouldn't set the header before we parse the response body, because the parse part may fail. | ||||
| 		// And then we will have to send an error response, but in this case, the header has already been set. | ||||
| 		// So the client will be confused by the response. | ||||
| 		// For example, Postman will report error, and we cannot check the response at all. | ||||
| 		for k, v := range resp.Header { | ||||
| 			c.Writer.Header().Set(k, v[0]) | ||||
| 		} | ||||
| 		c.Writer.WriteHeader(resp.StatusCode) | ||||
| 		_, err = io.Copy(c.Writer, resp.Body) | ||||
| 		if err != nil { | ||||
| 			return errorWrapper(err, "copy_response_body_failed", http.StatusOK) | ||||
| 		} | ||||
| 		err = resp.Body.Close() | ||||
| 		if err != nil { | ||||
| 			return errorWrapper(err, "close_response_body_failed", http.StatusOK) | ||||
| 		} | ||||
| 		return nil | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										91
									
								
								controller/relay-utils.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										91
									
								
								controller/relay-utils.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,91 @@ | ||||
| package controller | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"github.com/pkoukk/tiktoken-go" | ||||
| 	"one-api/common" | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| var tokenEncoderMap = map[string]*tiktoken.Tiktoken{} | ||||
|  | ||||
| func getTokenEncoder(model string) *tiktoken.Tiktoken { | ||||
| 	if tokenEncoder, ok := tokenEncoderMap[model]; ok { | ||||
| 		return tokenEncoder | ||||
| 	} | ||||
| 	tokenEncoder, err := tiktoken.EncodingForModel(model) | ||||
| 	if err != nil { | ||||
| 		common.SysError(fmt.Sprintf("failed to get token encoder for model %s: %s, using encoder for gpt-3.5-turbo", model, err.Error())) | ||||
| 		tokenEncoder, err = tiktoken.EncodingForModel("gpt-3.5-turbo") | ||||
| 		if err != nil { | ||||
| 			common.FatalLog(fmt.Sprintf("failed to get token encoder for model gpt-3.5-turbo: %s", err.Error())) | ||||
| 		} | ||||
| 	} | ||||
| 	tokenEncoderMap[model] = tokenEncoder | ||||
| 	return tokenEncoder | ||||
| } | ||||
|  | ||||
| func countTokenMessages(messages []Message, model string) int { | ||||
| 	tokenEncoder := getTokenEncoder(model) | ||||
| 	// Reference: | ||||
| 	// https://github.com/openai/openai-cookbook/blob/main/examples/How_to_count_tokens_with_tiktoken.ipynb | ||||
| 	// https://github.com/pkoukk/tiktoken-go/issues/6 | ||||
| 	// | ||||
| 	// Every message follows <|start|>{role/name}\n{content}<|end|>\n | ||||
| 	var tokensPerMessage int | ||||
| 	var tokensPerName int | ||||
| 	if strings.HasPrefix(model, "gpt-3.5") { | ||||
| 		tokensPerMessage = 4 | ||||
| 		tokensPerName = -1 // If there's a name, the role is omitted | ||||
| 	} else if strings.HasPrefix(model, "gpt-4") { | ||||
| 		tokensPerMessage = 3 | ||||
| 		tokensPerName = 1 | ||||
| 	} else { | ||||
| 		tokensPerMessage = 3 | ||||
| 		tokensPerName = 1 | ||||
| 	} | ||||
| 	tokenNum := 0 | ||||
| 	for _, message := range messages { | ||||
| 		tokenNum += tokensPerMessage | ||||
| 		tokenNum += len(tokenEncoder.Encode(message.Content, nil, nil)) | ||||
| 		tokenNum += len(tokenEncoder.Encode(message.Role, nil, nil)) | ||||
| 		if message.Name != nil { | ||||
| 			tokenNum += tokensPerName | ||||
| 			tokenNum += len(tokenEncoder.Encode(*message.Name, nil, nil)) | ||||
| 		} | ||||
| 	} | ||||
| 	tokenNum += 3 // Every reply is primed with <|start|>assistant<|message|> | ||||
| 	return tokenNum | ||||
| } | ||||
|  | ||||
| func countTokenInput(input any, model string) int { | ||||
| 	switch input.(type) { | ||||
| 	case string: | ||||
| 		return countTokenText(input.(string), model) | ||||
| 	case []string: | ||||
| 		text := "" | ||||
| 		for _, s := range input.([]string) { | ||||
| 			text += s | ||||
| 		} | ||||
| 		return countTokenText(text, model) | ||||
| 	} | ||||
| 	return 0 | ||||
| } | ||||
|  | ||||
| func countTokenText(text string, model string) int { | ||||
| 	tokenEncoder := getTokenEncoder(model) | ||||
| 	token := tokenEncoder.Encode(text, nil, nil) | ||||
| 	return len(token) | ||||
| } | ||||
|  | ||||
| func errorWrapper(err error, code string, statusCode int) *OpenAIErrorWithStatusCode { | ||||
| 	openAIError := OpenAIError{ | ||||
| 		Message: err.Error(), | ||||
| 		Type:    "one_api_error", | ||||
| 		Code:    code, | ||||
| 	} | ||||
| 	return &OpenAIErrorWithStatusCode{ | ||||
| 		OpenAIError: openAIError, | ||||
| 		StatusCode:  statusCode, | ||||
| 	} | ||||
| } | ||||
| @@ -3,56 +3,151 @@ package controller | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"github.com/gin-gonic/gin" | ||||
| 	"io" | ||||
| 	"net/http" | ||||
| 	"one-api/common" | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| func Relay(c *gin.Context) { | ||||
| 	channelType := c.GetInt("channel") | ||||
| 	baseURL := common.ChannelBaseURLs[channelType] | ||||
| 	if channelType == common.ChannelTypeCustom { | ||||
| 		baseURL = c.GetString("base_url") | ||||
| 	} | ||||
| 	req, err := http.NewRequest(c.Request.Method, fmt.Sprintf("%s%s", baseURL, c.Request.URL.String()), c.Request.Body) | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"error": gin.H{ | ||||
| 				"message": err.Error(), | ||||
| 				"type":    "one_api_error", | ||||
| 			}, | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	//req.Header = c.Request.Header.Clone() | ||||
| 	// Fix HTTP Decompression failed | ||||
| 	// https://github.com/stoplightio/prism/issues/1064#issuecomment-824682360 | ||||
| 	//req.Header.Del("Accept-Encoding") | ||||
| 	req.Header.Set("Authorization", c.Request.Header.Get("Authorization")) | ||||
| 	req.Header.Set("Content-Type", c.Request.Header.Get("Content-Type")) | ||||
| 	client := &http.Client{} | ||||
| type Message struct { | ||||
| 	Role    string  `json:"role"` | ||||
| 	Content string  `json:"content"` | ||||
| 	Name    *string `json:"name,omitempty"` | ||||
| } | ||||
|  | ||||
| 	resp, err := client.Do(req) | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"error": gin.H{ | ||||
| 				"message": err.Error(), | ||||
| 				"type":    "one_api_error", | ||||
| 			}, | ||||
| 		}) | ||||
| 		return | ||||
| const ( | ||||
| 	RelayModeUnknown = iota | ||||
| 	RelayModeChatCompletions | ||||
| 	RelayModeCompletions | ||||
| 	RelayModeEmbeddings | ||||
| 	RelayModeModeration | ||||
| 	RelayModeImagesGenerations | ||||
| ) | ||||
|  | ||||
| // https://platform.openai.com/docs/api-reference/chat | ||||
|  | ||||
| type GeneralOpenAIRequest struct { | ||||
| 	Model       string    `json:"model"` | ||||
| 	Messages    []Message `json:"messages"` | ||||
| 	Prompt      any       `json:"prompt"` | ||||
| 	Stream      bool      `json:"stream"` | ||||
| 	MaxTokens   int       `json:"max_tokens"` | ||||
| 	Temperature float64   `json:"temperature"` | ||||
| 	TopP        float64   `json:"top_p"` | ||||
| 	N           int       `json:"n"` | ||||
| 	Input       any       `json:"input"` | ||||
| } | ||||
|  | ||||
| type ChatRequest struct { | ||||
| 	Model     string    `json:"model"` | ||||
| 	Messages  []Message `json:"messages"` | ||||
| 	MaxTokens int       `json:"max_tokens"` | ||||
| } | ||||
|  | ||||
| type TextRequest struct { | ||||
| 	Model     string    `json:"model"` | ||||
| 	Messages  []Message `json:"messages"` | ||||
| 	Prompt    string    `json:"prompt"` | ||||
| 	MaxTokens int       `json:"max_tokens"` | ||||
| 	//Stream   bool      `json:"stream"` | ||||
| } | ||||
|  | ||||
| type Usage struct { | ||||
| 	PromptTokens     int `json:"prompt_tokens"` | ||||
| 	CompletionTokens int `json:"completion_tokens"` | ||||
| 	TotalTokens      int `json:"total_tokens"` | ||||
| } | ||||
|  | ||||
| type OpenAIError struct { | ||||
| 	Message string `json:"message"` | ||||
| 	Type    string `json:"type"` | ||||
| 	Param   string `json:"param"` | ||||
| 	Code    any    `json:"code"` | ||||
| } | ||||
|  | ||||
| type OpenAIErrorWithStatusCode struct { | ||||
| 	OpenAIError | ||||
| 	StatusCode int `json:"status_code"` | ||||
| } | ||||
|  | ||||
| type TextResponse struct { | ||||
| 	Usage `json:"usage"` | ||||
| 	Error OpenAIError `json:"error"` | ||||
| } | ||||
|  | ||||
| type ChatCompletionsStreamResponse struct { | ||||
| 	Choices []struct { | ||||
| 		Delta struct { | ||||
| 			Content string `json:"content"` | ||||
| 		} `json:"delta"` | ||||
| 		FinishReason string `json:"finish_reason"` | ||||
| 	} `json:"choices"` | ||||
| } | ||||
|  | ||||
| type CompletionsStreamResponse struct { | ||||
| 	Choices []struct { | ||||
| 		Text         string `json:"text"` | ||||
| 		FinishReason string `json:"finish_reason"` | ||||
| 	} `json:"choices"` | ||||
| } | ||||
|  | ||||
| func Relay(c *gin.Context) { | ||||
| 	relayMode := RelayModeUnknown | ||||
| 	if strings.HasPrefix(c.Request.URL.Path, "/v1/chat/completions") { | ||||
| 		relayMode = RelayModeChatCompletions | ||||
| 	} else if strings.HasPrefix(c.Request.URL.Path, "/v1/completions") { | ||||
| 		relayMode = RelayModeCompletions | ||||
| 	} else if strings.HasPrefix(c.Request.URL.Path, "/v1/embeddings") { | ||||
| 		relayMode = RelayModeEmbeddings | ||||
| 	} else if strings.HasPrefix(c.Request.URL.Path, "/v1/moderations") { | ||||
| 		relayMode = RelayModeModeration | ||||
| 	} else if strings.HasPrefix(c.Request.URL.Path, "/v1/images/generations") { | ||||
| 		relayMode = RelayModeImagesGenerations | ||||
| 	} | ||||
| 	for k, v := range resp.Header { | ||||
| 		c.Writer.Header().Set(k, v[0]) | ||||
| 	var err *OpenAIErrorWithStatusCode | ||||
| 	switch relayMode { | ||||
| 	case RelayModeImagesGenerations: | ||||
| 		err = relayImageHelper(c, relayMode) | ||||
| 	default: | ||||
| 		err = relayTextHelper(c, relayMode) | ||||
| 	} | ||||
| 	_, err = io.Copy(c.Writer, resp.Body) | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"error": gin.H{ | ||||
| 				"message": err.Error(), | ||||
| 				"type":    "one_api_error", | ||||
| 			}, | ||||
| 		if err.StatusCode == http.StatusTooManyRequests { | ||||
| 			err.OpenAIError.Message = "当前分组负载已饱和,请稍后再试,或升级账户以提升服务质量。" | ||||
| 		} | ||||
| 		c.JSON(err.StatusCode, gin.H{ | ||||
| 			"error": err.OpenAIError, | ||||
| 		}) | ||||
| 		return | ||||
| 		channelId := c.GetInt("channel_id") | ||||
| 		common.SysError(fmt.Sprintf("Relay error (channel #%d): %s", channelId, err.Message)) | ||||
| 		// https://platform.openai.com/docs/guides/error-codes/api-errors | ||||
| 		if common.AutomaticDisableChannelEnabled && (err.Type == "insufficient_quota" || err.Code == "invalid_api_key") { | ||||
| 			channelId := c.GetInt("channel_id") | ||||
| 			channelName := c.GetString("channel_name") | ||||
| 			disableChannel(channelId, channelName, err.Message) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func RelayNotImplemented(c *gin.Context) { | ||||
| 	err := OpenAIError{ | ||||
| 		Message: "API not implemented", | ||||
| 		Type:    "one_api_error", | ||||
| 		Param:   "", | ||||
| 		Code:    "api_not_implemented", | ||||
| 	} | ||||
| 	c.JSON(http.StatusOK, gin.H{ | ||||
| 		"error": err, | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func RelayNotFound(c *gin.Context) { | ||||
| 	err := OpenAIError{ | ||||
| 		Message: fmt.Sprintf("API not found: %s:%s", c.Request.Method, c.Request.URL.Path), | ||||
| 		Type:    "one_api_error", | ||||
| 		Param:   "", | ||||
| 		Code:    "api_not_found", | ||||
| 	} | ||||
| 	c.JSON(http.StatusOK, gin.H{ | ||||
| 		"error": err, | ||||
| 	}) | ||||
| } | ||||
|   | ||||
| @@ -75,6 +75,30 @@ func GetToken(c *gin.Context) { | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func GetTokenStatus(c *gin.Context) { | ||||
| 	tokenId := c.GetInt("token_id") | ||||
| 	userId := c.GetInt("id") | ||||
| 	token, err := model.GetTokenByIds(tokenId, userId) | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	expiredAt := token.ExpiredTime | ||||
| 	if expiredAt == -1 { | ||||
| 		expiredAt = 0 | ||||
| 	} | ||||
| 	c.JSON(http.StatusOK, gin.H{ | ||||
| 		"object":          "credit_summary", | ||||
| 		"total_granted":   token.RemainQuota, | ||||
| 		"total_used":      0, // not supported currently | ||||
| 		"total_available": token.RemainQuota, | ||||
| 		"expires_at":      expiredAt * 1000, | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func AddToken(c *gin.Context) { | ||||
| 	token := model.Token{} | ||||
| 	err := c.ShouldBindJSON(&token) | ||||
| @@ -93,13 +117,14 @@ func AddToken(c *gin.Context) { | ||||
| 		return | ||||
| 	} | ||||
| 	cleanToken := model.Token{ | ||||
| 		UserId:       c.GetInt("id"), | ||||
| 		Name:         token.Name, | ||||
| 		Key:          common.GetUUID(), | ||||
| 		CreatedTime:  common.GetTimestamp(), | ||||
| 		AccessedTime: common.GetTimestamp(), | ||||
| 		ExpiredTime:  token.ExpiredTime, | ||||
| 		RemainTimes:  token.RemainTimes, | ||||
| 		UserId:         c.GetInt("id"), | ||||
| 		Name:           token.Name, | ||||
| 		Key:            common.GenerateKey(), | ||||
| 		CreatedTime:    common.GetTimestamp(), | ||||
| 		AccessedTime:   common.GetTimestamp(), | ||||
| 		ExpiredTime:    token.ExpiredTime, | ||||
| 		RemainQuota:    token.RemainQuota, | ||||
| 		UnlimitedQuota: token.UnlimitedQuota, | ||||
| 	} | ||||
| 	err = cleanToken.Insert() | ||||
| 	if err != nil { | ||||
| @@ -136,6 +161,7 @@ func DeleteToken(c *gin.Context) { | ||||
|  | ||||
| func UpdateToken(c *gin.Context) { | ||||
| 	userId := c.GetInt("id") | ||||
| 	statusOnly := c.Query("status_only") | ||||
| 	token := model.Token{} | ||||
| 	err := c.ShouldBindJSON(&token) | ||||
| 	if err != nil { | ||||
| @@ -161,19 +187,23 @@ func UpdateToken(c *gin.Context) { | ||||
| 			}) | ||||
| 			return | ||||
| 		} | ||||
| 		if cleanToken.Status == common.TokenStatusExhausted && cleanToken.RemainTimes == 0 { | ||||
| 		if cleanToken.Status == common.TokenStatusExhausted && cleanToken.RemainQuota <= 0 && !cleanToken.UnlimitedQuota { | ||||
| 			c.JSON(http.StatusOK, gin.H{ | ||||
| 				"success": false, | ||||
| 				"message": "令牌可用次数已用尽,无法启用,请先修改令牌剩余次数", | ||||
| 				"message": "令牌可用额度已用尽,无法启用,请先修改令牌剩余额度,或者设置为无限额度", | ||||
| 			}) | ||||
| 			return | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	cleanToken.Name = token.Name | ||||
| 	cleanToken.Status = token.Status | ||||
| 	cleanToken.ExpiredTime = token.ExpiredTime | ||||
| 	cleanToken.RemainTimes = token.RemainTimes | ||||
| 	if statusOnly != "" { | ||||
| 		cleanToken.Status = token.Status | ||||
| 	} else { | ||||
| 		// If you add more fields, please also update token.Update() | ||||
| 		cleanToken.Name = token.Name | ||||
| 		cleanToken.ExpiredTime = token.ExpiredTime | ||||
| 		cleanToken.RemainQuota = token.RemainQuota | ||||
| 		cleanToken.UnlimitedQuota = token.UnlimitedQuota | ||||
| 	} | ||||
| 	err = cleanToken.Update() | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
|   | ||||
| @@ -2,6 +2,7 @@ package controller | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"github.com/gin-contrib/sessions" | ||||
| 	"github.com/gin-gonic/gin" | ||||
| 	"net/http" | ||||
| @@ -149,15 +150,18 @@ func Register(c *gin.Context) { | ||||
| 			return | ||||
| 		} | ||||
| 	} | ||||
| 	affCode := user.AffCode // this code is the inviter's code, not the user's own code | ||||
| 	inviterId, _ := model.GetUserIdByAffCode(affCode) | ||||
| 	cleanUser := model.User{ | ||||
| 		Username:    user.Username, | ||||
| 		Password:    user.Password, | ||||
| 		DisplayName: user.Username, | ||||
| 		InviterId:   inviterId, | ||||
| 	} | ||||
| 	if common.EmailVerificationEnabled { | ||||
| 		cleanUser.Email = user.Email | ||||
| 	} | ||||
| 	if err := cleanUser.Insert(); err != nil { | ||||
| 	if err := cleanUser.Insert(inviterId); err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| @@ -228,7 +232,7 @@ func GetUser(c *gin.Context) { | ||||
| 		return | ||||
| 	} | ||||
| 	myRole := c.GetInt("role") | ||||
| 	if myRole <= user.Role { | ||||
| 	if myRole <= user.Role && myRole != common.RoleRootUser { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": "无权获取同级或更高等级用户的信息", | ||||
| @@ -243,6 +247,70 @@ func GetUser(c *gin.Context) { | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func GenerateAccessToken(c *gin.Context) { | ||||
| 	id := c.GetInt("id") | ||||
| 	user, err := model.GetUserById(id, true) | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	user.AccessToken = common.GetUUID() | ||||
|  | ||||
| 	if model.DB.Where("access_token = ?", user.AccessToken).First(user).RowsAffected != 0 { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": "请重试,系统生成的 UUID 竟然重复了!", | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	if err := user.Update(false); err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	c.JSON(http.StatusOK, gin.H{ | ||||
| 		"success": true, | ||||
| 		"message": "", | ||||
| 		"data":    user.AccessToken, | ||||
| 	}) | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func GetAffCode(c *gin.Context) { | ||||
| 	id := c.GetInt("id") | ||||
| 	user, err := model.GetUserById(id, true) | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	if user.AffCode == "" { | ||||
| 		user.AffCode = common.GetRandomString(4) | ||||
| 		if err := user.Update(false); err != nil { | ||||
| 			c.JSON(http.StatusOK, gin.H{ | ||||
| 				"success": false, | ||||
| 				"message": err.Error(), | ||||
| 			}) | ||||
| 			return | ||||
| 		} | ||||
| 	} | ||||
| 	c.JSON(http.StatusOK, gin.H{ | ||||
| 		"success": true, | ||||
| 		"message": "", | ||||
| 		"data":    user.AffCode, | ||||
| 	}) | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func GetSelf(c *gin.Context) { | ||||
| 	id := c.GetInt("id") | ||||
| 	user, err := model.GetUserById(id, false) | ||||
| @@ -290,14 +358,14 @@ func UpdateUser(c *gin.Context) { | ||||
| 		return | ||||
| 	} | ||||
| 	myRole := c.GetInt("role") | ||||
| 	if myRole <= originUser.Role { | ||||
| 	if myRole <= originUser.Role && myRole != common.RoleRootUser { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": "无权更新同权限等级或更高权限等级的用户信息", | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	if myRole <= updatedUser.Role { | ||||
| 	if myRole <= updatedUser.Role && myRole != common.RoleRootUser { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": "无权将其他用户权限等级提升到大于等于自己的权限等级", | ||||
| @@ -315,6 +383,9 @@ func UpdateUser(c *gin.Context) { | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	if originUser.Quota != updatedUser.Quota { | ||||
| 		model.RecordLog(originUser.Id, model.LogTypeManage, fmt.Sprintf("管理员将用户额度从 %s修改为 %s", common.LogQuota(originUser.Quota), common.LogQuota(updatedUser.Quota))) | ||||
| 	} | ||||
| 	c.JSON(http.StatusOK, gin.H{ | ||||
| 		"success": true, | ||||
| 		"message": "", | ||||
| @@ -431,6 +502,13 @@ func CreateUser(c *gin.Context) { | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	if err := common.Validate.Struct(&user); err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": "输入不合法 " + err.Error(), | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	if user.DisplayName == "" { | ||||
| 		user.DisplayName = user.Username | ||||
| 	} | ||||
| @@ -448,7 +526,7 @@ func CreateUser(c *gin.Context) { | ||||
| 		Password:    user.Password, | ||||
| 		DisplayName: user.DisplayName, | ||||
| 	} | ||||
| 	if err := cleanUser.Insert(); err != nil { | ||||
| 	if err := cleanUser.Insert(0); err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| @@ -503,9 +581,23 @@ func ManageUser(c *gin.Context) { | ||||
| 	switch req.Action { | ||||
| 	case "disable": | ||||
| 		user.Status = common.UserStatusDisabled | ||||
| 		if user.Role == common.RoleRootUser { | ||||
| 			c.JSON(http.StatusOK, gin.H{ | ||||
| 				"success": false, | ||||
| 				"message": "无法禁用超级管理员用户", | ||||
| 			}) | ||||
| 			return | ||||
| 		} | ||||
| 	case "enable": | ||||
| 		user.Status = common.UserStatusEnabled | ||||
| 	case "delete": | ||||
| 		if user.Role == common.RoleRootUser { | ||||
| 			c.JSON(http.StatusOK, gin.H{ | ||||
| 				"success": false, | ||||
| 				"message": "无法删除超级管理员用户", | ||||
| 			}) | ||||
| 			return | ||||
| 		} | ||||
| 		if err := user.Delete(); err != nil { | ||||
| 			c.JSON(http.StatusOK, gin.H{ | ||||
| 				"success": false, | ||||
| @@ -521,8 +613,29 @@ func ManageUser(c *gin.Context) { | ||||
| 			}) | ||||
| 			return | ||||
| 		} | ||||
| 		if user.Role >= common.RoleAdminUser { | ||||
| 			c.JSON(http.StatusOK, gin.H{ | ||||
| 				"success": false, | ||||
| 				"message": "该用户已经是管理员", | ||||
| 			}) | ||||
| 			return | ||||
| 		} | ||||
| 		user.Role = common.RoleAdminUser | ||||
| 	case "demote": | ||||
| 		if user.Role == common.RoleRootUser { | ||||
| 			c.JSON(http.StatusOK, gin.H{ | ||||
| 				"success": false, | ||||
| 				"message": "无法降级超级管理员用户", | ||||
| 			}) | ||||
| 			return | ||||
| 		} | ||||
| 		if user.Role == common.RoleCommonUser { | ||||
| 			c.JSON(http.StatusOK, gin.H{ | ||||
| 				"success": false, | ||||
| 				"message": "该用户已经是普通用户", | ||||
| 			}) | ||||
| 			return | ||||
| 		} | ||||
| 		user.Role = common.RoleCommonUser | ||||
| 	} | ||||
|  | ||||
| @@ -577,9 +690,43 @@ func EmailBind(c *gin.Context) { | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	if user.Role == common.RoleRootUser { | ||||
| 		common.RootUserEmail = email | ||||
| 	} | ||||
| 	c.JSON(http.StatusOK, gin.H{ | ||||
| 		"success": true, | ||||
| 		"message": "", | ||||
| 	}) | ||||
| 	return | ||||
| } | ||||
|  | ||||
| type topUpRequest struct { | ||||
| 	Key string `json:"key"` | ||||
| } | ||||
|  | ||||
| func TopUp(c *gin.Context) { | ||||
| 	req := topUpRequest{} | ||||
| 	err := c.ShouldBindJSON(&req) | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	id := c.GetInt("id") | ||||
| 	quota, err := model.Redeem(req.Key, id) | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	c.JSON(http.StatusOK, gin.H{ | ||||
| 		"success": true, | ||||
| 		"message": "", | ||||
| 		"data":    quota, | ||||
| 	}) | ||||
| 	return | ||||
| } | ||||
|   | ||||
| @@ -85,7 +85,7 @@ func WeChatAuth(c *gin.Context) { | ||||
| 			user.Role = common.RoleCommonUser | ||||
| 			user.Status = common.UserStatusEnabled | ||||
|  | ||||
| 			if err := user.Insert(); err != nil { | ||||
| 			if err := user.Insert(0); err != nil { | ||||
| 				c.JSON(http.StatusOK, gin.H{ | ||||
| 					"success": false, | ||||
| 					"message": err.Error(), | ||||
|   | ||||
							
								
								
									
										22
									
								
								docker-compose.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								docker-compose.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,22 @@ | ||||
| version: '3.4' | ||||
|  | ||||
| services: | ||||
|   one-api: | ||||
|     image: ghcr.io/songquanpeng/one-api:latest | ||||
|     container_name: one-api | ||||
|     restart: always | ||||
|     command: --log-dir /app/logs | ||||
|     ports: | ||||
|       - "3000:3000" | ||||
|     volumes: | ||||
|       - ./data:/data | ||||
|       - ./logs:/app/logs | ||||
|     # environment: | ||||
|     #   REDIS_CONN_STRING: redis://default:redispw@localhost:49153 | ||||
|     #   SESSION_SECRET: random_string | ||||
|     #   SQL_DSN: root:123456@tcp(localhost:3306)/one-api | ||||
|     healthcheck: | ||||
|       test: ["CMD-SHELL", "curl -s http://localhost:3000/api/status | grep -o '\"success\":\\s*true' | awk '{print $2}' | grep 'true'"] | ||||
|       interval: 30s | ||||
|       timeout: 10s | ||||
|       retries: 3 | ||||
							
								
								
									
										42
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										42
									
								
								go.mod
									
									
									
									
									
								
							| @@ -8,12 +8,12 @@ require ( | ||||
| 	github.com/gin-contrib/gzip v0.0.6 | ||||
| 	github.com/gin-contrib/sessions v0.0.5 | ||||
| 	github.com/gin-contrib/static v0.0.1 | ||||
| 	github.com/gin-gonic/gin v1.8.1 | ||||
| 	github.com/go-playground/validator/v10 v10.11.1 | ||||
| 	github.com/gin-gonic/gin v1.9.1 | ||||
| 	github.com/go-playground/validator/v10 v10.14.0 | ||||
| 	github.com/go-redis/redis/v8 v8.11.5 | ||||
| 	github.com/google/uuid v1.3.0 | ||||
| 	golang.org/x/crypto v0.1.0 | ||||
| 	gopkg.in/gomail.v2 v2.0.0-20160411212932-81ebce5c23df | ||||
| 	github.com/pkoukk/tiktoken-go v0.1.1 | ||||
| 	golang.org/x/crypto v0.9.0 | ||||
| 	gorm.io/driver/mysql v1.4.3 | ||||
| 	gorm.io/driver/sqlite v1.4.3 | ||||
| 	gorm.io/gorm v1.24.0 | ||||
| @@ -21,13 +21,17 @@ require ( | ||||
|  | ||||
| require ( | ||||
| 	github.com/boj/redistore v0.0.0-20180917114910-cd5dcc76aeff // indirect | ||||
| 	github.com/bytedance/sonic v1.9.1 // indirect | ||||
| 	github.com/cespare/xxhash/v2 v2.1.2 // indirect | ||||
| 	github.com/chenzhuoyu/base64x v0.0.0-20221115062448-fe3a3abad311 // indirect | ||||
| 	github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f // indirect | ||||
| 	github.com/dlclark/regexp2 v1.8.1 // indirect | ||||
| 	github.com/gabriel-vasile/mimetype v1.4.2 // indirect | ||||
| 	github.com/gin-contrib/sse v0.1.0 // indirect | ||||
| 	github.com/go-playground/locales v0.14.0 // indirect | ||||
| 	github.com/go-playground/universal-translator v0.18.0 // indirect | ||||
| 	github.com/go-playground/locales v0.14.1 // indirect | ||||
| 	github.com/go-playground/universal-translator v0.18.1 // indirect | ||||
| 	github.com/go-sql-driver/mysql v1.6.0 // indirect | ||||
| 	github.com/goccy/go-json v0.9.7 // indirect | ||||
| 	github.com/goccy/go-json v0.10.2 // indirect | ||||
| 	github.com/gomodule/redigo v2.0.0+incompatible // indirect | ||||
| 	github.com/gorilla/context v1.1.1 // indirect | ||||
| 	github.com/gorilla/securecookie v1.1.1 // indirect | ||||
| @@ -35,17 +39,19 @@ require ( | ||||
| 	github.com/jinzhu/inflection v1.0.0 // indirect | ||||
| 	github.com/jinzhu/now v1.1.5 // indirect | ||||
| 	github.com/json-iterator/go v1.1.12 // indirect | ||||
| 	github.com/leodido/go-urn v1.2.1 // indirect | ||||
| 	github.com/mattn/go-isatty v0.0.14 // indirect | ||||
| 	github.com/klauspost/cpuid/v2 v2.2.4 // indirect | ||||
| 	github.com/leodido/go-urn v1.2.4 // indirect | ||||
| 	github.com/mattn/go-isatty v0.0.19 // indirect | ||||
| 	github.com/mattn/go-sqlite3 v2.0.3+incompatible // indirect | ||||
| 	github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421 // indirect | ||||
| 	github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd // indirect | ||||
| 	github.com/modern-go/reflect2 v1.0.2 // indirect | ||||
| 	github.com/pelletier/go-toml/v2 v2.0.1 // indirect | ||||
| 	github.com/ugorji/go/codec v1.2.7 // indirect | ||||
| 	golang.org/x/net v0.7.0 // indirect | ||||
| 	golang.org/x/sys v0.5.0 // indirect | ||||
| 	golang.org/x/text v0.7.0 // indirect | ||||
| 	google.golang.org/protobuf v1.28.0 // indirect | ||||
| 	gopkg.in/alexcesaro/quotedprintable.v3 v3.0.0-20150716171945-2caba252f4dc // indirect | ||||
| 	gopkg.in/yaml.v2 v2.4.0 // indirect | ||||
| 	github.com/pelletier/go-toml/v2 v2.0.8 // indirect | ||||
| 	github.com/twitchyliquid64/golang-asm v0.15.1 // indirect | ||||
| 	github.com/ugorji/go/codec v1.2.11 // indirect | ||||
| 	golang.org/x/arch v0.3.0 // indirect | ||||
| 	golang.org/x/net v0.10.0 // indirect | ||||
| 	golang.org/x/sys v0.8.0 // indirect | ||||
| 	golang.org/x/text v0.9.0 // indirect | ||||
| 	google.golang.org/protobuf v1.30.0 // indirect | ||||
| 	gopkg.in/yaml.v3 v3.0.1 // indirect | ||||
| ) | ||||
|   | ||||
							
								
								
									
										89
									
								
								go.sum
									
									
									
									
									
								
							
							
						
						
									
										89
									
								
								go.sum
									
									
									
									
									
								
							| @@ -1,14 +1,24 @@ | ||||
| github.com/boj/redistore v0.0.0-20180917114910-cd5dcc76aeff h1:RmdPFa+slIr4SCBg4st/l/vZWVe9QJKMXGO60Bxbe04= | ||||
| github.com/boj/redistore v0.0.0-20180917114910-cd5dcc76aeff/go.mod h1:+RTT1BOk5P97fT2CiHkbFQwkK3mjsFAP6zCYV2aXtjw= | ||||
| github.com/bytedance/sonic v1.5.0/go.mod h1:ED5hyg4y6t3/9Ku1R6dU/4KyJ48DZ4jPhfY1O2AihPM= | ||||
| github.com/bytedance/sonic v1.9.1 h1:6iJ6NqdoxCDr6mbY8h18oSO+cShGSMRGCEo7F2h0x8s= | ||||
| github.com/bytedance/sonic v1.9.1/go.mod h1:i736AoUSYt75HyZLoJW9ERYxcy6eaN6h4BZXU064P/U= | ||||
| github.com/cespare/xxhash/v2 v2.1.2 h1:YRXhKfTDauu4ajMg1TPgFO5jnlC2HCbmLXMcTG5cbYE= | ||||
| github.com/cespare/xxhash/v2 v2.1.2/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= | ||||
| github.com/chenzhuoyu/base64x v0.0.0-20211019084208-fb5309c8db06/go.mod h1:DH46F32mSOjUmXrMHnKwZdA8wcEefY7UVqBKYGjpdQY= | ||||
| github.com/chenzhuoyu/base64x v0.0.0-20221115062448-fe3a3abad311 h1:qSGYFH7+jGhDF8vLC+iwCD4WpbV1EBDSzWkJODFLams= | ||||
| github.com/chenzhuoyu/base64x v0.0.0-20221115062448-fe3a3abad311/go.mod h1:b583jCggY9gE99b6G5LEC39OIiVsWj+R97kbl5odCEk= | ||||
| github.com/creack/pty v1.1.9/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ33E= | ||||
| github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | ||||
| github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= | ||||
| github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | ||||
| github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f h1:lO4WD4F/rVNCu3HqELle0jiPLLBs70cWOduZpkS1E78= | ||||
| github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f/go.mod h1:cuUVRXasLTGF7a8hSLbxyZXjz+1KgoB3wDUb6vlszIc= | ||||
| github.com/dlclark/regexp2 v1.8.1 h1:6Lcdwya6GjPUNsBct8Lg/yRPwMhABj269AAzdGSiR+0= | ||||
| github.com/dlclark/regexp2 v1.8.1/go.mod h1:DHkYz0B9wPfa6wondMfaivmHpzrQ3v9q8cnmRbL6yW8= | ||||
| github.com/fsnotify/fsnotify v1.4.9 h1:hsms1Qyu0jgnwNXIxa+/V/PDsU6CfLf6CNO8H7IWoS4= | ||||
| github.com/gabriel-vasile/mimetype v1.4.2 h1:w5qFW6JKBz9Y393Y4q372O9A7cUSequkh1Q7OhCmWKU= | ||||
| github.com/gabriel-vasile/mimetype v1.4.2/go.mod h1:zApsH/mKG4w07erKIaJPFiX0Tsq9BFQgN3qGY5GnNgA= | ||||
| github.com/gin-contrib/cors v1.4.0 h1:oJ6gwtUl3lqV0WEIwM/LxPF1QZ5qe2lGWdY2+bz7y0g= | ||||
| github.com/gin-contrib/cors v1.4.0/go.mod h1:bs9pNM0x/UsmHPBWT2xZz9ROh8xYjYkiURUfmBoMlcs= | ||||
| github.com/gin-contrib/gzip v0.0.6 h1:NjcunTcGAj5CO1gn4N8jHOSIeRFHIbn51z6K+xaN4d4= | ||||
| @@ -20,26 +30,30 @@ github.com/gin-contrib/sse v0.1.0/go.mod h1:RHrZQHXnP2xjPF+u1gW/2HnVO7nvIa9PG3Gm | ||||
| github.com/gin-contrib/static v0.0.1 h1:JVxuvHPuUfkoul12N7dtQw7KRn/pSMq7Ue1Va9Swm1U= | ||||
| github.com/gin-contrib/static v0.0.1/go.mod h1:CSxeF+wep05e0kCOsqWdAWbSszmc31zTIbD8TvWl7Hs= | ||||
| github.com/gin-gonic/gin v1.6.3/go.mod h1:75u5sXoLsGZoRN5Sgbi1eraJ4GU3++wFwWzhwvtwp4M= | ||||
| github.com/gin-gonic/gin v1.8.1 h1:4+fr/el88TOO3ewCmQr8cx/CtZ/umlIRIs5M4NTNjf8= | ||||
| github.com/gin-gonic/gin v1.8.1/go.mod h1:ji8BvRH1azfM+SYow9zQ6SZMvR8qOMZHmsCuWR9tTTk= | ||||
| github.com/go-playground/assert/v2 v2.0.1 h1:MsBgLAaY856+nPRTKrp3/OZK38U/wa0CcBYNjji3q3A= | ||||
| github.com/gin-gonic/gin v1.9.1 h1:4idEAncQnU5cB7BeOkPtxjfCSye0AAm1R0RVIqJ+Jmg= | ||||
| github.com/gin-gonic/gin v1.9.1/go.mod h1:hPrL7YrpYKXt5YId3A/Tnip5kqbEAP+KLuI3SUcPTeU= | ||||
| github.com/go-playground/assert/v2 v2.0.1/go.mod h1:VDjEfimB/XKnb+ZQfWdccd7VUvScMdVu0Titje2rxJ4= | ||||
| github.com/go-playground/assert/v2 v2.2.0 h1:JvknZsQTYeFEAhQwI4qEt9cyV5ONwRHC+lYKSsYSR8s= | ||||
| github.com/go-playground/locales v0.13.0/go.mod h1:taPMhCMXrRLJO55olJkUXHZBHCxTMfnGwq/HNwmWNS8= | ||||
| github.com/go-playground/locales v0.14.0 h1:u50s323jtVGugKlcYeyzC0etD1HifMjqmJqb8WugfUU= | ||||
| github.com/go-playground/locales v0.14.0/go.mod h1:sawfccIbzZTqEDETgFXqTho0QybSa7l++s0DH+LDiLs= | ||||
| github.com/go-playground/locales v0.14.1 h1:EWaQ/wswjilfKLTECiXz7Rh+3BjFhfDFKv/oXslEjJA= | ||||
| github.com/go-playground/locales v0.14.1/go.mod h1:hxrqLVvrK65+Rwrd5Fc6F2O76J/NuW9t0sjnWqG1slY= | ||||
| github.com/go-playground/universal-translator v0.17.0/go.mod h1:UkSxE5sNxxRwHyU+Scu5vgOQjsIJAF8j9muTVoKLVtA= | ||||
| github.com/go-playground/universal-translator v0.18.0 h1:82dyy6p4OuJq4/CByFNOn/jYrnRPArHwAcmLoJZxyho= | ||||
| github.com/go-playground/universal-translator v0.18.0/go.mod h1:UvRDBj+xPUEGrFYl+lu/H90nyDXpg0fqeB/AQUGNTVA= | ||||
| github.com/go-playground/universal-translator v0.18.1 h1:Bcnm0ZwsGyWbCzImXv+pAJnYK9S473LQFuzCbDbfSFY= | ||||
| github.com/go-playground/universal-translator v0.18.1/go.mod h1:xekY+UJKNuX9WP91TpwSH2VMlDf28Uj24BCp08ZFTUY= | ||||
| github.com/go-playground/validator/v10 v10.2.0/go.mod h1:uOYAAleCW8F/7oMFd6aG0GOhaH6EGOAJShg8Id5JGkI= | ||||
| github.com/go-playground/validator/v10 v10.10.0/go.mod h1:74x4gJWsvQexRdW8Pn3dXSGrTK4nAUsbPlLADvpJkos= | ||||
| github.com/go-playground/validator/v10 v10.11.1 h1:prmOlTVv+YjZjmRmNSF3VmspqJIxJWXmqUsHwfTRRkQ= | ||||
| github.com/go-playground/validator/v10 v10.11.1/go.mod h1:i+3WkQ1FvaUjjxh1kSvIA4dMGDBiPU55YFDl0WbKdWU= | ||||
| github.com/go-playground/validator/v10 v10.14.0 h1:vgvQWe3XCz3gIeFDm/HnTIbj6UGmg/+t63MyGU2n5js= | ||||
| github.com/go-playground/validator/v10 v10.14.0/go.mod h1:9iXMNT7sEkjXb0I+enO7QXmzG6QCsPWY4zveKFVRSyU= | ||||
| github.com/go-redis/redis/v8 v8.11.5 h1:AcZZR7igkdvfVmQTPnu9WE37LRrO/YrBH5zWyjDC0oI= | ||||
| github.com/go-redis/redis/v8 v8.11.5/go.mod h1:gREzHqY1hg6oD9ngVRbLStwAWKhA0FEgq8Jd4h5lpwo= | ||||
| github.com/go-sql-driver/mysql v1.6.0 h1:BCTh4TKNUYmOmMUcQ3IipzF5prigylS7XXjEkfCHuOE= | ||||
| github.com/go-sql-driver/mysql v1.6.0/go.mod h1:DCzpHaOWr8IXmIStZouvnhqoel9Qv2LBy8hT2VhHyBg= | ||||
| github.com/goccy/go-json v0.9.7 h1:IcB+Aqpx/iMHu5Yooh7jEzJk1JZ7Pjtmys2ukPr7EeM= | ||||
| github.com/goccy/go-json v0.9.7/go.mod h1:6MelG93GURQebXPDq3khkgXZkazVtN9CRI+MGFi0w8I= | ||||
| github.com/goccy/go-json v0.10.2 h1:CrxCmQqYDkv1z7lO7Wbh2HN93uovUHgrECaO5ZrCXAU= | ||||
| github.com/goccy/go-json v0.10.2/go.mod h1:6MelG93GURQebXPDq3khkgXZkazVtN9CRI+MGFi0w8I= | ||||
| github.com/golang/protobuf v1.3.3/go.mod h1:vzj43D7+SQXF/4pzW/hwtAqwc6iTitCiVSaWz5lYuqw= | ||||
| github.com/golang/protobuf v1.5.0/go.mod h1:FsONVRAS9T7sI+LIUmWTfcYkHO4aIWwzhcaSAoJOfIk= | ||||
| github.com/gomodule/redigo v2.0.0+incompatible h1:K/R+8tc58AaqLkqG2Ol3Qk+DR/TlNuhuh457pBFPtt0= | ||||
| @@ -64,6 +78,9 @@ github.com/jinzhu/now v1.1.5/go.mod h1:d3SSVoowX0Lcu0IBviAWJpolVfI5UJVZZ7cO71lE/ | ||||
| github.com/json-iterator/go v1.1.9/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4= | ||||
| github.com/json-iterator/go v1.1.12 h1:PV8peI4a0ysnczrg+LtxykD8LfKY9ML6u2jnxaEnrnM= | ||||
| github.com/json-iterator/go v1.1.12/go.mod h1:e30LSqwooZae/UwlEbR2852Gd8hjQvJoHmT4TnhNGBo= | ||||
| github.com/klauspost/cpuid/v2 v2.0.9/go.mod h1:FInQzS24/EEf25PyTYn52gqo7WaD8xa0213Md/qVLRg= | ||||
| github.com/klauspost/cpuid/v2 v2.2.4 h1:acbojRNwl3o09bUq+yDCtZFc1aiwaAAxtcn8YkZXnvk= | ||||
| github.com/klauspost/cpuid/v2 v2.2.4/go.mod h1:RVVoqg1df56z8g3pUjL/3lE5UfnlrJX8tyFgg4nqhuY= | ||||
| github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= | ||||
| github.com/kr/pretty v0.2.1/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI= | ||||
| github.com/kr/pretty v0.3.0 h1:WgNl7dwNpEZ6jJ9k1snq4pZsg7DOEN8hP9Xw0Tsjwk0= | ||||
| @@ -73,25 +90,31 @@ github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= | ||||
| github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= | ||||
| github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= | ||||
| github.com/leodido/go-urn v1.2.0/go.mod h1:+8+nEpDfqqsY+g338gtMEUOtuK+4dEMhiQEgxpxOKII= | ||||
| github.com/leodido/go-urn v1.2.1 h1:BqpAaACuzVSgi/VLzGZIobT2z4v53pjosyNd9Yv6n/w= | ||||
| github.com/leodido/go-urn v1.2.1/go.mod h1:zt4jvISO2HfUBqxjfIshjdMTYS56ZS/qv49ictyFfxY= | ||||
| github.com/leodido/go-urn v1.2.4 h1:XlAE/cm/ms7TE/VMVoduSpNBoyc2dOxHs5MZSwAN63Q= | ||||
| github.com/leodido/go-urn v1.2.4/go.mod h1:7ZrI8mTSeBSHl/UaRyKQW1qZeMgak41ANeCNaVckg+4= | ||||
| github.com/mattn/go-isatty v0.0.12/go.mod h1:cbi8OIDigv2wuxKPP5vlRcQ1OAZbq2CE4Kysco4FUpU= | ||||
| github.com/mattn/go-isatty v0.0.14 h1:yVuAays6BHfxijgZPzw+3Zlu5yQgKGP2/hcQbHb7S9Y= | ||||
| github.com/mattn/go-isatty v0.0.14/go.mod h1:7GGIvUiUoEMVVmxf/4nioHXj79iQHKdU27kJ6hsGG94= | ||||
| github.com/mattn/go-isatty v0.0.19 h1:JITubQf0MOLdlGRuRq+jtsDlekdYPia9ZFsB8h/APPA= | ||||
| github.com/mattn/go-isatty v0.0.19/go.mod h1:W+V8PltTTMOvKvAeJH7IuucS94S2C6jfK/D7dTCTo3Y= | ||||
| github.com/mattn/go-sqlite3 v1.14.15/go.mod h1:2eHXhiwb8IkHr+BDWZGa96P6+rkvnG63S2DGjv9HUNg= | ||||
| github.com/mattn/go-sqlite3 v2.0.3+incompatible h1:gXHsfypPkaMZrKbD5209QV9jbUTJKjyR5WD3HYQSd+U= | ||||
| github.com/mattn/go-sqlite3 v2.0.3+incompatible/go.mod h1:FPy6KqzDD04eiIsT53CuJW3U88zkxoIYsOqkbpncsNc= | ||||
| github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421 h1:ZqeYNhU3OHLH3mGKHDcjJRFFRrJa6eAM5H+CtDdOsPc= | ||||
| github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q= | ||||
| github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd h1:TRLaZ9cD/w8PVh93nsPXa1VrQ6jlwL5oN8l14QlcNfg= | ||||
| github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q= | ||||
| github.com/modern-go/reflect2 v0.0.0-20180701023420-4b7aa43c6742/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0= | ||||
| github.com/modern-go/reflect2 v1.0.2 h1:xBagoLtFs94CBntxluKeaWgTMpvLxC4ur3nMaC9Gz0M= | ||||
| github.com/modern-go/reflect2 v1.0.2/go.mod h1:yWuevngMOJpCy52FWWMvUC8ws7m/LJsjYzDa0/r8luk= | ||||
| github.com/nxadm/tail v1.4.8 h1:nPr65rt6Y5JFSKQO7qToXr7pePgD6Gwiw05lkbyAQTE= | ||||
| github.com/onsi/ginkgo v1.16.5 h1:8xi0RTUf59SOSfEtZMvwTvXYMzG4gV23XVHOZiXNtnE= | ||||
| github.com/onsi/gomega v1.18.1 h1:M1GfJqGRrBrrGGsbxzV5dqM2U2ApXefZCQpkukxYRLE= | ||||
| github.com/pelletier/go-toml/v2 v2.0.1 h1:8e3L2cCQzLFi2CR4g7vGFuFxX7Jl1kKX8gW+iV0GUKU= | ||||
| github.com/pelletier/go-toml/v2 v2.0.1/go.mod h1:r9LEWfGN8R5k0VXJ+0BkIe7MYkRdwZOjgMj2KwnJFUo= | ||||
| github.com/pelletier/go-toml/v2 v2.0.8 h1:0ctb6s9mE31h0/lhu+J6OPmVeDxJn+kYnJc2jZR9tGQ= | ||||
| github.com/pelletier/go-toml/v2 v2.0.8/go.mod h1:vuYfssBdrU2XDZ9bYydBu6t+6a6PYNcZljzZR9VXg+4= | ||||
| github.com/pkg/diff v0.0.0-20210226163009-20ebb0f2a09e/go.mod h1:pJLUxLENpZxwdsKMEsNbx1VGcRFpLqf3715MtcvvzbA= | ||||
| github.com/pkoukk/tiktoken-go v0.1.1 h1:jtkYlIECjyM9OW1w4rjPmTohK4arORP9V25y6TM6nXo= | ||||
| github.com/pkoukk/tiktoken-go v0.1.1/go.mod h1:boMWvk9pQCOTx11pgu0DrIdrAKgQzzJKUP6vLXaz7Rw= | ||||
| github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= | ||||
| github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= | ||||
| github.com/rogpeppe/go-internal v1.6.1/go.mod h1:xXDCJY+GAPziupqXw64V24skbSoqbTEfhy4qGm1nDQc= | ||||
| @@ -99,56 +122,61 @@ github.com/rogpeppe/go-internal v1.8.0 h1:FCbCCtXNOY3UtUuHUYaghJg4y7Fd14rXifAYUA | ||||
| github.com/rogpeppe/go-internal v1.8.0/go.mod h1:WmiCO8CzOY8rg0OYDC4/i/2WRWAB6poM+XZ2dLUbcbE= | ||||
| github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= | ||||
| github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw= | ||||
| github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo= | ||||
| github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= | ||||
| github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4= | ||||
| github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= | ||||
| github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= | ||||
| github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= | ||||
| github.com/stretchr/testify v1.8.0 h1:pSgiaMZlXftHpm5L7V1+rVB+AZJydKsMxsQBIJw4PKk= | ||||
| github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU= | ||||
| github.com/stretchr/testify v1.8.1/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4= | ||||
| github.com/stretchr/testify v1.8.2/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4= | ||||
| github.com/stretchr/testify v1.8.3 h1:RP3t2pwF7cMEbC1dqtB6poj3niw/9gnV4Cjg5oW5gtY= | ||||
| github.com/stretchr/testify v1.8.3/go.mod h1:sz/lmYIOXD/1dqDmKjjqLyZ2RngseejIcXlSw2iwfAo= | ||||
| github.com/twitchyliquid64/golang-asm v0.15.1 h1:SU5vSMR7hnwNxj24w34ZyCi/FmDZTkS4MhqMhdFk5YI= | ||||
| github.com/twitchyliquid64/golang-asm v0.15.1/go.mod h1:a1lVb/DtPvCB8fslRZhAngC2+aY1QWCk3Cedj/Gdt08= | ||||
| github.com/ugorji/go v1.1.7/go.mod h1:kZn38zHttfInRq0xu/PH0az30d+z6vm202qpg1oXVMw= | ||||
| github.com/ugorji/go v1.2.7/go.mod h1:nF9osbDWLy6bDVv/Rtoh6QgnvNDpmCalQV5urGCCS6M= | ||||
| github.com/ugorji/go/codec v1.1.7/go.mod h1:Ax+UKWsSmolVDwsd+7N3ZtXu+yMGCf907BLYF3GoBXY= | ||||
| github.com/ugorji/go/codec v1.2.7 h1:YPXUKf7fYbp/y8xloBqZOw2qaVggbfwMlI8WM3wZUJ0= | ||||
| github.com/ugorji/go/codec v1.2.7/go.mod h1:WGN1fab3R1fzQlVQTkfxVtIBhWDRqOviHU95kRgeqEY= | ||||
| github.com/ugorji/go/codec v1.2.11 h1:BMaWp1Bb6fHwEtbplGBGJ498wD+LKlNSl25MjdZY4dU= | ||||
| github.com/ugorji/go/codec v1.2.11/go.mod h1:UNopzCgEMSXjBc6AOMqYvWC1ktqTAfzJZUZgYf6w6lg= | ||||
| golang.org/x/arch v0.0.0-20210923205945-b76863e36670/go.mod h1:5om86z9Hs0C8fWVUuoMHwpExlXzs5Tkyp9hOrfG7pp8= | ||||
| golang.org/x/arch v0.3.0 h1:02VY4/ZcO/gBOH6PUaoiptASxtXU10jazRCP865E97k= | ||||
| golang.org/x/arch v0.3.0/go.mod h1:5om86z9Hs0C8fWVUuoMHwpExlXzs5Tkyp9hOrfG7pp8= | ||||
| golang.org/x/crypto v0.0.0-20210711020723-a769d52b0f97/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= | ||||
| golang.org/x/crypto v0.0.0-20211215153901-e495a2d5b3d3/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4= | ||||
| golang.org/x/crypto v0.1.0 h1:MDRAIl0xIo9Io2xV565hzXHw3zVseKrJKodhohM5CjU= | ||||
| golang.org/x/crypto v0.1.0/go.mod h1:RecgLatLF4+eUMCP1PoPZQb+cVrJcOPbHkTkbkB9sbw= | ||||
| golang.org/x/crypto v0.9.0 h1:LF6fAI+IutBocDJ2OT0Q1g8plpYljMZ4+lty+dsqw3g= | ||||
| golang.org/x/crypto v0.9.0/go.mod h1:yrmDGqONDYtNj3tH8X9dzUun2m2lzPa9ngI6/RUPGR0= | ||||
| golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= | ||||
| golang.org/x/net v0.0.0-20211112202133-69e39bad7dc2/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= | ||||
| golang.org/x/net v0.7.0 h1:rJrUqqhjsgNp7KqAIc25s9pZnjU7TUcSY7HcVZjdn1g= | ||||
| golang.org/x/net v0.7.0/go.mod h1:2Tu9+aMcznHK/AK1HMvgo6xiTLG5rD5rZLDS+rp2Bjs= | ||||
| golang.org/x/net v0.10.0 h1:X2//UzNDwYmtCLn7To6G58Wr6f5ahEAQgKNzv9Y951M= | ||||
| golang.org/x/net v0.10.0/go.mod h1:0qNGK6F8kojg2nk9dLZ2mShWaEBan6FAoqfSigmmuDg= | ||||
| golang.org/x/sys v0.0.0-20200116001909-b77594299b42/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210423082822-04245dca01da/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.0.0-20210630005230-0f9fa26af87c/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.0.0-20210806184541-e5e7981a1069/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.5.0 h1:MUK/U/4lj1t1oPg0HfuXDN/Z1wv31ZJ/YcPiGccS4DU= | ||||
| golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.0.0-20220704084225-05e143d24a9e/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.8.0 h1:EBmGv8NaZBZTWvrbjNoL6HVt+IVy3QDQpJs7VRIw3tU= | ||||
| golang.org/x/sys v0.8.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= | ||||
| golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk= | ||||
| golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||
| golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||
| golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ= | ||||
| golang.org/x/text v0.7.0 h1:4BRB4x83lYWy72KwLD/qYDuTu7q9PjSagHvijDw7cLo= | ||||
| golang.org/x/text v0.7.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8= | ||||
| golang.org/x/text v0.9.0 h1:2sjJmO8cDvYveuX97RDLsxlyUxLl+GHoLxBiRdHllBE= | ||||
| golang.org/x/text v0.9.0/go.mod h1:e1OnstbJyHTd6l/uOt8jFFHp6TRDWZR/bV3emEE/zU8= | ||||
| golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= | ||||
| golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543 h1:E7g+9GITq07hpfrRu66IVDexMakfv52eLZ2CXBWiKr4= | ||||
| golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| google.golang.org/protobuf v1.26.0-rc.1/go.mod h1:jlhhOSvTdKEhbULTjvd4ARK9grFBp09yW+WbY/TyQbw= | ||||
| google.golang.org/protobuf v1.28.0 h1:w43yiav+6bVFTBQFZX0r7ipe9JQ1QsbMgHwbBziscLw= | ||||
| google.golang.org/protobuf v1.28.0/go.mod h1:HV8QOd/L58Z+nl8r43ehVNZIU/HEI6OcFqwMG9pJV4I= | ||||
| gopkg.in/alexcesaro/quotedprintable.v3 v3.0.0-20150716171945-2caba252f4dc h1:2gGKlE2+asNV9m7xrywl36YYNnBG5ZQ0r/BOOxqPpmk= | ||||
| gopkg.in/alexcesaro/quotedprintable.v3 v3.0.0-20150716171945-2caba252f4dc/go.mod h1:m7x9LTH6d71AHyAX77c9yqWCCa3UKHcVEj9y7hAtKDk= | ||||
| google.golang.org/protobuf v1.30.0 h1:kPPoIgf3TsEvrm0PFe15JQ+570QVxYzEvvHqChK+cng= | ||||
| google.golang.org/protobuf v1.30.0/go.mod h1:HV8QOd/L58Z+nl8r43ehVNZIU/HEI6OcFqwMG9pJV4I= | ||||
| gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||
| gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||
| gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c h1:Hei/4ADfdWqJk1ZMxUNpqntNwaWcugrBjAiHlqqRiVk= | ||||
| gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c/go.mod h1:JHkPIbrfpd72SG/EVd6muEfDQjcINNoR0C8j2r3qZ4Q= | ||||
| gopkg.in/errgo.v2 v2.1.0/go.mod h1:hNsd1EY+bozCKY1Ytp96fpM3vjJbqLJn88ws8XvfDNI= | ||||
| gopkg.in/gomail.v2 v2.0.0-20160411212932-81ebce5c23df h1:n7WqCuqOuCbNr617RXOY0AWRXxgwEyPp2z+p0+hgMuE= | ||||
| gopkg.in/gomail.v2 v2.0.0-20160411212932-81ebce5c23df/go.mod h1:LRQQ+SO6ZHR7tOkpBDuZnXENFzX8qRjMDMyPD6BRkCw= | ||||
| gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7 h1:uRGJdciOHaEIrze2W8Q3AKkepLTh2hOroT7a+7czfdQ= | ||||
| gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||
| gopkg.in/yaml.v2 v2.2.8/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||
| @@ -165,3 +193,4 @@ gorm.io/driver/sqlite v1.4.3/go.mod h1:0Aq3iPO+v9ZKbcdiz8gLWRw5VOPcBOPUQJFLq5e2e | ||||
| gorm.io/gorm v1.23.8/go.mod h1:l2lP/RyAtc1ynaTjFksBde/O8v9oOGIApu2/xRitmZk= | ||||
| gorm.io/gorm v1.24.0 h1:j/CoiSm6xpRpmzbFJsQHYj+I8bGYWLXVHeYEyyKlF74= | ||||
| gorm.io/gorm v1.24.0/go.mod h1:DVrVomtaYTbqs7gB/x2uVvqnXzv0nqjB396B8cG4dBA= | ||||
| rsc.io/pdf v0.1.1/go.mod h1:n8OzWcQ6Sp37PL01nO98y4iUCRdTGarVfzxY20ICaU4= | ||||
|   | ||||
							
								
								
									
										17
									
								
								main.go
									
									
									
									
									
								
							
							
						
						
									
										17
									
								
								main.go
									
									
									
									
									
								
							| @@ -2,7 +2,6 @@ package main | ||||
|  | ||||
| import ( | ||||
| 	"embed" | ||||
| 	"github.com/gin-contrib/gzip" | ||||
| 	"github.com/gin-contrib/sessions" | ||||
| 	"github.com/gin-contrib/sessions/cookie" | ||||
| 	"github.com/gin-contrib/sessions/redis" | ||||
| @@ -48,10 +47,24 @@ func main() { | ||||
|  | ||||
| 	// Initialize options | ||||
| 	model.InitOptionMap() | ||||
| 	if common.RedisEnabled { | ||||
| 		model.InitChannelCache() | ||||
| 	} | ||||
| 	if os.Getenv("SYNC_FREQUENCY") != "" { | ||||
| 		frequency, err := strconv.Atoi(os.Getenv("SYNC_FREQUENCY")) | ||||
| 		if err != nil { | ||||
| 			common.FatalLog(err) | ||||
| 		} | ||||
| 		go model.SyncOptions(frequency) | ||||
| 		if common.RedisEnabled { | ||||
| 			go model.SyncChannelCache(frequency) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	// Initialize HTTP server | ||||
| 	server := gin.Default() | ||||
| 	server.Use(gzip.Gzip(gzip.DefaultCompression)) | ||||
| 	// This will cause SSE not to work!!! | ||||
| 	//server.Use(gzip.Gzip(gzip.DefaultCompression)) | ||||
| 	server.Use(middleware.CORS()) | ||||
|  | ||||
| 	// Initialize session store | ||||
|   | ||||
| @@ -16,12 +16,31 @@ func authHelper(c *gin.Context, minRole int) { | ||||
| 	id := session.Get("id") | ||||
| 	status := session.Get("status") | ||||
| 	if username == nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": "无权进行此操作,未登录", | ||||
| 		}) | ||||
| 		c.Abort() | ||||
| 		return | ||||
| 		// Check access token | ||||
| 		accessToken := c.Request.Header.Get("Authorization") | ||||
| 		if accessToken == "" { | ||||
| 			c.JSON(http.StatusUnauthorized, gin.H{ | ||||
| 				"success": false, | ||||
| 				"message": "无权进行此操作,未登录且未提供 access token", | ||||
| 			}) | ||||
| 			c.Abort() | ||||
| 			return | ||||
| 		} | ||||
| 		user := model.ValidateAccessToken(accessToken) | ||||
| 		if user != nil && user.Username != "" { | ||||
| 			// Token is valid | ||||
| 			username = user.Username | ||||
| 			role = user.Role | ||||
| 			id = user.Id | ||||
| 			status = user.Status | ||||
| 		} else { | ||||
| 			c.JSON(http.StatusOK, gin.H{ | ||||
| 				"success": false, | ||||
| 				"message": "无权进行此操作,access token 无效", | ||||
| 			}) | ||||
| 			c.Abort() | ||||
| 			return | ||||
| 		} | ||||
| 	} | ||||
| 	if status.(int) == common.UserStatusDisabled { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| @@ -66,6 +85,8 @@ func RootAuth() func(c *gin.Context) { | ||||
| func TokenAuth() func(c *gin.Context) { | ||||
| 	return func(c *gin.Context) { | ||||
| 		key := c.Request.Header.Get("Authorization") | ||||
| 		key = strings.TrimPrefix(key, "Bearer ") | ||||
| 		key = strings.TrimPrefix(key, "sk-") | ||||
| 		parts := strings.Split(key, "-") | ||||
| 		key = parts[0] | ||||
| 		token, err := model.ValidateUserToken(key) | ||||
| @@ -79,9 +100,38 @@ func TokenAuth() func(c *gin.Context) { | ||||
| 			c.Abort() | ||||
| 			return | ||||
| 		} | ||||
| 		if !model.IsUserEnabled(token.UserId) { | ||||
| 			c.JSON(http.StatusOK, gin.H{ | ||||
| 				"error": gin.H{ | ||||
| 					"message": "用户已被封禁", | ||||
| 					"type":    "one_api_error", | ||||
| 				}, | ||||
| 			}) | ||||
| 			c.Abort() | ||||
| 			return | ||||
| 		} | ||||
| 		c.Set("id", token.UserId) | ||||
| 		c.Set("token_id", token.Id) | ||||
| 		c.Set("token_name", token.Name) | ||||
| 		requestURL := c.Request.URL.String() | ||||
| 		consumeQuota := true | ||||
| 		if strings.HasPrefix(requestURL, "/v1/models") { | ||||
| 			consumeQuota = false | ||||
| 		} | ||||
| 		c.Set("consume_quota", consumeQuota) | ||||
| 		if len(parts) > 1 { | ||||
| 			c.Set("channelId", parts[1]) | ||||
| 			if model.IsAdmin(token.UserId) { | ||||
| 				c.Set("channelId", parts[1]) | ||||
| 			} else { | ||||
| 				c.JSON(http.StatusOK, gin.H{ | ||||
| 					"error": gin.H{ | ||||
| 						"message": "普通用户不支持指定渠道", | ||||
| 						"type":    "one_api_error", | ||||
| 					}, | ||||
| 				}) | ||||
| 				c.Abort() | ||||
| 				return | ||||
| 			} | ||||
| 		} | ||||
| 		c.Next() | ||||
| 	} | ||||
|   | ||||
| @@ -6,7 +6,11 @@ import ( | ||||
|  | ||||
| func Cache() func(c *gin.Context) { | ||||
| 	return func(c *gin.Context) { | ||||
| 		c.Header("Cache-Control", "max-age=604800") // one week | ||||
| 		if c.Request.RequestURI == "/" { | ||||
| 			c.Header("Cache-Control", "no-cache") | ||||
| 		} else { | ||||
| 			c.Header("Cache-Control", "max-age=604800") // one week | ||||
| 		} | ||||
| 		c.Next() | ||||
| 	} | ||||
| } | ||||
|   | ||||
| @@ -7,6 +7,9 @@ import ( | ||||
|  | ||||
| func CORS() gin.HandlerFunc { | ||||
| 	config := cors.DefaultConfig() | ||||
| 	config.AllowOrigins = []string{"https://one-api.vercel.app", "http://localhost:3000/"} | ||||
| 	config.AllowAllOrigins = true | ||||
| 	config.AllowCredentials = true | ||||
| 	config.AllowMethods = []string{"GET", "POST", "PUT", "DELETE", "OPTIONS"} | ||||
| 	config.AllowHeaders = []string{"*"} | ||||
| 	return cors.New(config) | ||||
| } | ||||
|   | ||||
| @@ -7,10 +7,18 @@ import ( | ||||
| 	"one-api/common" | ||||
| 	"one-api/model" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| type ModelRequest struct { | ||||
| 	Model string `json:"model"` | ||||
| } | ||||
|  | ||||
| func Distribute() func(c *gin.Context) { | ||||
| 	return func(c *gin.Context) { | ||||
| 		userId := c.GetInt("id") | ||||
| 		userGroup, _ := model.CacheGetUserGroup(userId) | ||||
| 		c.Set("group", userGroup) | ||||
| 		var channel *model.Channel | ||||
| 		channelId, ok := c.Get("channelId") | ||||
| 		if ok { | ||||
| @@ -48,8 +56,24 @@ func Distribute() func(c *gin.Context) { | ||||
| 			} | ||||
| 		} else { | ||||
| 			// Select a channel for the user | ||||
| 			var err error | ||||
| 			channel, err = model.GetRandomChannel() | ||||
| 			var modelRequest ModelRequest | ||||
| 			err := common.UnmarshalBodyReusable(c, &modelRequest) | ||||
| 			if err != nil { | ||||
| 				c.JSON(200, gin.H{ | ||||
| 					"error": gin.H{ | ||||
| 						"message": "无效的请求", | ||||
| 						"type":    "one_api_error", | ||||
| 					}, | ||||
| 				}) | ||||
| 				c.Abort() | ||||
| 				return | ||||
| 			} | ||||
| 			if strings.HasPrefix(c.Request.URL.Path, "/v1/moderations") { | ||||
| 				if modelRequest.Model == "" { | ||||
| 					modelRequest.Model = "text-moderation-stable" | ||||
| 				} | ||||
| 			} | ||||
| 			channel, err = model.CacheGetRandomSatisfiedChannel(userGroup, modelRequest.Model) | ||||
| 			if err != nil { | ||||
| 				c.JSON(200, gin.H{ | ||||
| 					"error": gin.H{ | ||||
| @@ -62,9 +86,12 @@ func Distribute() func(c *gin.Context) { | ||||
| 			} | ||||
| 		} | ||||
| 		c.Set("channel", channel.Type) | ||||
| 		c.Set("channel_id", channel.Id) | ||||
| 		c.Set("channel_name", channel.Name) | ||||
| 		c.Request.Header.Set("Authorization", fmt.Sprintf("Bearer %s", channel.Key)) | ||||
| 		if channel.Type == common.ChannelTypeCustom { | ||||
| 			c.Set("base_url", channel.BaseURL) | ||||
| 		c.Set("base_url", channel.BaseURL) | ||||
| 		if channel.Type == common.ChannelTypeAzure { | ||||
| 			c.Set("api_version", channel.Other) | ||||
| 		} | ||||
| 		c.Next() | ||||
| 	} | ||||
|   | ||||
							
								
								
									
										72
									
								
								model/ability.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										72
									
								
								model/ability.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,72 @@ | ||||
| package model | ||||
|  | ||||
| import ( | ||||
| 	"one-api/common" | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| type Ability struct { | ||||
| 	Group     string `json:"group" gorm:"type:varchar(32);primaryKey;autoIncrement:false"` | ||||
| 	Model     string `json:"model" gorm:"primaryKey;autoIncrement:false"` | ||||
| 	ChannelId int    `json:"channel_id" gorm:"primaryKey;autoIncrement:false;index"` | ||||
| 	Enabled   bool   `json:"enabled"` | ||||
| } | ||||
|  | ||||
| func GetRandomSatisfiedChannel(group string, model string) (*Channel, error) { | ||||
| 	ability := Ability{} | ||||
| 	var err error = nil | ||||
| 	if common.UsingSQLite { | ||||
| 		err = DB.Where("`group` = ? and model = ? and enabled = 1", group, model).Order("RANDOM()").Limit(1).First(&ability).Error | ||||
| 	} else { | ||||
| 		err = DB.Where("`group` = ? and model = ? and enabled = 1", group, model).Order("RAND()").Limit(1).First(&ability).Error | ||||
| 	} | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	channel := Channel{} | ||||
| 	err = DB.First(&channel, "id = ?", ability.ChannelId).Error | ||||
| 	return &channel, err | ||||
| } | ||||
|  | ||||
| func (channel *Channel) AddAbilities() error { | ||||
| 	models_ := strings.Split(channel.Models, ",") | ||||
| 	groups_ := strings.Split(channel.Group, ",") | ||||
| 	abilities := make([]Ability, 0, len(models_)) | ||||
| 	for _, model := range models_ { | ||||
| 		for _, group := range groups_ { | ||||
| 			ability := Ability{ | ||||
| 				Group:     group, | ||||
| 				Model:     model, | ||||
| 				ChannelId: channel.Id, | ||||
| 				Enabled:   channel.Status == common.ChannelStatusEnabled, | ||||
| 			} | ||||
| 			abilities = append(abilities, ability) | ||||
| 		} | ||||
| 	} | ||||
| 	return DB.Create(&abilities).Error | ||||
| } | ||||
|  | ||||
| func (channel *Channel) DeleteAbilities() error { | ||||
| 	return DB.Where("channel_id = ?", channel.Id).Delete(&Ability{}).Error | ||||
| } | ||||
|  | ||||
| // UpdateAbilities updates abilities of this channel. | ||||
| // Make sure the channel is completed before calling this function. | ||||
| func (channel *Channel) UpdateAbilities() error { | ||||
| 	// A quick and dirty way to update abilities | ||||
| 	// First delete all abilities of this channel | ||||
| 	err := channel.DeleteAbilities() | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	// Then add new abilities | ||||
| 	err = channel.AddAbilities() | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func UpdateAbilityStatus(channelId int, status bool) error { | ||||
| 	return DB.Model(&Ability{}).Where("channel_id = ?", channelId).Select("enabled").Update("enabled", status).Error | ||||
| } | ||||
							
								
								
									
										101
									
								
								model/cache.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										101
									
								
								model/cache.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,101 @@ | ||||
| package model | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"one-api/common" | ||||
| 	"sync" | ||||
| 	"time" | ||||
| ) | ||||
|  | ||||
| const ( | ||||
| 	TokenCacheSeconds        = 60 * 60 | ||||
| 	UserId2GroupCacheSeconds = 60 * 60 | ||||
| ) | ||||
|  | ||||
| func CacheGetTokenByKey(key string) (*Token, error) { | ||||
| 	var token Token | ||||
| 	if !common.RedisEnabled { | ||||
| 		err := DB.Where("`key` = ?", key).First(&token).Error | ||||
| 		return &token, err | ||||
| 	} | ||||
| 	tokenObjectString, err := common.RedisGet(fmt.Sprintf("token:%s", key)) | ||||
| 	if err != nil { | ||||
| 		err := DB.Where("`key` = ?", key).First(&token).Error | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 		jsonBytes, err := json.Marshal(token) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 		err = common.RedisSet(fmt.Sprintf("token:%s", key), string(jsonBytes), TokenCacheSeconds*time.Second) | ||||
| 		if err != nil { | ||||
| 			common.SysError("Redis set token error: " + err.Error()) | ||||
| 		} | ||||
| 		return &token, nil | ||||
| 	} | ||||
| 	err = json.Unmarshal([]byte(tokenObjectString), &token) | ||||
| 	return &token, err | ||||
| } | ||||
|  | ||||
| func CacheGetUserGroup(id int) (group string, err error) { | ||||
| 	if !common.RedisEnabled { | ||||
| 		return GetUserGroup(id) | ||||
| 	} | ||||
| 	group, err = common.RedisGet(fmt.Sprintf("user_group:%d", id)) | ||||
| 	if err != nil { | ||||
| 		group, err = GetUserGroup(id) | ||||
| 		if err != nil { | ||||
| 			return "", err | ||||
| 		} | ||||
| 		err = common.RedisSet(fmt.Sprintf("user_group:%d", id), group, UserId2GroupCacheSeconds*time.Second) | ||||
| 		if err != nil { | ||||
| 			common.SysError("Redis set user group error: " + err.Error()) | ||||
| 		} | ||||
| 	} | ||||
| 	return group, err | ||||
| } | ||||
|  | ||||
| var channelId2channel map[int]*Channel | ||||
| var channelSyncLock sync.RWMutex | ||||
| var group2model2channels map[string]map[string][]*Channel | ||||
|  | ||||
| func InitChannelCache() { | ||||
| 	channelSyncLock.Lock() | ||||
| 	defer channelSyncLock.Unlock() | ||||
| 	channelId2channel = make(map[int]*Channel) | ||||
| 	var channels []*Channel | ||||
| 	DB.Find(&channels) | ||||
| 	for _, channel := range channels { | ||||
| 		channelId2channel[channel.Id] = channel | ||||
| 	} | ||||
| 	var abilities []*Ability | ||||
| 	DB.Find(&abilities) | ||||
| 	groups := make(map[string]bool) | ||||
| 	for _, ability := range abilities { | ||||
| 		groups[ability.Group] = true | ||||
| 	} | ||||
| 	group2model2channels = make(map[string]map[string][]*Channel) | ||||
| 	for group := range groups { | ||||
| 		group2model2channels[group] = make(map[string][]*Channel) | ||||
| 		// TODO: implement this | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func SyncChannelCache(frequency int) { | ||||
| 	for { | ||||
| 		time.Sleep(time.Duration(frequency) * time.Second) | ||||
| 		common.SysLog("Syncing channels from database") | ||||
| 		InitChannelCache() | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func CacheGetRandomSatisfiedChannel(group string, model string) (*Channel, error) { | ||||
| 	if !common.RedisEnabled { | ||||
| 		return GetRandomSatisfiedChannel(group, model) | ||||
| 	} | ||||
| 	return GetRandomSatisfiedChannel(group, model) | ||||
| 	// TODO: implement this | ||||
| 	return nil, nil | ||||
| } | ||||
							
								
								
									
										106
									
								
								model/channel.go
									
									
									
									
									
								
							
							
						
						
									
										106
									
								
								model/channel.go
									
									
									
									
									
								
							| @@ -1,26 +1,37 @@ | ||||
| package model | ||||
|  | ||||
| import ( | ||||
| 	_ "gorm.io/driver/sqlite" | ||||
| 	"gorm.io/gorm" | ||||
| 	"one-api/common" | ||||
| ) | ||||
|  | ||||
| type Channel struct { | ||||
| 	Id           int    `json:"id"` | ||||
| 	Type         int    `json:"type" gorm:"default:0"` | ||||
| 	Key          string `json:"key" gorm:"not null"` | ||||
| 	Status       int    `json:"status" gorm:"default:1"` | ||||
| 	Name         string `json:"name" gorm:"index"` | ||||
| 	Weight       int    `json:"weight"` | ||||
| 	CreatedTime  int64  `json:"created_time" gorm:"bigint"` | ||||
| 	AccessedTime int64  `json:"accessed_time" gorm:"bigint"` | ||||
| 	BaseURL      string `json:"base_url" gorm:"column:base_url"` | ||||
| 	Id                 int     `json:"id"` | ||||
| 	Type               int     `json:"type" gorm:"default:0"` | ||||
| 	Key                string  `json:"key" gorm:"not null"` | ||||
| 	Status             int     `json:"status" gorm:"default:1"` | ||||
| 	Name               string  `json:"name" gorm:"index"` | ||||
| 	Weight             int     `json:"weight"` | ||||
| 	CreatedTime        int64   `json:"created_time" gorm:"bigint"` | ||||
| 	TestTime           int64   `json:"test_time" gorm:"bigint"` | ||||
| 	ResponseTime       int     `json:"response_time"` // in milliseconds | ||||
| 	BaseURL            string  `json:"base_url" gorm:"column:base_url"` | ||||
| 	Other              string  `json:"other"` | ||||
| 	Balance            float64 `json:"balance"` // in USD | ||||
| 	BalanceUpdatedTime int64   `json:"balance_updated_time" gorm:"bigint"` | ||||
| 	Models             string  `json:"models"` | ||||
| 	Group              string  `json:"group" gorm:"type:varchar(32);default:'default'"` | ||||
| 	UsedQuota          int64   `json:"used_quota" gorm:"bigint;default:0"` | ||||
| } | ||||
|  | ||||
| func GetAllChannels(startIdx int, num int) ([]*Channel, error) { | ||||
| func GetAllChannels(startIdx int, num int, selectAll bool) ([]*Channel, error) { | ||||
| 	var channels []*Channel | ||||
| 	var err error | ||||
| 	err = DB.Order("id desc").Limit(num).Offset(startIdx).Omit("key").Find(&channels).Error | ||||
| 	if selectAll { | ||||
| 		err = DB.Order("id desc").Find(&channels).Error | ||||
| 	} else { | ||||
| 		err = DB.Order("id desc").Limit(num).Offset(startIdx).Omit("key").Find(&channels).Error | ||||
| 	} | ||||
| 	return channels, err | ||||
| } | ||||
|  | ||||
| @@ -41,31 +52,96 @@ func GetChannelById(id int, selectAll bool) (*Channel, error) { | ||||
| } | ||||
|  | ||||
| func GetRandomChannel() (*Channel, error) { | ||||
| 	// TODO: consider weight | ||||
| 	channel := Channel{} | ||||
| 	var err error = nil | ||||
| 	if common.UsingSQLite { | ||||
| 		err = DB.Where("status = ?", common.ChannelStatusEnabled).Order("RANDOM()").Limit(1).First(&channel).Error | ||||
| 		err = DB.Where("status = ? and `group` = ?", common.ChannelStatusEnabled, "default").Order("RANDOM()").Limit(1).First(&channel).Error | ||||
| 	} else { | ||||
| 		err = DB.Where("status = ?", common.ChannelStatusEnabled).Order("RAND()").Limit(1).First(&channel).Error | ||||
| 		err = DB.Where("status = ? and `group` = ?", common.ChannelStatusEnabled, "default").Order("RAND()").Limit(1).First(&channel).Error | ||||
| 	} | ||||
| 	return &channel, err | ||||
| } | ||||
|  | ||||
| func BatchInsertChannels(channels []Channel) error { | ||||
| 	var err error | ||||
| 	err = DB.Create(&channels).Error | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	for _, channel_ := range channels { | ||||
| 		err = channel_.AddAbilities() | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (channel *Channel) Insert() error { | ||||
| 	var err error | ||||
| 	err = DB.Create(channel).Error | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	err = channel.AddAbilities() | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func (channel *Channel) Update() error { | ||||
| 	var err error | ||||
| 	err = DB.Model(channel).Updates(channel).Error | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	DB.Model(channel).First(channel, "id = ?", channel.Id) | ||||
| 	err = channel.UpdateAbilities() | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func (channel *Channel) UpdateResponseTime(responseTime int64) { | ||||
| 	err := DB.Model(channel).Select("response_time", "test_time").Updates(Channel{ | ||||
| 		TestTime:     common.GetTimestamp(), | ||||
| 		ResponseTime: int(responseTime), | ||||
| 	}).Error | ||||
| 	if err != nil { | ||||
| 		common.SysError("failed to update response time: " + err.Error()) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (channel *Channel) UpdateBalance(balance float64) { | ||||
| 	err := DB.Model(channel).Select("balance_updated_time", "balance").Updates(Channel{ | ||||
| 		BalanceUpdatedTime: common.GetTimestamp(), | ||||
| 		Balance:            balance, | ||||
| 	}).Error | ||||
| 	if err != nil { | ||||
| 		common.SysError("failed to update balance: " + err.Error()) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (channel *Channel) Delete() error { | ||||
| 	var err error | ||||
| 	err = DB.Delete(channel).Error | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	err = channel.DeleteAbilities() | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func UpdateChannelStatusById(id int, status int) { | ||||
| 	err := UpdateAbilityStatus(id, status == common.ChannelStatusEnabled) | ||||
| 	if err != nil { | ||||
| 		common.SysError("failed to update ability status: " + err.Error()) | ||||
| 	} | ||||
| 	err = DB.Model(&Channel{}).Where("id = ?", id).Update("status", status).Error | ||||
| 	if err != nil { | ||||
| 		common.SysError("failed to update channel status: " + err.Error()) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func UpdateChannelUsedQuota(id int, quota int) { | ||||
| 	err := DB.Model(&Channel{}).Where("id = ?", id).Update("used_quota", gorm.Expr("used_quota + ?", quota)).Error | ||||
| 	if err != nil { | ||||
| 		common.SysError("failed to update channel used quota: " + err.Error()) | ||||
| 	} | ||||
| } | ||||
|   | ||||
							
								
								
									
										70
									
								
								model/log.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										70
									
								
								model/log.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,70 @@ | ||||
| package model | ||||
|  | ||||
| import ( | ||||
| 	"gorm.io/gorm" | ||||
| 	"one-api/common" | ||||
| ) | ||||
|  | ||||
| type Log struct { | ||||
| 	Id        int    `json:"id"` | ||||
| 	UserId    int    `json:"user_id" gorm:"index"` | ||||
| 	CreatedAt int64  `json:"created_at" gorm:"bigint"` | ||||
| 	Type      int    `json:"type" gorm:"index"` | ||||
| 	Content   string `json:"content"` | ||||
| } | ||||
|  | ||||
| const ( | ||||
| 	LogTypeUnknown = iota | ||||
| 	LogTypeTopup | ||||
| 	LogTypeConsume | ||||
| 	LogTypeManage | ||||
| 	LogTypeSystem | ||||
| ) | ||||
|  | ||||
| func RecordLog(userId int, logType int, content string) { | ||||
| 	if logType == LogTypeConsume && !common.LogConsumeEnabled { | ||||
| 		return | ||||
| 	} | ||||
| 	log := &Log{ | ||||
| 		UserId:    userId, | ||||
| 		CreatedAt: common.GetTimestamp(), | ||||
| 		Type:      logType, | ||||
| 		Content:   content, | ||||
| 	} | ||||
| 	err := DB.Create(log).Error | ||||
| 	if err != nil { | ||||
| 		common.SysError("failed to record log: " + err.Error()) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func GetAllLogs(logType int, startIdx int, num int) (logs []*Log, err error) { | ||||
| 	var tx *gorm.DB | ||||
| 	if logType == LogTypeUnknown { | ||||
| 		tx = DB | ||||
| 	} else { | ||||
| 		tx = DB.Where("type = ?", logType) | ||||
| 	} | ||||
| 	err = tx.Order("id desc").Limit(num).Offset(startIdx).Find(&logs).Error | ||||
| 	return logs, err | ||||
| } | ||||
|  | ||||
| func GetUserLogs(userId int, logType int, startIdx int, num int) (logs []*Log, err error) { | ||||
| 	var tx *gorm.DB | ||||
| 	if logType == LogTypeUnknown { | ||||
| 		tx = DB.Where("user_id = ?", userId) | ||||
| 	} else { | ||||
| 		tx = DB.Where("user_id = ? and type = ?", userId, logType) | ||||
| 	} | ||||
| 	err = tx.Order("id desc").Limit(num).Offset(startIdx).Omit("id").Find(&logs).Error | ||||
| 	return logs, err | ||||
| } | ||||
|  | ||||
| func SearchAllLogs(keyword string) (logs []*Log, err error) { | ||||
| 	err = DB.Where("type = ? or content LIKE ?", keyword, keyword+"%").Order("id desc").Limit(common.MaxRecentItems).Find(&logs).Error | ||||
| 	return logs, err | ||||
| } | ||||
|  | ||||
| func SearchUserLogs(userId int, keyword string) (logs []*Log, err error) { | ||||
| 	err = DB.Where("user_id = ? and type = ?", userId, keyword).Order("id desc").Limit(common.MaxRecentItems).Omit("id").Find(&logs).Error | ||||
| 	return logs, err | ||||
| } | ||||
| @@ -25,6 +25,8 @@ func createRootAccountIfNeed() error { | ||||
| 			Role:        common.RoleRootUser, | ||||
| 			Status:      common.UserStatusEnabled, | ||||
| 			DisplayName: "Root User", | ||||
| 			AccessToken: common.GetUUID(), | ||||
| 			Quota:       100000000, | ||||
| 		} | ||||
| 		DB.Create(&rootUser) | ||||
| 	} | ||||
| @@ -69,6 +71,18 @@ func InitDB() (err error) { | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		err = db.AutoMigrate(&Redemption{}) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		err = db.AutoMigrate(&Ability{}) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		err = db.AutoMigrate(&Log{}) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		err = createRootAccountIfNeed() | ||||
| 		return err | ||||
| 	} else { | ||||
|   | ||||
| @@ -4,6 +4,7 @@ import ( | ||||
| 	"one-api/common" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| 	"time" | ||||
| ) | ||||
|  | ||||
| type Option struct { | ||||
| @@ -32,12 +33,21 @@ func InitOptionMap() { | ||||
| 	common.OptionMap["WeChatAuthEnabled"] = strconv.FormatBool(common.WeChatAuthEnabled) | ||||
| 	common.OptionMap["TurnstileCheckEnabled"] = strconv.FormatBool(common.TurnstileCheckEnabled) | ||||
| 	common.OptionMap["RegisterEnabled"] = strconv.FormatBool(common.RegisterEnabled) | ||||
| 	common.OptionMap["AutomaticDisableChannelEnabled"] = strconv.FormatBool(common.AutomaticDisableChannelEnabled) | ||||
| 	common.OptionMap["LogConsumeEnabled"] = strconv.FormatBool(common.LogConsumeEnabled) | ||||
| 	common.OptionMap["DisplayInCurrencyEnabled"] = strconv.FormatBool(common.DisplayInCurrencyEnabled) | ||||
| 	common.OptionMap["ChannelDisableThreshold"] = strconv.FormatFloat(common.ChannelDisableThreshold, 'f', -1, 64) | ||||
| 	common.OptionMap["SMTPServer"] = "" | ||||
| 	common.OptionMap["SMTPFrom"] = "" | ||||
| 	common.OptionMap["SMTPPort"] = strconv.Itoa(common.SMTPPort) | ||||
| 	common.OptionMap["SMTPAccount"] = "" | ||||
| 	common.OptionMap["SMTPToken"] = "" | ||||
| 	common.OptionMap["Notice"] = "" | ||||
| 	common.OptionMap["About"] = "" | ||||
| 	common.OptionMap["HomePageContent"] = "" | ||||
| 	common.OptionMap["Footer"] = common.Footer | ||||
| 	common.OptionMap["SystemName"] = common.SystemName | ||||
| 	common.OptionMap["Logo"] = common.Logo | ||||
| 	common.OptionMap["ServerAddress"] = "" | ||||
| 	common.OptionMap["GitHubClientId"] = "" | ||||
| 	common.OptionMap["GitHubClientSecret"] = "" | ||||
| @@ -46,10 +56,35 @@ func InitOptionMap() { | ||||
| 	common.OptionMap["WeChatAccountQRCodeImageURL"] = "" | ||||
| 	common.OptionMap["TurnstileSiteKey"] = "" | ||||
| 	common.OptionMap["TurnstileSecretKey"] = "" | ||||
| 	common.OptionMap["QuotaForNewUser"] = strconv.Itoa(common.QuotaForNewUser) | ||||
| 	common.OptionMap["QuotaForInviter"] = strconv.Itoa(common.QuotaForInviter) | ||||
| 	common.OptionMap["QuotaForInvitee"] = strconv.Itoa(common.QuotaForInvitee) | ||||
| 	common.OptionMap["QuotaRemindThreshold"] = strconv.Itoa(common.QuotaRemindThreshold) | ||||
| 	common.OptionMap["PreConsumedQuota"] = strconv.Itoa(common.PreConsumedQuota) | ||||
| 	common.OptionMap["ModelRatio"] = common.ModelRatio2JSONString() | ||||
| 	common.OptionMap["GroupRatio"] = common.GroupRatio2JSONString() | ||||
| 	common.OptionMap["TopUpLink"] = common.TopUpLink | ||||
| 	common.OptionMap["ChatLink"] = common.ChatLink | ||||
| 	common.OptionMap["QuotaPerUnit"] = strconv.FormatFloat(common.QuotaPerUnit, 'f', -1, 64) | ||||
| 	common.OptionMapRWMutex.Unlock() | ||||
| 	loadOptionsFromDatabase() | ||||
| } | ||||
|  | ||||
| func loadOptionsFromDatabase() { | ||||
| 	options, _ := AllOption() | ||||
| 	for _, option := range options { | ||||
| 		updateOptionMap(option.Key, option.Value) | ||||
| 		err := updateOptionMap(option.Key, option.Value) | ||||
| 		if err != nil { | ||||
| 			common.SysError("Failed to update option map: " + err.Error()) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func SyncOptions(frequency int) { | ||||
| 	for { | ||||
| 		time.Sleep(time.Duration(frequency) * time.Second) | ||||
| 		common.SysLog("Syncing options from database") | ||||
| 		loadOptionsFromDatabase() | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -66,11 +101,10 @@ func UpdateOption(key string, value string) error { | ||||
| 	// otherwise it will execute Update (with all fields). | ||||
| 	DB.Save(&option) | ||||
| 	// Update OptionMap | ||||
| 	updateOptionMap(key, value) | ||||
| 	return nil | ||||
| 	return updateOptionMap(key, value) | ||||
| } | ||||
|  | ||||
| func updateOptionMap(key string, value string) { | ||||
| func updateOptionMap(key string, value string) (err error) { | ||||
| 	common.OptionMapRWMutex.Lock() | ||||
| 	defer common.OptionMapRWMutex.Unlock() | ||||
| 	common.OptionMap[key] = value | ||||
| @@ -104,13 +138,24 @@ func updateOptionMap(key string, value string) { | ||||
| 			common.TurnstileCheckEnabled = boolValue | ||||
| 		case "RegisterEnabled": | ||||
| 			common.RegisterEnabled = boolValue | ||||
| 		case "AutomaticDisableChannelEnabled": | ||||
| 			common.AutomaticDisableChannelEnabled = boolValue | ||||
| 		case "LogConsumeEnabled": | ||||
| 			common.LogConsumeEnabled = boolValue | ||||
| 		case "DisplayInCurrencyEnabled": | ||||
| 			common.DisplayInCurrencyEnabled = boolValue | ||||
| 		} | ||||
| 	} | ||||
| 	switch key { | ||||
| 	case "SMTPServer": | ||||
| 		common.SMTPServer = value | ||||
| 	case "SMTPPort": | ||||
| 		intValue, _ := strconv.Atoi(value) | ||||
| 		common.SMTPPort = intValue | ||||
| 	case "SMTPAccount": | ||||
| 		common.SMTPAccount = value | ||||
| 	case "SMTPFrom": | ||||
| 		common.SMTPFrom = value | ||||
| 	case "SMTPToken": | ||||
| 		common.SMTPToken = value | ||||
| 	case "ServerAddress": | ||||
| @@ -121,6 +166,10 @@ func updateOptionMap(key string, value string) { | ||||
| 		common.GitHubClientSecret = value | ||||
| 	case "Footer": | ||||
| 		common.Footer = value | ||||
| 	case "SystemName": | ||||
| 		common.SystemName = value | ||||
| 	case "Logo": | ||||
| 		common.Logo = value | ||||
| 	case "WeChatServerAddress": | ||||
| 		common.WeChatServerAddress = value | ||||
| 	case "WeChatServerToken": | ||||
| @@ -131,5 +180,28 @@ func updateOptionMap(key string, value string) { | ||||
| 		common.TurnstileSiteKey = value | ||||
| 	case "TurnstileSecretKey": | ||||
| 		common.TurnstileSecretKey = value | ||||
| 	case "QuotaForNewUser": | ||||
| 		common.QuotaForNewUser, _ = strconv.Atoi(value) | ||||
| 	case "QuotaForInviter": | ||||
| 		common.QuotaForInviter, _ = strconv.Atoi(value) | ||||
| 	case "QuotaForInvitee": | ||||
| 		common.QuotaForInvitee, _ = strconv.Atoi(value) | ||||
| 	case "QuotaRemindThreshold": | ||||
| 		common.QuotaRemindThreshold, _ = strconv.Atoi(value) | ||||
| 	case "PreConsumedQuota": | ||||
| 		common.PreConsumedQuota, _ = strconv.Atoi(value) | ||||
| 	case "ModelRatio": | ||||
| 		err = common.UpdateModelRatioByJSONString(value) | ||||
| 	case "GroupRatio": | ||||
| 		err = common.UpdateGroupRatioByJSONString(value) | ||||
| 	case "TopUpLink": | ||||
| 		common.TopUpLink = value | ||||
| 	case "ChatLink": | ||||
| 		common.ChatLink = value | ||||
| 	case "ChannelDisableThreshold": | ||||
| 		common.ChannelDisableThreshold, _ = strconv.ParseFloat(value, 64) | ||||
| 	case "QuotaPerUnit": | ||||
| 		common.QuotaPerUnit, _ = strconv.ParseFloat(value, 64) | ||||
| 	} | ||||
| 	return err | ||||
| } | ||||
|   | ||||
							
								
								
									
										108
									
								
								model/redemption.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										108
									
								
								model/redemption.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,108 @@ | ||||
| package model | ||||
|  | ||||
| import ( | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"one-api/common" | ||||
| ) | ||||
|  | ||||
| type Redemption struct { | ||||
| 	Id           int    `json:"id"` | ||||
| 	UserId       int    `json:"user_id"` | ||||
| 	Key          string `json:"key" gorm:"type:char(32);uniqueIndex"` | ||||
| 	Status       int    `json:"status" gorm:"default:1"` | ||||
| 	Name         string `json:"name" gorm:"index"` | ||||
| 	Quota        int    `json:"quota" gorm:"default:100"` | ||||
| 	CreatedTime  int64  `json:"created_time" gorm:"bigint"` | ||||
| 	RedeemedTime int64  `json:"redeemed_time" gorm:"bigint"` | ||||
| 	Count        int    `json:"count" gorm:"-:all"` // only for api request | ||||
| } | ||||
|  | ||||
| func GetAllRedemptions(startIdx int, num int) ([]*Redemption, error) { | ||||
| 	var redemptions []*Redemption | ||||
| 	var err error | ||||
| 	err = DB.Order("id desc").Limit(num).Offset(startIdx).Find(&redemptions).Error | ||||
| 	return redemptions, err | ||||
| } | ||||
|  | ||||
| func SearchRedemptions(keyword string) (redemptions []*Redemption, err error) { | ||||
| 	err = DB.Where("id = ? or name LIKE ?", keyword, keyword+"%").Find(&redemptions).Error | ||||
| 	return redemptions, err | ||||
| } | ||||
|  | ||||
| func GetRedemptionById(id int) (*Redemption, error) { | ||||
| 	if id == 0 { | ||||
| 		return nil, errors.New("id 为空!") | ||||
| 	} | ||||
| 	redemption := Redemption{Id: id} | ||||
| 	var err error = nil | ||||
| 	err = DB.First(&redemption, "id = ?", id).Error | ||||
| 	return &redemption, err | ||||
| } | ||||
|  | ||||
| func Redeem(key string, userId int) (quota int, err error) { | ||||
| 	if key == "" { | ||||
| 		return 0, errors.New("未提供兑换码") | ||||
| 	} | ||||
| 	if userId == 0 { | ||||
| 		return 0, errors.New("无效的 user id") | ||||
| 	} | ||||
| 	redemption := &Redemption{} | ||||
| 	err = DB.Where("`key` = ?", key).First(redemption).Error | ||||
| 	if err != nil { | ||||
| 		return 0, errors.New("无效的兑换码") | ||||
| 	} | ||||
| 	if redemption.Status != common.RedemptionCodeStatusEnabled { | ||||
| 		return 0, errors.New("该兑换码已被使用") | ||||
| 	} | ||||
| 	err = IncreaseUserQuota(userId, redemption.Quota) | ||||
| 	if err != nil { | ||||
| 		return 0, err | ||||
| 	} | ||||
| 	go func() { | ||||
| 		redemption.RedeemedTime = common.GetTimestamp() | ||||
| 		redemption.Status = common.RedemptionCodeStatusUsed | ||||
| 		err := redemption.SelectUpdate() | ||||
| 		if err != nil { | ||||
| 			common.SysError("更新兑换码状态失败:" + err.Error()) | ||||
| 		} | ||||
| 		RecordLog(userId, LogTypeTopup, fmt.Sprintf("通过兑换码充值 %s", common.LogQuota(redemption.Quota))) | ||||
| 	}() | ||||
| 	return redemption.Quota, nil | ||||
| } | ||||
|  | ||||
| func (redemption *Redemption) Insert() error { | ||||
| 	var err error | ||||
| 	err = DB.Create(redemption).Error | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func (redemption *Redemption) SelectUpdate() error { | ||||
| 	// This can update zero values | ||||
| 	return DB.Model(redemption).Select("redeemed_time", "status").Updates(redemption).Error | ||||
| } | ||||
|  | ||||
| // Update Make sure your token's fields is completed, because this will update non-zero values | ||||
| func (redemption *Redemption) Update() error { | ||||
| 	var err error | ||||
| 	err = DB.Model(redemption).Select("name", "status", "quota", "redeemed_time").Updates(redemption).Error | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func (redemption *Redemption) Delete() error { | ||||
| 	var err error | ||||
| 	err = DB.Delete(redemption).Error | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func DeleteRedemptionById(id int) (err error) { | ||||
| 	if id == 0 { | ||||
| 		return errors.New("id 为空!") | ||||
| 	} | ||||
| 	redemption := Redemption{Id: id} | ||||
| 	err = DB.Where(redemption).First(&redemption).Error | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	return redemption.Delete() | ||||
| } | ||||
							
								
								
									
										159
									
								
								model/token.go
									
									
									
									
									
								
							
							
						
						
									
										159
									
								
								model/token.go
									
									
									
									
									
								
							| @@ -2,21 +2,22 @@ package model | ||||
|  | ||||
| import ( | ||||
| 	"errors" | ||||
| 	_ "gorm.io/driver/sqlite" | ||||
| 	"fmt" | ||||
| 	"gorm.io/gorm" | ||||
| 	"one-api/common" | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| type Token struct { | ||||
| 	Id           int    `json:"id"` | ||||
| 	UserId       int    `json:"user_id"` | ||||
| 	Key          string `json:"key" gorm:"uniqueIndex"` | ||||
| 	Status       int    `json:"status" gorm:"default:1"` | ||||
| 	Name         string `json:"name" gorm:"index" ` | ||||
| 	CreatedTime  int64  `json:"created_time" gorm:"bigint"` | ||||
| 	AccessedTime int64  `json:"accessed_time" gorm:"bigint"` | ||||
| 	ExpiredTime  int64  `json:"expired_time" gorm:"bigint;default:-1"` // -1 means never expired | ||||
| 	RemainTimes  int    `json:"remain_times" gorm:"default:-1"`        // -1 means infinite times | ||||
| 	Id             int    `json:"id"` | ||||
| 	UserId         int    `json:"user_id"` | ||||
| 	Key            string `json:"key" gorm:"type:char(48);uniqueIndex"` | ||||
| 	Status         int    `json:"status" gorm:"default:1"` | ||||
| 	Name           string `json:"name" gorm:"index" ` | ||||
| 	CreatedTime    int64  `json:"created_time" gorm:"bigint"` | ||||
| 	AccessedTime   int64  `json:"accessed_time" gorm:"bigint"` | ||||
| 	ExpiredTime    int64  `json:"expired_time" gorm:"bigint;default:-1"` // -1 means never expired | ||||
| 	RemainQuota    int    `json:"remain_quota" gorm:"default:0"` | ||||
| 	UnlimitedQuota bool   `json:"unlimited_quota" gorm:"default:false"` | ||||
| } | ||||
|  | ||||
| func GetAllUserTokens(userId int, startIdx int, num int) ([]*Token, error) { | ||||
| @@ -27,45 +28,45 @@ func GetAllUserTokens(userId int, startIdx int, num int) ([]*Token, error) { | ||||
| } | ||||
|  | ||||
| func SearchUserTokens(userId int, keyword string) (tokens []*Token, err error) { | ||||
| 	err = DB.Where("user_id = ?", userId).Where("id = ? or name LIKE ?", keyword, keyword+"%").Find(&tokens).Error | ||||
| 	err = DB.Where("user_id = ?", userId).Where("name LIKE ?", keyword+"%").Find(&tokens).Error | ||||
| 	return tokens, err | ||||
| } | ||||
|  | ||||
| func ValidateUserToken(key string) (token *Token, err error) { | ||||
| 	if key == "" { | ||||
| 		return nil, errors.New("未提供 token") | ||||
| 		return nil, errors.New("未提供令牌") | ||||
| 	} | ||||
| 	key = strings.Replace(key, "Bearer ", "", 1) | ||||
| 	token = &Token{} | ||||
| 	err = DB.Where("key = ?", key).First(token).Error | ||||
| 	token, err = CacheGetTokenByKey(key) | ||||
| 	if err == nil { | ||||
| 		if token.Status != common.TokenStatusEnabled { | ||||
| 			return nil, errors.New("该 token 状态不可用") | ||||
| 			return nil, errors.New("该令牌状态不可用") | ||||
| 		} | ||||
| 		if token.ExpiredTime != -1 && token.ExpiredTime < common.GetTimestamp() { | ||||
| 			token.Status = common.TokenStatusExpired | ||||
| 			err := token.SelectUpdate() | ||||
| 			if err != nil { | ||||
| 				common.SysError("更新 token 状态失败:" + err.Error()) | ||||
| 				common.SysError("更新令牌状态失败:" + err.Error()) | ||||
| 			} | ||||
| 			return nil, errors.New("该 token 已过期") | ||||
| 			return nil, errors.New("该令牌已过期") | ||||
| 		} | ||||
| 		if !token.UnlimitedQuota && token.RemainQuota <= 0 { | ||||
| 			token.Status = common.TokenStatusExhausted | ||||
| 			err := token.SelectUpdate() | ||||
| 			if err != nil { | ||||
| 				common.SysError("更新令牌状态失败:" + err.Error()) | ||||
| 			} | ||||
| 			return nil, errors.New("该令牌额度已用尽") | ||||
| 		} | ||||
| 		go func() { | ||||
| 			token.AccessedTime = common.GetTimestamp() | ||||
| 			if token.RemainTimes > 0 { | ||||
| 				token.RemainTimes-- | ||||
| 				if token.RemainTimes == 0 { | ||||
| 					token.Status = common.TokenStatusExhausted | ||||
| 				} | ||||
| 			} | ||||
| 			err := token.SelectUpdate() | ||||
| 			if err != nil { | ||||
| 				common.SysError("更新 token 失败:" + err.Error()) | ||||
| 				common.SysError("更新令牌失败:" + err.Error()) | ||||
| 			} | ||||
| 		}() | ||||
| 		return token, nil | ||||
| 	} | ||||
| 	return nil, err | ||||
| 	return nil, errors.New("无效的令牌") | ||||
| } | ||||
|  | ||||
| func GetTokenByIds(id int, userId int) (*Token, error) { | ||||
| @@ -78,21 +79,32 @@ func GetTokenByIds(id int, userId int) (*Token, error) { | ||||
| 	return &token, err | ||||
| } | ||||
|  | ||||
| func GetTokenById(id int) (*Token, error) { | ||||
| 	if id == 0 { | ||||
| 		return nil, errors.New("id 为空!") | ||||
| 	} | ||||
| 	token := Token{Id: id} | ||||
| 	var err error = nil | ||||
| 	err = DB.First(&token, "id = ?", id).Error | ||||
| 	return &token, err | ||||
| } | ||||
|  | ||||
| func (token *Token) Insert() error { | ||||
| 	var err error | ||||
| 	err = DB.Create(token).Error | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| // Update Make sure your token's fields is completed, because this will update non-zero values | ||||
| func (token *Token) Update() error { | ||||
| 	var err error | ||||
| 	err = DB.Model(token).Updates(token).Error | ||||
| 	err = DB.Model(token).Select("name", "status", "expired_time", "remain_quota", "unlimited_quota").Updates(token).Error | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func (token *Token) SelectUpdate() error { | ||||
| 	// This can update zero values | ||||
| 	return DB.Model(token).Select("accessed_time", "remain_times", "status").Updates(token).Error | ||||
| 	return DB.Model(token).Select("accessed_time", "status").Updates(token).Error | ||||
| } | ||||
|  | ||||
| func (token *Token) Delete() error { | ||||
| @@ -113,3 +125,92 @@ func DeleteTokenById(id int, userId int) (err error) { | ||||
| 	} | ||||
| 	return token.Delete() | ||||
| } | ||||
|  | ||||
| func IncreaseTokenQuota(id int, quota int) (err error) { | ||||
| 	if quota < 0 { | ||||
| 		return errors.New("quota 不能为负数!") | ||||
| 	} | ||||
| 	err = DB.Model(&Token{}).Where("id = ?", id).Update("remain_quota", gorm.Expr("remain_quota + ?", quota)).Error | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func DecreaseTokenQuota(id int, quota int) (err error) { | ||||
| 	if quota < 0 { | ||||
| 		return errors.New("quota 不能为负数!") | ||||
| 	} | ||||
| 	err = DB.Model(&Token{}).Where("id = ?", id).Update("remain_quota", gorm.Expr("remain_quota - ?", quota)).Error | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func PreConsumeTokenQuota(tokenId int, quota int) (err error) { | ||||
| 	if quota < 0 { | ||||
| 		return errors.New("quota 不能为负数!") | ||||
| 	} | ||||
| 	token, err := GetTokenById(tokenId) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	if !token.UnlimitedQuota && token.RemainQuota < quota { | ||||
| 		return errors.New("令牌额度不足") | ||||
| 	} | ||||
| 	userQuota, err := GetUserQuota(token.UserId) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	if userQuota < quota { | ||||
| 		return errors.New("用户额度不足") | ||||
| 	} | ||||
| 	quotaTooLow := userQuota >= common.QuotaRemindThreshold && userQuota-quota < common.QuotaRemindThreshold | ||||
| 	noMoreQuota := userQuota-quota <= 0 | ||||
| 	if quotaTooLow || noMoreQuota { | ||||
| 		go func() { | ||||
| 			email, err := GetUserEmail(token.UserId) | ||||
| 			if err != nil { | ||||
| 				common.SysError("获取用户邮箱失败:" + err.Error()) | ||||
| 			} | ||||
| 			prompt := "您的额度即将用尽" | ||||
| 			if noMoreQuota { | ||||
| 				prompt = "您的额度已用尽" | ||||
| 			} | ||||
| 			if email != "" { | ||||
| 				topUpLink := fmt.Sprintf("%s/topup", common.ServerAddress) | ||||
| 				err = common.SendEmail(prompt, email, | ||||
| 					fmt.Sprintf("%s,当前剩余额度为 %d,为了不影响您的使用,请及时充值。<br/>充值链接:<a href='%s'>%s</a>", prompt, userQuota, topUpLink, topUpLink)) | ||||
| 				if err != nil { | ||||
| 					common.SysError("发送邮件失败:" + err.Error()) | ||||
| 				} | ||||
| 			} | ||||
| 		}() | ||||
| 	} | ||||
| 	if !token.UnlimitedQuota { | ||||
| 		err = DecreaseTokenQuota(tokenId, quota) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
| 	err = DecreaseUserQuota(token.UserId, quota) | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func PostConsumeTokenQuota(tokenId int, quota int) (err error) { | ||||
| 	token, err := GetTokenById(tokenId) | ||||
| 	if quota > 0 { | ||||
| 		err = DecreaseUserQuota(token.UserId, quota) | ||||
| 	} else { | ||||
| 		err = IncreaseUserQuota(token.UserId, -quota) | ||||
| 	} | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	if !token.UnlimitedQuota { | ||||
| 		if quota > 0 { | ||||
| 			err = DecreaseTokenQuota(tokenId, quota) | ||||
| 		} else { | ||||
| 			err = IncreaseTokenQuota(tokenId, -quota) | ||||
| 		} | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|   | ||||
							
								
								
									
										138
									
								
								model/user.go
									
									
									
									
									
								
							
							
						
						
									
										138
									
								
								model/user.go
									
									
									
									
									
								
							| @@ -2,7 +2,10 @@ package model | ||||
|  | ||||
| import ( | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"gorm.io/gorm" | ||||
| 	"one-api/common" | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| // User if you add sensitive fields, don't forget to clean them in setupLogin function. | ||||
| @@ -17,8 +20,14 @@ type User struct { | ||||
| 	Email            string `json:"email" gorm:"index" validate:"max=50"` | ||||
| 	GitHubId         string `json:"github_id" gorm:"column:github_id;index"` | ||||
| 	WeChatId         string `json:"wechat_id" gorm:"column:wechat_id;index"` | ||||
| 	VerificationCode string `json:"verification_code" gorm:"-:all"` // this field is only for Email verification, don't save it to database! | ||||
| 	Balance          int    `json:"balance" gorm:"type:int;default:0"` | ||||
| 	VerificationCode string `json:"verification_code" gorm:"-:all"`                                    // this field is only for Email verification, don't save it to database! | ||||
| 	AccessToken      string `json:"access_token" gorm:"type:char(32);column:access_token;uniqueIndex"` // this token is for system management | ||||
| 	Quota            int    `json:"quota" gorm:"type:int;default:0"` | ||||
| 	UsedQuota        int    `json:"used_quota" gorm:"type:int;default:0;column:used_quota"` // used quota | ||||
| 	RequestCount     int    `json:"request_count" gorm:"type:int;default:0;"`               // request number | ||||
| 	Group            string `json:"group" gorm:"type:varchar(32);default:'default'"` | ||||
| 	AffCode          string `json:"aff_code" gorm:"type:varchar(32);column:aff_code;uniqueIndex"` | ||||
| 	InviterId        int    `json:"inviter_id" gorm:"type:int;column:inviter_id;index"` | ||||
| } | ||||
|  | ||||
| func GetMaxUserId() int { | ||||
| @@ -51,6 +60,15 @@ func GetUserById(id int, selectAll bool) (*User, error) { | ||||
| 	return &user, err | ||||
| } | ||||
|  | ||||
| func GetUserIdByAffCode(affCode string) (int, error) { | ||||
| 	if affCode == "" { | ||||
| 		return 0, errors.New("affCode 为空!") | ||||
| 	} | ||||
| 	var user User | ||||
| 	err := DB.Select("id").First(&user, "aff_code = ?", affCode).Error | ||||
| 	return user.Id, err | ||||
| } | ||||
|  | ||||
| func DeleteUserById(id int) (err error) { | ||||
| 	if id == 0 { | ||||
| 		return errors.New("id 为空!") | ||||
| @@ -59,7 +77,7 @@ func DeleteUserById(id int) (err error) { | ||||
| 	return user.Delete() | ||||
| } | ||||
|  | ||||
| func (user *User) Insert() error { | ||||
| func (user *User) Insert(inviterId int) error { | ||||
| 	var err error | ||||
| 	if user.Password != "" { | ||||
| 		user.Password, err = common.Password2Hash(user.Password) | ||||
| @@ -67,8 +85,27 @@ func (user *User) Insert() error { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
| 	err = DB.Create(user).Error | ||||
| 	return err | ||||
| 	user.Quota = common.QuotaForNewUser | ||||
| 	user.AccessToken = common.GetUUID() | ||||
| 	user.AffCode = common.GetRandomString(4) | ||||
| 	result := DB.Create(user) | ||||
| 	if result.Error != nil { | ||||
| 		return result.Error | ||||
| 	} | ||||
| 	if common.QuotaForNewUser > 0 { | ||||
| 		RecordLog(user.Id, LogTypeSystem, fmt.Sprintf("新用户注册赠送 %s", common.LogQuota(common.QuotaForNewUser))) | ||||
| 	} | ||||
| 	if inviterId != 0 { | ||||
| 		if common.QuotaForInvitee > 0 { | ||||
| 			_ = IncreaseUserQuota(user.Id, common.QuotaForInvitee) | ||||
| 			RecordLog(user.Id, LogTypeSystem, fmt.Sprintf("使用邀请码赠送 %s", common.LogQuota(common.QuotaForInvitee))) | ||||
| 		} | ||||
| 		if common.QuotaForInviter > 0 { | ||||
| 			_ = IncreaseUserQuota(inviterId, common.QuotaForInviter) | ||||
| 			RecordLog(inviterId, LogTypeSystem, fmt.Sprintf("邀请用户赠送 %s", common.LogQuota(common.QuotaForInviter))) | ||||
| 		} | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (user *User) Update(updatePassword bool) error { | ||||
| @@ -175,3 +212,94 @@ func ResetUserPasswordByEmail(email string, password string) error { | ||||
| 	err = DB.Model(&User{}).Where("email = ?", email).Update("password", hashedPassword).Error | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func IsAdmin(userId int) bool { | ||||
| 	if userId == 0 { | ||||
| 		return false | ||||
| 	} | ||||
| 	var user User | ||||
| 	err := DB.Where("id = ?", userId).Select("role").Find(&user).Error | ||||
| 	if err != nil { | ||||
| 		common.SysError("No such user " + err.Error()) | ||||
| 		return false | ||||
| 	} | ||||
| 	return user.Role >= common.RoleAdminUser | ||||
| } | ||||
|  | ||||
| func IsUserEnabled(userId int) bool { | ||||
| 	if userId == 0 { | ||||
| 		return false | ||||
| 	} | ||||
| 	var user User | ||||
| 	err := DB.Where("id = ?", userId).Select("status").Find(&user).Error | ||||
| 	if err != nil { | ||||
| 		common.SysError("No such user " + err.Error()) | ||||
| 		return false | ||||
| 	} | ||||
| 	return user.Status == common.UserStatusEnabled | ||||
| } | ||||
|  | ||||
| func ValidateAccessToken(token string) (user *User) { | ||||
| 	if token == "" { | ||||
| 		return nil | ||||
| 	} | ||||
| 	token = strings.Replace(token, "Bearer ", "", 1) | ||||
| 	user = &User{} | ||||
| 	if DB.Where("access_token = ?", token).First(user).RowsAffected == 1 { | ||||
| 		return user | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func GetUserQuota(id int) (quota int, err error) { | ||||
| 	err = DB.Model(&User{}).Where("id = ?", id).Select("quota").Find("a).Error | ||||
| 	return quota, err | ||||
| } | ||||
|  | ||||
| func GetUserUsedQuota(id int) (quota int, err error) { | ||||
| 	err = DB.Model(&User{}).Where("id = ?", id).Select("used_quota").Find("a).Error | ||||
| 	return quota, err | ||||
| } | ||||
|  | ||||
| func GetUserEmail(id int) (email string, err error) { | ||||
| 	err = DB.Model(&User{}).Where("id = ?", id).Select("email").Find(&email).Error | ||||
| 	return email, err | ||||
| } | ||||
|  | ||||
| func GetUserGroup(id int) (group string, err error) { | ||||
| 	err = DB.Model(&User{}).Where("id = ?", id).Select("`group`").Find(&group).Error | ||||
| 	return group, err | ||||
| } | ||||
|  | ||||
| func IncreaseUserQuota(id int, quota int) (err error) { | ||||
| 	if quota < 0 { | ||||
| 		return errors.New("quota 不能为负数!") | ||||
| 	} | ||||
| 	err = DB.Model(&User{}).Where("id = ?", id).Update("quota", gorm.Expr("quota + ?", quota)).Error | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func DecreaseUserQuota(id int, quota int) (err error) { | ||||
| 	if quota < 0 { | ||||
| 		return errors.New("quota 不能为负数!") | ||||
| 	} | ||||
| 	err = DB.Model(&User{}).Where("id = ?", id).Update("quota", gorm.Expr("quota - ?", quota)).Error | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func GetRootUserEmail() (email string) { | ||||
| 	DB.Model(&User{}).Where("role = ?", common.RoleRootUser).Select("email").Find(&email) | ||||
| 	return email | ||||
| } | ||||
|  | ||||
| func UpdateUserUsedQuotaAndRequestCount(id int, quota int) { | ||||
| 	err := DB.Model(&User{}).Where("id = ?", id).Updates( | ||||
| 		map[string]interface{}{ | ||||
| 			"used_quota":    gorm.Expr("used_quota + ?", quota), | ||||
| 			"request_count": gorm.Expr("request_count + ?", 1), | ||||
| 		}, | ||||
| 	).Error | ||||
| 	if err != nil { | ||||
| 		common.SysError("Failed to update user used quota and request count: " + err.Error()) | ||||
| 	} | ||||
| } | ||||
|   | ||||
							
								
								
									
										13
									
								
								one-api.service
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								one-api.service
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | ||||
| [Unit] | ||||
| Description=One API Service | ||||
| After=network.target | ||||
|  | ||||
| [Service] | ||||
| User=yourusername                  # 守护进程用户名 | ||||
| WorkingDirectory=/path/to/One-API  # One API运行路径 | ||||
| ExecStart=/path/to/One-API/one-api --port 3000 --log-dir /path/to/One-API/logs  # 端口 | ||||
| Restart=always | ||||
| RestartSec=5 | ||||
|  | ||||
| [Install] | ||||
| WantedBy=multi-user.target | ||||
| @@ -1,6 +1,7 @@ | ||||
| package router | ||||
|  | ||||
| import ( | ||||
| 	"github.com/gin-contrib/gzip" | ||||
| 	"github.com/gin-gonic/gin" | ||||
| 	"one-api/controller" | ||||
| 	"one-api/middleware" | ||||
| @@ -8,11 +9,13 @@ import ( | ||||
|  | ||||
| func SetApiRouter(router *gin.Engine) { | ||||
| 	apiRouter := router.Group("/api") | ||||
| 	apiRouter.Use(gzip.Gzip(gzip.DefaultCompression)) | ||||
| 	apiRouter.Use(middleware.GlobalAPIRateLimit()) | ||||
| 	{ | ||||
| 		apiRouter.GET("/status", controller.GetStatus) | ||||
| 		apiRouter.GET("/notice", controller.GetNotice) | ||||
| 		apiRouter.GET("/about", controller.GetAbout) | ||||
| 		apiRouter.GET("/home_page_content", controller.GetHomePageContent) | ||||
| 		apiRouter.GET("/verification", middleware.CriticalRateLimit(), middleware.TurnstileCheck(), controller.SendEmailVerification) | ||||
| 		apiRouter.GET("/reset_password", middleware.CriticalRateLimit(), middleware.TurnstileCheck(), controller.SendPasswordResetEmail) | ||||
| 		apiRouter.POST("/user/reset", middleware.CriticalRateLimit(), controller.ResetPassword) | ||||
| @@ -33,6 +36,9 @@ func SetApiRouter(router *gin.Engine) { | ||||
| 				selfRoute.GET("/self", controller.GetSelf) | ||||
| 				selfRoute.PUT("/self", controller.UpdateSelf) | ||||
| 				selfRoute.DELETE("/self", controller.DeleteSelf) | ||||
| 				selfRoute.GET("/token", controller.GenerateAccessToken) | ||||
| 				selfRoute.GET("/aff", controller.GetAffCode) | ||||
| 				selfRoute.POST("/topup", controller.TopUp) | ||||
| 			} | ||||
|  | ||||
| 			adminRoute := userRoute.Group("/") | ||||
| @@ -58,7 +64,12 @@ func SetApiRouter(router *gin.Engine) { | ||||
| 		{ | ||||
| 			channelRoute.GET("/", controller.GetAllChannels) | ||||
| 			channelRoute.GET("/search", controller.SearchChannels) | ||||
| 			channelRoute.GET("/models", controller.ListModels) | ||||
| 			channelRoute.GET("/:id", controller.GetChannel) | ||||
| 			channelRoute.GET("/test", controller.TestAllChannels) | ||||
| 			channelRoute.GET("/test/:id", controller.TestChannel) | ||||
| 			channelRoute.GET("/update_balance", controller.UpdateAllChannelsBalance) | ||||
| 			channelRoute.GET("/update_balance/:id", controller.UpdateChannelBalance) | ||||
| 			channelRoute.POST("/", controller.AddChannel) | ||||
| 			channelRoute.PUT("/", controller.UpdateChannel) | ||||
| 			channelRoute.DELETE("/:id", controller.DeleteChannel) | ||||
| @@ -73,5 +84,25 @@ func SetApiRouter(router *gin.Engine) { | ||||
| 			tokenRoute.PUT("/", controller.UpdateToken) | ||||
| 			tokenRoute.DELETE("/:id", controller.DeleteToken) | ||||
| 		} | ||||
| 		redemptionRoute := apiRouter.Group("/redemption") | ||||
| 		redemptionRoute.Use(middleware.AdminAuth()) | ||||
| 		{ | ||||
| 			redemptionRoute.GET("/", controller.GetAllRedemptions) | ||||
| 			redemptionRoute.GET("/search", controller.SearchRedemptions) | ||||
| 			redemptionRoute.GET("/:id", controller.GetRedemption) | ||||
| 			redemptionRoute.POST("/", controller.AddRedemption) | ||||
| 			redemptionRoute.PUT("/", controller.UpdateRedemption) | ||||
| 			redemptionRoute.DELETE("/:id", controller.DeleteRedemption) | ||||
| 		} | ||||
| 		logRoute := apiRouter.Group("/log") | ||||
| 		logRoute.GET("/", middleware.AdminAuth(), controller.GetAllLogs) | ||||
| 		logRoute.GET("/search", middleware.AdminAuth(), controller.SearchAllLogs) | ||||
| 		logRoute.GET("/self", middleware.UserAuth(), controller.GetUserLogs) | ||||
| 		logRoute.GET("/self/search", middleware.UserAuth(), controller.SearchUserLogs) | ||||
| 		groupRoute := apiRouter.Group("/group") | ||||
| 		groupRoute.Use(middleware.AdminAuth()) | ||||
| 		{ | ||||
| 			groupRoute.GET("/", controller.GetGroups) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|   | ||||
							
								
								
									
										21
									
								
								router/dashboard.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								router/dashboard.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,21 @@ | ||||
| package router | ||||
|  | ||||
| import ( | ||||
| 	"github.com/gin-contrib/gzip" | ||||
| 	"github.com/gin-gonic/gin" | ||||
| 	"one-api/controller" | ||||
| 	"one-api/middleware" | ||||
| ) | ||||
|  | ||||
| func SetDashboardRouter(router *gin.Engine) { | ||||
| 	apiRouter := router.Group("/") | ||||
| 	apiRouter.Use(gzip.Gzip(gzip.DefaultCompression)) | ||||
| 	apiRouter.Use(middleware.GlobalAPIRateLimit()) | ||||
| 	apiRouter.Use(middleware.TokenAuth()) | ||||
| 	{ | ||||
| 		apiRouter.GET("/dashboard/billing/subscription", controller.GetSubscription) | ||||
| 		apiRouter.GET("/v1/dashboard/billing/subscription", controller.GetSubscription) | ||||
| 		apiRouter.GET("/dashboard/billing/usage", controller.GetUsage) | ||||
| 		apiRouter.GET("/v1/dashboard/billing/usage", controller.GetUsage) | ||||
| 	} | ||||
| } | ||||
| @@ -2,11 +2,24 @@ package router | ||||
|  | ||||
| import ( | ||||
| 	"embed" | ||||
| 	"fmt" | ||||
| 	"github.com/gin-gonic/gin" | ||||
| 	"net/http" | ||||
| 	"os" | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| func SetRouter(router *gin.Engine, buildFS embed.FS, indexPage []byte) { | ||||
| 	SetApiRouter(router) | ||||
| 	SetDashboardRouter(router) | ||||
| 	SetRelayRouter(router) | ||||
| 	setWebRouter(router, buildFS, indexPage) | ||||
| 	frontendBaseUrl := os.Getenv("FRONTEND_BASE_URL") | ||||
| 	if frontendBaseUrl == "" { | ||||
| 		SetWebRouter(router, buildFS, indexPage) | ||||
| 	} else { | ||||
| 		frontendBaseUrl = strings.TrimSuffix(frontendBaseUrl, "/") | ||||
| 		router.NoRoute(func(c *gin.Context) { | ||||
| 			c.Redirect(http.StatusMovedPermanently, fmt.Sprintf("%s%s", frontendBaseUrl, c.Request.RequestURI)) | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|   | ||||
| @@ -7,9 +7,36 @@ import ( | ||||
| ) | ||||
|  | ||||
| func SetRelayRouter(router *gin.Engine) { | ||||
| 	relayRouter := router.Group("/v1") | ||||
| 	relayRouter.Use(middleware.GlobalAPIRateLimit(), middleware.TokenAuth(), middleware.Distribute()) | ||||
| 	// https://platform.openai.com/docs/api-reference/introduction | ||||
| 	modelsRouter := router.Group("/v1/models") | ||||
| 	modelsRouter.Use(middleware.TokenAuth()) | ||||
| 	{ | ||||
| 		relayRouter.Any("/*path", controller.Relay) | ||||
| 		modelsRouter.GET("/", controller.ListModels) | ||||
| 		modelsRouter.GET("/:model", controller.RetrieveModel) | ||||
| 	} | ||||
| 	relayV1Router := router.Group("/v1") | ||||
| 	relayV1Router.Use(middleware.TokenAuth(), middleware.Distribute()) | ||||
| 	{ | ||||
| 		relayV1Router.POST("/completions", controller.Relay) | ||||
| 		relayV1Router.POST("/chat/completions", controller.Relay) | ||||
| 		relayV1Router.POST("/edits", controller.RelayNotImplemented) | ||||
| 		relayV1Router.POST("/images/generations", controller.RelayNotImplemented) | ||||
| 		relayV1Router.POST("/images/edits", controller.RelayNotImplemented) | ||||
| 		relayV1Router.POST("/images/variations", controller.RelayNotImplemented) | ||||
| 		relayV1Router.POST("/embeddings", controller.Relay) | ||||
| 		relayV1Router.POST("/audio/transcriptions", controller.RelayNotImplemented) | ||||
| 		relayV1Router.POST("/audio/translations", controller.RelayNotImplemented) | ||||
| 		relayV1Router.GET("/files", controller.RelayNotImplemented) | ||||
| 		relayV1Router.POST("/files", controller.RelayNotImplemented) | ||||
| 		relayV1Router.DELETE("/files/:id", controller.RelayNotImplemented) | ||||
| 		relayV1Router.GET("/files/:id", controller.RelayNotImplemented) | ||||
| 		relayV1Router.GET("/files/:id/content", controller.RelayNotImplemented) | ||||
| 		relayV1Router.POST("/fine-tunes", controller.RelayNotImplemented) | ||||
| 		relayV1Router.GET("/fine-tunes", controller.RelayNotImplemented) | ||||
| 		relayV1Router.GET("/fine-tunes/:id", controller.RelayNotImplemented) | ||||
| 		relayV1Router.POST("/fine-tunes/:id/cancel", controller.RelayNotImplemented) | ||||
| 		relayV1Router.GET("/fine-tunes/:id/events", controller.RelayNotImplemented) | ||||
| 		relayV1Router.DELETE("/models/:model", controller.RelayNotImplemented) | ||||
| 		relayV1Router.POST("/moderations", controller.Relay) | ||||
| 	} | ||||
| } | ||||
|   | ||||
| @@ -2,18 +2,27 @@ package router | ||||
|  | ||||
| import ( | ||||
| 	"embed" | ||||
| 	"github.com/gin-contrib/gzip" | ||||
| 	"github.com/gin-contrib/static" | ||||
| 	"github.com/gin-gonic/gin" | ||||
| 	"net/http" | ||||
| 	"one-api/common" | ||||
| 	"one-api/controller" | ||||
| 	"one-api/middleware" | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| func setWebRouter(router *gin.Engine, buildFS embed.FS, indexPage []byte) { | ||||
| func SetWebRouter(router *gin.Engine, buildFS embed.FS, indexPage []byte) { | ||||
| 	router.Use(gzip.Gzip(gzip.DefaultCompression)) | ||||
| 	router.Use(middleware.GlobalWebRateLimit()) | ||||
| 	router.Use(middleware.Cache()) | ||||
| 	router.Use(static.Serve("/", common.EmbedFolder(buildFS, "web/build"))) | ||||
| 	router.NoRoute(func(c *gin.Context) { | ||||
| 		if strings.HasPrefix(c.Request.RequestURI, "/v1") { | ||||
| 			controller.RelayNotFound(c) | ||||
| 			return | ||||
| 		} | ||||
| 		c.Header("Cache-Control", "no-cache") | ||||
| 		c.Data(http.StatusOK, "text/html; charset=utf-8", indexPage) | ||||
| 	}) | ||||
| } | ||||
|   | ||||
| @@ -7,7 +7,7 @@ | ||||
|     <meta name="theme-color" content="#ffffff" /> | ||||
|     <meta | ||||
|       name="description" | ||||
|       content="Web site created using create-react-app" | ||||
|       content="OpenAI 接口聚合管理,支持多种渠道包括 Azure,可用于二次分发管理 key,仅单可执行文件,已打包好 Docker 镜像,一键部署,开箱即用" | ||||
|     /> | ||||
|     <title>One API</title> | ||||
|   </head> | ||||
|   | ||||
| @@ -9,7 +9,7 @@ import NotFound from './pages/NotFound'; | ||||
| import Setting from './pages/Setting'; | ||||
| import EditUser from './pages/User/EditUser'; | ||||
| import AddUser from './pages/User/AddUser'; | ||||
| import { API, showError, showNotice } from './helpers'; | ||||
| import { API, getLogo, getSystemName, showError, showNotice } from './helpers'; | ||||
| import PasswordResetForm from './components/PasswordResetForm'; | ||||
| import GitHubOAuth from './components/GitHubOAuth'; | ||||
| import PasswordResetConfirm from './components/PasswordResetConfirm'; | ||||
| @@ -19,7 +19,11 @@ import Channel from './pages/Channel'; | ||||
| import Token from './pages/Token'; | ||||
| import EditToken from './pages/Token/EditToken'; | ||||
| import EditChannel from './pages/Channel/EditChannel'; | ||||
| import AddChannel from './pages/Channel/AddChannel'; | ||||
| import Redemption from './pages/Redemption'; | ||||
| import EditRedemption from './pages/Redemption/EditRedemption'; | ||||
| import TopUp from './pages/TopUp'; | ||||
| import Log from './pages/Log'; | ||||
| import Chat from './pages/Chat'; | ||||
|  | ||||
| const Home = lazy(() => import('./pages/Home')); | ||||
| const About = lazy(() => import('./pages/About')); | ||||
| @@ -41,7 +45,16 @@ function App() { | ||||
|     if (success) { | ||||
|       localStorage.setItem('status', JSON.stringify(data)); | ||||
|       statusDispatch({ type: 'set', payload: data }); | ||||
|       localStorage.setItem('system_name', data.system_name); | ||||
|       localStorage.setItem('logo', data.logo); | ||||
|       localStorage.setItem('footer_html', data.footer_html); | ||||
|       localStorage.setItem('quota_per_unit', data.quota_per_unit); | ||||
|       localStorage.setItem('display_in_currency', data.display_in_currency); | ||||
|       if (data.chat_link) { | ||||
|         localStorage.setItem('chat_link', data.chat_link); | ||||
|       } else { | ||||
|         localStorage.removeItem('chat_link'); | ||||
|       } | ||||
|       if ( | ||||
|         data.version !== process.env.REACT_APP_VERSION && | ||||
|         data.version !== 'v0.0.0' && | ||||
| @@ -59,6 +72,17 @@ function App() { | ||||
|   useEffect(() => { | ||||
|     loadUser(); | ||||
|     loadStatus().then(); | ||||
|     let systemName = getSystemName(); | ||||
|     if (systemName) { | ||||
|       document.title = systemName; | ||||
|     } | ||||
|     let logo = getLogo(); | ||||
|     if (logo) { | ||||
|       let linkElement = document.querySelector("link[rel~='icon']"); | ||||
|       if (linkElement) { | ||||
|         linkElement.href = logo; | ||||
|       } | ||||
|     } | ||||
|   }, []); | ||||
|  | ||||
|   return ( | ||||
| @@ -91,7 +115,7 @@ function App() { | ||||
|         path='/channel/add' | ||||
|         element={ | ||||
|           <Suspense fallback={<Loading></Loading>}> | ||||
|             <AddChannel /> | ||||
|             <EditChannel /> | ||||
|           </Suspense> | ||||
|         } | ||||
|       /> | ||||
| @@ -119,6 +143,30 @@ function App() { | ||||
|           </Suspense> | ||||
|         } | ||||
|       /> | ||||
|       <Route | ||||
|         path='/redemption' | ||||
|         element={ | ||||
|           <PrivateRoute> | ||||
|             <Redemption /> | ||||
|           </PrivateRoute> | ||||
|         } | ||||
|       /> | ||||
|       <Route | ||||
|         path='/redemption/edit/:id' | ||||
|         element={ | ||||
|           <Suspense fallback={<Loading></Loading>}> | ||||
|             <EditRedemption /> | ||||
|           </Suspense> | ||||
|         } | ||||
|       /> | ||||
|       <Route | ||||
|         path='/redemption/add' | ||||
|         element={ | ||||
|           <Suspense fallback={<Loading></Loading>}> | ||||
|             <EditRedemption /> | ||||
|           </Suspense> | ||||
|         } | ||||
|       /> | ||||
|       <Route | ||||
|         path='/user' | ||||
|         element={ | ||||
| @@ -201,6 +249,24 @@ function App() { | ||||
|           </PrivateRoute> | ||||
|         } | ||||
|       /> | ||||
|       <Route | ||||
|         path='/topup' | ||||
|         element={ | ||||
|         <PrivateRoute> | ||||
|           <Suspense fallback={<Loading></Loading>}> | ||||
|             <TopUp /> | ||||
|           </Suspense> | ||||
|         </PrivateRoute> | ||||
|         } | ||||
|       /> | ||||
|       <Route | ||||
|         path='/log' | ||||
|         element={ | ||||
|           <PrivateRoute> | ||||
|             <Log /> | ||||
|           </PrivateRoute> | ||||
|         } | ||||
|       /> | ||||
|       <Route | ||||
|         path='/about' | ||||
|         element={ | ||||
| @@ -209,6 +275,14 @@ function App() { | ||||
|           </Suspense> | ||||
|         } | ||||
|       /> | ||||
|       <Route | ||||
|         path='/chat' | ||||
|         element={ | ||||
|           <Suspense fallback={<Loading></Loading>}> | ||||
|             <Chat /> | ||||
|           </Suspense> | ||||
|         } | ||||
|       /> | ||||
|       <Route path='*' element={NotFound} /> | ||||
|     </Routes> | ||||
|   ); | ||||
|   | ||||
| @@ -1,9 +1,10 @@ | ||||
| import React, { useEffect, useState } from 'react'; | ||||
| import { Button, Form, Label, Pagination, Table } from 'semantic-ui-react'; | ||||
| import { Button, Form, Label, Pagination, Popup, Table } from 'semantic-ui-react'; | ||||
| import { Link } from 'react-router-dom'; | ||||
| import { API, copy, showError, showSuccess, timestamp2string } from '../helpers'; | ||||
| import { API, showError, showInfo, showSuccess, timestamp2string } from '../helpers'; | ||||
|  | ||||
| import { CHANNEL_OPTIONS, ITEMS_PER_PAGE } from '../constants'; | ||||
| import { renderGroup, renderNumber } from '../helpers/render'; | ||||
|  | ||||
| function renderTimestamp(timestamp) { | ||||
|   return ( | ||||
| @@ -26,12 +27,29 @@ function renderType(type) { | ||||
|   return <Label basic color={type2label[type].color}>{type2label[type].text}</Label>; | ||||
| } | ||||
|  | ||||
| function renderBalance(type, balance) { | ||||
|   switch (type) { | ||||
|     case 1: // OpenAI | ||||
|     case 8: // 自定义 | ||||
|       return <span>${balance.toFixed(2)}</span>; | ||||
|     case 5: // OpenAI-SB | ||||
|       return <span>¥{(balance / 10000).toFixed(2)}</span>; | ||||
|     case 10: // AI Proxy | ||||
|       return <span>{renderNumber(balance)}</span>; | ||||
|     case 12: // API2GPT | ||||
|       return <span>¥{balance.toFixed(2)}</span>; | ||||
|     default: | ||||
|       return <span>不支持</span>; | ||||
|   } | ||||
| } | ||||
|  | ||||
| const ChannelsTable = () => { | ||||
|   const [channels, setChannels] = useState([]); | ||||
|   const [loading, setLoading] = useState(true); | ||||
|   const [activePage, setActivePage] = useState(1); | ||||
|   const [searchKeyword, setSearchKeyword] = useState(''); | ||||
|   const [searching, setSearching] = useState(false); | ||||
|   const [updatingBalance, setUpdatingBalance] = useState(false); | ||||
|  | ||||
|   const loadChannels = async (startIdx) => { | ||||
|     const res = await API.get(`/api/channel/?p=${startIdx}`); | ||||
| @@ -60,6 +78,11 @@ const ChannelsTable = () => { | ||||
|     })(); | ||||
|   }; | ||||
|  | ||||
|   const refresh = async () => { | ||||
|     setLoading(true); | ||||
|     await loadChannels(0); | ||||
|   }; | ||||
|  | ||||
|   useEffect(() => { | ||||
|     loadChannels(0) | ||||
|       .then() | ||||
| @@ -120,6 +143,22 @@ const ChannelsTable = () => { | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   const renderResponseTime = (responseTime) => { | ||||
|     let time = responseTime / 1000; | ||||
|     time = time.toFixed(2) + ' 秒'; | ||||
|     if (responseTime === 0) { | ||||
|       return <Label basic color='grey'>未测试</Label>; | ||||
|     } else if (responseTime <= 1000) { | ||||
|       return <Label basic color='green'>{time}</Label>; | ||||
|     } else if (responseTime <= 3000) { | ||||
|       return <Label basic color='olive'>{time}</Label>; | ||||
|     } else if (responseTime <= 5000) { | ||||
|       return <Label basic color='yellow'>{time}</Label>; | ||||
|     } else { | ||||
|       return <Label basic color='red'>{time}</Label>; | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   const searchChannels = async () => { | ||||
|     if (searchKeyword === '') { | ||||
|       // if keyword is blank, load files instead. | ||||
| @@ -139,6 +178,58 @@ const ChannelsTable = () => { | ||||
|     setSearching(false); | ||||
|   }; | ||||
|  | ||||
|   const testChannel = async (id, name, idx) => { | ||||
|     const res = await API.get(`/api/channel/test/${id}/`); | ||||
|     const { success, message, time } = res.data; | ||||
|     if (success) { | ||||
|       let newChannels = [...channels]; | ||||
|       let realIdx = (activePage - 1) * ITEMS_PER_PAGE + idx; | ||||
|       newChannels[realIdx].response_time = time * 1000; | ||||
|       newChannels[realIdx].test_time = Date.now() / 1000; | ||||
|       setChannels(newChannels); | ||||
|       showInfo(`通道 ${name} 测试成功,耗时 ${time.toFixed(2)} 秒。`); | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   const testAllChannels = async () => { | ||||
|     const res = await API.get(`/api/channel/test`); | ||||
|     const { success, message } = res.data; | ||||
|     if (success) { | ||||
|       showInfo('已成功开始测试所有已启用通道,请刷新页面查看结果。'); | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   const updateChannelBalance = async (id, name, idx) => { | ||||
|     const res = await API.get(`/api/channel/update_balance/${id}/`); | ||||
|     const { success, message, balance } = res.data; | ||||
|     if (success) { | ||||
|       let newChannels = [...channels]; | ||||
|       let realIdx = (activePage - 1) * ITEMS_PER_PAGE + idx; | ||||
|       newChannels[realIdx].balance = balance; | ||||
|       newChannels[realIdx].balance_updated_time = Date.now() / 1000; | ||||
|       setChannels(newChannels); | ||||
|       showInfo(`通道 ${name} 余额更新成功!`); | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   const updateAllChannelsBalance = async () => { | ||||
|     setUpdatingBalance(true); | ||||
|     const res = await API.get(`/api/channel/update_balance`); | ||||
|     const { success, message } = res.data; | ||||
|     if (success) { | ||||
|       showInfo('已更新完毕所有已启用通道余额!'); | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
|     setUpdatingBalance(false); | ||||
|   }; | ||||
|  | ||||
|   const handleKeywordChange = async (e, { value }) => { | ||||
|     setSearchKeyword(value.trim()); | ||||
|   }; | ||||
| @@ -171,7 +262,7 @@ const ChannelsTable = () => { | ||||
|         /> | ||||
|       </Form> | ||||
|  | ||||
|       <Table basic> | ||||
|       <Table basic compact size='small'> | ||||
|         <Table.Header> | ||||
|           <Table.Row> | ||||
|             <Table.HeaderCell | ||||
| @@ -190,6 +281,14 @@ const ChannelsTable = () => { | ||||
|             > | ||||
|               名称 | ||||
|             </Table.HeaderCell> | ||||
|             <Table.HeaderCell | ||||
|               style={{ cursor: 'pointer' }} | ||||
|               onClick={() => { | ||||
|                 sortChannel('group'); | ||||
|               }} | ||||
|             > | ||||
|               分组 | ||||
|             </Table.HeaderCell> | ||||
|             <Table.HeaderCell | ||||
|               style={{ cursor: 'pointer' }} | ||||
|               onClick={() => { | ||||
| @@ -209,18 +308,18 @@ const ChannelsTable = () => { | ||||
|             <Table.HeaderCell | ||||
|               style={{ cursor: 'pointer' }} | ||||
|               onClick={() => { | ||||
|                 sortChannel('created_time'); | ||||
|                 sortChannel('response_time'); | ||||
|               }} | ||||
|             > | ||||
|               创建时间 | ||||
|               响应时间 | ||||
|             </Table.HeaderCell> | ||||
|             <Table.HeaderCell | ||||
|               style={{ cursor: 'pointer' }} | ||||
|               onClick={() => { | ||||
|                 sortChannel('accessed_time'); | ||||
|                 sortChannel('balance'); | ||||
|               }} | ||||
|             > | ||||
|               访问时间 | ||||
|               余额 | ||||
|             </Table.HeaderCell> | ||||
|             <Table.HeaderCell>操作</Table.HeaderCell> | ||||
|           </Table.Row> | ||||
| @@ -238,21 +337,65 @@ const ChannelsTable = () => { | ||||
|                 <Table.Row key={channel.id}> | ||||
|                   <Table.Cell>{channel.id}</Table.Cell> | ||||
|                   <Table.Cell>{channel.name ? channel.name : '无'}</Table.Cell> | ||||
|                   <Table.Cell>{renderGroup(channel.group)}</Table.Cell> | ||||
|                   <Table.Cell>{renderType(channel.type)}</Table.Cell> | ||||
|                   <Table.Cell>{renderStatus(channel.status)}</Table.Cell> | ||||
|                   <Table.Cell>{renderTimestamp(channel.created_time)}</Table.Cell> | ||||
|                   <Table.Cell>{renderTimestamp(channel.accessed_time)}</Table.Cell> | ||||
|                   <Table.Cell> | ||||
|                     <Popup | ||||
|                       content={channel.test_time ? renderTimestamp(channel.test_time) : '未测试'} | ||||
|                       key={channel.id} | ||||
|                       trigger={renderResponseTime(channel.response_time)} | ||||
|                       basic | ||||
|                     /> | ||||
|                   </Table.Cell> | ||||
|                   <Table.Cell> | ||||
|                     <Popup | ||||
|                       content={channel.balance_updated_time ? renderTimestamp(channel.balance_updated_time) : '未更新'} | ||||
|                       key={channel.id} | ||||
|                       trigger={renderBalance(channel.type, channel.balance)} | ||||
|                       basic | ||||
|                     /> | ||||
|                   </Table.Cell> | ||||
|                   <Table.Cell> | ||||
|                     <div> | ||||
|                       <Button | ||||
|                         size={'small'} | ||||
|                         negative | ||||
|                         positive | ||||
|                         onClick={() => { | ||||
|                           manageChannel(channel.id, 'delete', idx); | ||||
|                           testChannel(channel.id, channel.name, idx); | ||||
|                         }} | ||||
|                       > | ||||
|                         删除 | ||||
|                         测试 | ||||
|                       </Button> | ||||
|                       <Button | ||||
|                         size={'small'} | ||||
|                         positive | ||||
|                         loading={updatingBalance} | ||||
|                         onClick={() => { | ||||
|                           updateChannelBalance(channel.id, channel.name, idx); | ||||
|                         }} | ||||
|                       > | ||||
|                         更新余额 | ||||
|                       </Button> | ||||
|                       <Popup | ||||
|                         trigger={ | ||||
|                           <Button size='small' negative> | ||||
|                             删除 | ||||
|                           </Button> | ||||
|                         } | ||||
|                         on='click' | ||||
|                         flowing | ||||
|                         hoverable | ||||
|                       > | ||||
|                         <Button | ||||
|                           negative | ||||
|                           onClick={() => { | ||||
|                             manageChannel(channel.id, 'delete', idx); | ||||
|                           }} | ||||
|                         > | ||||
|                           删除渠道 {channel.name} | ||||
|                         </Button> | ||||
|                       </Popup> | ||||
|                       <Button | ||||
|                         size={'small'} | ||||
|                         onClick={() => { | ||||
| @@ -281,10 +424,15 @@ const ChannelsTable = () => { | ||||
|  | ||||
|         <Table.Footer> | ||||
|           <Table.Row> | ||||
|             <Table.HeaderCell colSpan='7'> | ||||
|             <Table.HeaderCell colSpan='8'> | ||||
|               <Button size='small' as={Link} to='/channel/add' loading={loading}> | ||||
|                 添加新的渠道 | ||||
|               </Button> | ||||
|               <Button size='small' loading={loading} onClick={testAllChannels}> | ||||
|                 测试所有已启用通道 | ||||
|               </Button> | ||||
|               <Button size='small' onClick={updateAllChannelsBalance} | ||||
|                       loading={loading || updatingBalance}>更新所有已启用通道余额</Button> | ||||
|               <Pagination | ||||
|                 floated='right' | ||||
|                 activePage={activePage} | ||||
| @@ -296,6 +444,7 @@ const ChannelsTable = () => { | ||||
|                   (channels.length % ITEMS_PER_PAGE === 0 ? 1 : 0) | ||||
|                 } | ||||
|               /> | ||||
|               <Button size='small' onClick={refresh} loading={loading}>刷新</Button> | ||||
|             </Table.HeaderCell> | ||||
|           </Table.Row> | ||||
|         </Table.Footer> | ||||
|   | ||||
| @@ -1,40 +1,57 @@ | ||||
| import React, { useEffect, useState } from 'react'; | ||||
|  | ||||
| import { Container, Segment } from 'semantic-ui-react'; | ||||
| import { getFooterHTML, getSystemName } from '../helpers'; | ||||
|  | ||||
| const Footer = () => { | ||||
|   const [Footer, setFooter] = useState(''); | ||||
|   const systemName = getSystemName(); | ||||
|   const [footer, setFooter] = useState(getFooterHTML()); | ||||
|   let remainCheckTimes = 5; | ||||
|  | ||||
|   const loadFooter = () => { | ||||
|     let footer_html = localStorage.getItem('footer_html'); | ||||
|     if (footer_html) { | ||||
|       setFooter(footer_html); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   useEffect(() => { | ||||
|     let savedFooter = localStorage.getItem('footer_html'); | ||||
|     if (!savedFooter) savedFooter = ''; | ||||
|     setFooter(savedFooter); | ||||
|   }); | ||||
|     const timer = setInterval(() => { | ||||
|       if (remainCheckTimes <= 0) { | ||||
|         clearInterval(timer); | ||||
|         return; | ||||
|       } | ||||
|       remainCheckTimes--; | ||||
|       loadFooter(); | ||||
|     }, 200); | ||||
|     return () => clearTimeout(timer); | ||||
|   }, []); | ||||
|  | ||||
|   return ( | ||||
|     <Segment vertical> | ||||
|       <Container textAlign="center"> | ||||
|         {Footer === '' ? ( | ||||
|           <div className="custom-footer"> | ||||
|       <Container textAlign='center'> | ||||
|         {footer ? ( | ||||
|           <div | ||||
|             className='custom-footer' | ||||
|             dangerouslySetInnerHTML={{ __html: footer }} | ||||
|           ></div> | ||||
|         ) : ( | ||||
|           <div className='custom-footer'> | ||||
|             <a | ||||
|               href="https://github.com/songquanpeng/one-api" | ||||
|               target="_blank" | ||||
|               href='https://github.com/songquanpeng/one-api' | ||||
|               target='_blank' | ||||
|             > | ||||
|               One API {process.env.REACT_APP_VERSION}{' '} | ||||
|               {systemName} {process.env.REACT_APP_VERSION}{' '} | ||||
|             </a> | ||||
|             由{' '} | ||||
|             <a href="https://github.com/songquanpeng" target="_blank"> | ||||
|             <a href='https://github.com/songquanpeng' target='_blank'> | ||||
|               JustSong | ||||
|             </a>{' '} | ||||
|             构建,源代码遵循{' '} | ||||
|             <a href="https://opensource.org/licenses/mit-license.php"> | ||||
|             <a href='https://opensource.org/licenses/mit-license.php'> | ||||
|               MIT 协议 | ||||
|             </a> | ||||
|           </div> | ||||
|         ) : ( | ||||
|           <div | ||||
|             className="custom-footer" | ||||
|             dangerouslySetInnerHTML={{ __html: Footer }} | ||||
|           ></div> | ||||
|         )} | ||||
|       </Container> | ||||
|     </Segment> | ||||
|   | ||||
| @@ -3,50 +3,76 @@ import { Link, useNavigate } from 'react-router-dom'; | ||||
| import { UserContext } from '../context/User'; | ||||
|  | ||||
| import { Button, Container, Dropdown, Icon, Menu, Segment } from 'semantic-ui-react'; | ||||
| import { API, isAdmin, isMobile, showSuccess } from '../helpers'; | ||||
| import { API, getLogo, getSystemName, isAdmin, isMobile, showSuccess } from '../helpers'; | ||||
| import '../index.css'; | ||||
|  | ||||
| // Header Buttons | ||||
| const headerButtons = [ | ||||
| let headerButtons = [ | ||||
|   { | ||||
|     name: '首页', | ||||
|     to: '/', | ||||
|     icon: 'home', | ||||
|     icon: 'home' | ||||
|   }, | ||||
|   { | ||||
|     name: '渠道', | ||||
|     to: '/channel', | ||||
|     icon: 'sitemap', | ||||
|     admin: true, | ||||
|     admin: true | ||||
|   }, | ||||
|   { | ||||
|     name: '令牌', | ||||
|     to: '/token', | ||||
|     icon: 'key', | ||||
|     icon: 'key' | ||||
|   }, | ||||
|   { | ||||
|     name: '兑换', | ||||
|     to: '/redemption', | ||||
|     icon: 'dollar sign', | ||||
|     admin: true | ||||
|   }, | ||||
|   { | ||||
|     name: '充值', | ||||
|     to: '/topup', | ||||
|     icon: 'cart' | ||||
|   }, | ||||
|   { | ||||
|     name: '用户', | ||||
|     to: '/user', | ||||
|     icon: 'user', | ||||
|     admin: true, | ||||
|     admin: true | ||||
|   }, | ||||
|   { | ||||
|     name: '日志', | ||||
|     to: '/log', | ||||
|     icon: 'book' | ||||
|   }, | ||||
|   { | ||||
|     name: '设置', | ||||
|     to: '/setting', | ||||
|     icon: 'setting', | ||||
|     icon: 'setting' | ||||
|   }, | ||||
|   { | ||||
|     name: '关于', | ||||
|     to: '/about', | ||||
|     icon: 'info circle', | ||||
|   }, | ||||
|     icon: 'info circle' | ||||
|   } | ||||
| ]; | ||||
|  | ||||
| if (localStorage.getItem('chat_link')) { | ||||
|   headerButtons.splice(1, 0, { | ||||
|     name: '聊天', | ||||
|     to: '/chat', | ||||
|     icon: 'comments' | ||||
|   }); | ||||
| } | ||||
|  | ||||
| const Header = () => { | ||||
|   const [userState, userDispatch] = useContext(UserContext); | ||||
|   let navigate = useNavigate(); | ||||
|  | ||||
|   const [showSidebar, setShowSidebar] = useState(false); | ||||
|   const systemName = getSystemName(); | ||||
|   const logo = getLogo(); | ||||
|  | ||||
|   async function logout() { | ||||
|     setShowSidebar(false); | ||||
| @@ -94,23 +120,23 @@ const Header = () => { | ||||
|           style={ | ||||
|             showSidebar | ||||
|               ? { | ||||
|                   borderBottom: 'none', | ||||
|                   marginBottom: '0', | ||||
|                   borderTop: 'none', | ||||
|                   height: '51px', | ||||
|                 } | ||||
|                 borderBottom: 'none', | ||||
|                 marginBottom: '0', | ||||
|                 borderTop: 'none', | ||||
|                 height: '51px' | ||||
|               } | ||||
|               : { borderTop: 'none', height: '52px' } | ||||
|           } | ||||
|         > | ||||
|           <Container> | ||||
|             <Menu.Item as={Link} to='/'> | ||||
|               <img | ||||
|                 src='/logo.png' | ||||
|                 src={logo} | ||||
|                 alt='logo' | ||||
|                 style={{ marginRight: '0.75em' }} | ||||
|               /> | ||||
|               <div style={{ fontSize: '20px' }}> | ||||
|                 <b>One API</b> | ||||
|                 <b>{systemName}</b> | ||||
|               </div> | ||||
|             </Menu.Item> | ||||
|             <Menu.Menu position='right'> | ||||
| @@ -162,9 +188,9 @@ const Header = () => { | ||||
|       <Menu borderless style={{ borderTop: 'none' }}> | ||||
|         <Container> | ||||
|           <Menu.Item as={Link} to='/' className={'hide-on-mobile'}> | ||||
|             <img src='/logo.png' alt='logo' style={{ marginRight: '0.75em' }} /> | ||||
|             <img src={logo} alt='logo' style={{ marginRight: '0.75em' }} /> | ||||
|             <div style={{ fontSize: '20px' }}> | ||||
|               <b>One API</b> | ||||
|               <b>{systemName}</b> | ||||
|             </div> | ||||
|           </Menu.Item> | ||||
|           {renderButtons(false)} | ||||
|   | ||||
| @@ -10,9 +10,9 @@ import { | ||||
|   Modal, | ||||
|   Segment, | ||||
| } from 'semantic-ui-react'; | ||||
| import { Link, useNavigate } from 'react-router-dom'; | ||||
| import { Link, useNavigate, useSearchParams } from 'react-router-dom'; | ||||
| import { UserContext } from '../context/User'; | ||||
| import { API, showError, showSuccess } from '../helpers'; | ||||
| import { API, getLogo, showError, showSuccess } from '../helpers'; | ||||
|  | ||||
| const LoginForm = () => { | ||||
|   const [inputs, setInputs] = useState({ | ||||
| @@ -20,14 +20,19 @@ const LoginForm = () => { | ||||
|     password: '', | ||||
|     wechat_verification_code: '', | ||||
|   }); | ||||
|   const [searchParams, setSearchParams] = useSearchParams(); | ||||
|   const [submitted, setSubmitted] = useState(false); | ||||
|   const { username, password } = inputs; | ||||
|   const [userState, userDispatch] = useContext(UserContext); | ||||
|   let navigate = useNavigate(); | ||||
|  | ||||
|   const [status, setStatus] = useState({}); | ||||
|   const logo = getLogo(); | ||||
|  | ||||
|   useEffect(() => { | ||||
|     if (searchParams.get("expired")) { | ||||
|       showError('未登录或登录已过期,请重新登录!'); | ||||
|     } | ||||
|     let status = localStorage.getItem('status'); | ||||
|     if (status) { | ||||
|       status = JSON.parse(status); | ||||
| @@ -91,7 +96,7 @@ const LoginForm = () => { | ||||
|     <Grid textAlign="center" style={{ marginTop: '48px' }}> | ||||
|       <Grid.Column style={{ maxWidth: 450 }}> | ||||
|         <Header as="h2" color="" textAlign="center"> | ||||
|           <Image src="/logo.png" /> 用户登录 | ||||
|           <Image src={logo} /> 用户登录 | ||||
|         </Header> | ||||
|         <Form size="large"> | ||||
|           <Segment> | ||||
|   | ||||
							
								
								
									
										256
									
								
								web/src/components/LogsTable.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										256
									
								
								web/src/components/LogsTable.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,256 @@ | ||||
| import React, { useEffect, useState } from 'react'; | ||||
| import { Button, Label, Pagination, Select, Table } from 'semantic-ui-react'; | ||||
| import { API, isAdmin, showError, timestamp2string } from '../helpers'; | ||||
|  | ||||
| import { ITEMS_PER_PAGE } from '../constants'; | ||||
|  | ||||
| function renderTimestamp(timestamp) { | ||||
|   return ( | ||||
|     <> | ||||
|       {timestamp2string(timestamp)} | ||||
|     </> | ||||
|   ); | ||||
| } | ||||
|  | ||||
| const MODE_OPTIONS = [ | ||||
|   { key: 'all', text: '全部用户', value: 'all' }, | ||||
|   { key: 'self', text: '当前用户', value: 'self' }, | ||||
| ]; | ||||
|  | ||||
| const LOG_OPTIONS = [ | ||||
|   { key: '0', text: '全部', value: 0 }, | ||||
|   { key: '1', text: '充值', value: 1 }, | ||||
|   { key: '2', text: '消费', value: 2 }, | ||||
|   { key: '3', text: '管理', value: 3 }, | ||||
|   { key: '4', text: '系统', value: 4 } | ||||
| ]; | ||||
|  | ||||
| function renderType(type) { | ||||
|   switch (type) { | ||||
|     case 1: | ||||
|       return <Label basic color='green'> 充值 </Label>; | ||||
|     case 2: | ||||
|       return <Label basic color='olive'> 消费 </Label>; | ||||
|     case 3: | ||||
|       return <Label basic color='orange'> 管理 </Label>; | ||||
|     case 4: | ||||
|       return <Label basic color='purple'> 系统 </Label>; | ||||
|     default: | ||||
|       return <Label basic color='black'> 未知 </Label>; | ||||
|   } | ||||
| } | ||||
|  | ||||
| const LogsTable = () => { | ||||
|   const [logs, setLogs] = useState([]); | ||||
|   const [loading, setLoading] = useState(true); | ||||
|   const [activePage, setActivePage] = useState(1); | ||||
|   const [searchKeyword, setSearchKeyword] = useState(''); | ||||
|   const [searching, setSearching] = useState(false); | ||||
|   const [logType, setLogType] = useState(0); | ||||
|   const [mode, setMode] = useState('self'); // all, self | ||||
|   const showModePanel = isAdmin(); | ||||
|  | ||||
|   const loadLogs = async (startIdx) => { | ||||
|     let url = `/api/log/self/?p=${startIdx}&type=${logType}`; | ||||
|     if (mode === 'all') { | ||||
|       url = `/api/log/?p=${startIdx}&type=${logType}`; | ||||
|     } | ||||
|     const res = await API.get(url); | ||||
|     const { success, message, data } = res.data; | ||||
|     if (success) { | ||||
|       if (startIdx === 0) { | ||||
|         setLogs(data); | ||||
|       } else { | ||||
|         let newLogs = logs; | ||||
|         newLogs.push(...data); | ||||
|         setLogs(newLogs); | ||||
|       } | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
|     setLoading(false); | ||||
|   }; | ||||
|  | ||||
|   const onPaginationChange = (e, { activePage }) => { | ||||
|     (async () => { | ||||
|       if (activePage === Math.ceil(logs.length / ITEMS_PER_PAGE) + 1) { | ||||
|         // In this case we have to load more data and then append them. | ||||
|         await loadLogs(activePage - 1); | ||||
|       } | ||||
|       setActivePage(activePage); | ||||
|     })(); | ||||
|   }; | ||||
|  | ||||
|   const refresh = async () => { | ||||
|     setLoading(true); | ||||
|     await loadLogs(0); | ||||
|   }; | ||||
|  | ||||
|   useEffect(() => { | ||||
|     loadLogs(0) | ||||
|       .then() | ||||
|       .catch((reason) => { | ||||
|         showError(reason); | ||||
|       }); | ||||
|   }, []); | ||||
|  | ||||
|   useEffect(() => { | ||||
|     refresh().then(); | ||||
|   }, [mode, logType]); | ||||
|  | ||||
|   const searchLogs = async () => { | ||||
|     if (searchKeyword === '') { | ||||
|       // if keyword is blank, load files instead. | ||||
|       await loadLogs(0); | ||||
|       setActivePage(1); | ||||
|       return; | ||||
|     } | ||||
|     setSearching(true); | ||||
|     const res = await API.get(`/api/log/self/search?keyword=${searchKeyword}`); | ||||
|     const { success, message, data } = res.data; | ||||
|     if (success) { | ||||
|       setLogs(data); | ||||
|       setActivePage(1); | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
|     setSearching(false); | ||||
|   }; | ||||
|  | ||||
|   const handleKeywordChange = async (e, { value }) => { | ||||
|     setSearchKeyword(value.trim()); | ||||
|   }; | ||||
|  | ||||
|   const sortLog = (key) => { | ||||
|     if (logs.length === 0) return; | ||||
|     setLoading(true); | ||||
|     let sortedLogs = [...logs]; | ||||
|     sortedLogs.sort((a, b) => { | ||||
|       return ('' + a[key]).localeCompare(b[key]); | ||||
|     }); | ||||
|     if (sortedLogs[0].id === logs[0].id) { | ||||
|       sortedLogs.reverse(); | ||||
|     } | ||||
|     setLogs(sortedLogs); | ||||
|     setLoading(false); | ||||
|   }; | ||||
|  | ||||
|   return ( | ||||
|     <> | ||||
|       <Table basic> | ||||
|         <Table.Header> | ||||
|           <Table.Row> | ||||
|             <Table.HeaderCell | ||||
|               style={{ cursor: 'pointer' }} | ||||
|               onClick={() => { | ||||
|                 sortLog('created_time'); | ||||
|               }} | ||||
|               width={3} | ||||
|             > | ||||
|               时间 | ||||
|             </Table.HeaderCell> | ||||
|             { | ||||
|               showModePanel && ( | ||||
|                 <Table.HeaderCell | ||||
|                   style={{ cursor: 'pointer' }} | ||||
|                   onClick={() => { | ||||
|                     sortLog('user_id'); | ||||
|                   }} | ||||
|                   width={1} | ||||
|                 > | ||||
|                   用户 | ||||
|                 </Table.HeaderCell> | ||||
|               ) | ||||
|             } | ||||
|             <Table.HeaderCell | ||||
|               style={{ cursor: 'pointer' }} | ||||
|               onClick={() => { | ||||
|                 sortLog('type'); | ||||
|               }} | ||||
|               width={2} | ||||
|             > | ||||
|               类型 | ||||
|             </Table.HeaderCell> | ||||
|             <Table.HeaderCell | ||||
|               style={{ cursor: 'pointer' }} | ||||
|               onClick={() => { | ||||
|                 sortLog('content'); | ||||
|               }} | ||||
|               width={showModePanel ? 10 : 11} | ||||
|             > | ||||
|               详情 | ||||
|             </Table.HeaderCell> | ||||
|           </Table.Row> | ||||
|         </Table.Header> | ||||
|  | ||||
|         <Table.Body> | ||||
|           {logs | ||||
|             .slice( | ||||
|               (activePage - 1) * ITEMS_PER_PAGE, | ||||
|               activePage * ITEMS_PER_PAGE | ||||
|             ) | ||||
|             .map((log, idx) => { | ||||
|               if (log.deleted) return <></>; | ||||
|               return ( | ||||
|                 <Table.Row key={log.created_at}> | ||||
|                   <Table.Cell>{renderTimestamp(log.created_at)}</Table.Cell> | ||||
|                   { | ||||
|                     showModePanel && ( | ||||
|                       <Table.Cell><Label>{log.user_id}</Label></Table.Cell> | ||||
|                     ) | ||||
|                   } | ||||
|                   <Table.Cell>{renderType(log.type)}</Table.Cell> | ||||
|                   <Table.Cell>{log.content}</Table.Cell> | ||||
|                 </Table.Row> | ||||
|               ); | ||||
|             })} | ||||
|         </Table.Body> | ||||
|  | ||||
|         <Table.Footer> | ||||
|           <Table.Row> | ||||
|             <Table.HeaderCell colSpan={showModePanel ? '5' : '4'}> | ||||
|               { | ||||
|                 showModePanel && ( | ||||
|                   <Select | ||||
|                     placeholder='选择模式' | ||||
|                     options={MODE_OPTIONS} | ||||
|                     style={{ marginRight: '8px' }} | ||||
|                     name='mode' | ||||
|                     value={mode} | ||||
|                     onChange={(e, { name, value }) => { | ||||
|                       setMode(value); | ||||
|                     }} | ||||
|                   /> | ||||
|                 ) | ||||
|               } | ||||
|               <Select | ||||
|                 placeholder='选择明细分类' | ||||
|                 options={LOG_OPTIONS} | ||||
|                 style={{ marginRight: '8px' }} | ||||
|                 name='logType' | ||||
|                 value={logType} | ||||
|                 onChange={(e, { name, value }) => { | ||||
|                   setLogType(value); | ||||
|                 }} | ||||
|               /> | ||||
|               <Button size='small' onClick={refresh} loading={loading}>刷新</Button> | ||||
|               <Pagination | ||||
|                 floated='right' | ||||
|                 activePage={activePage} | ||||
|                 onPageChange={onPaginationChange} | ||||
|                 size='small' | ||||
|                 siblingRange={1} | ||||
|                 totalPages={ | ||||
|                   Math.ceil(logs.length / ITEMS_PER_PAGE) + | ||||
|                   (logs.length % ITEMS_PER_PAGE === 0 ? 1 : 0) | ||||
|                 } | ||||
|               /> | ||||
|             </Table.HeaderCell> | ||||
|           </Table.Row> | ||||
|         </Table.Footer> | ||||
|       </Table> | ||||
|     </> | ||||
|   ); | ||||
| }; | ||||
|  | ||||
| export default LogsTable; | ||||
							
								
								
									
										305
									
								
								web/src/components/OperationSetting.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										305
									
								
								web/src/components/OperationSetting.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,305 @@ | ||||
| import React, { useEffect, useState } from 'react'; | ||||
| import { Divider, Form, Grid, Header } from 'semantic-ui-react'; | ||||
| import { API, showError, verifyJSON } from '../helpers'; | ||||
|  | ||||
| const OperationSetting = () => { | ||||
|   let [inputs, setInputs] = useState({ | ||||
|     QuotaForNewUser: 0, | ||||
|     QuotaForInviter: 0, | ||||
|     QuotaForInvitee: 0, | ||||
|     QuotaRemindThreshold: 0, | ||||
|     PreConsumedQuota: 0, | ||||
|     ModelRatio: '', | ||||
|     GroupRatio: '', | ||||
|     TopUpLink: '', | ||||
|     ChatLink: '', | ||||
|     QuotaPerUnit: 0, | ||||
|     AutomaticDisableChannelEnabled: '', | ||||
|     ChannelDisableThreshold: 0, | ||||
|     LogConsumeEnabled: '', | ||||
|     DisplayInCurrencyEnabled: '' | ||||
|   }); | ||||
|   const [originInputs, setOriginInputs] = useState({}); | ||||
|   let [loading, setLoading] = useState(false); | ||||
|  | ||||
|   const getOptions = async () => { | ||||
|     const res = await API.get('/api/option/'); | ||||
|     const { success, message, data } = res.data; | ||||
|     if (success) { | ||||
|       let newInputs = {}; | ||||
|       data.forEach((item) => { | ||||
|         if (item.key === 'ModelRatio' || item.key === 'GroupRatio') { | ||||
|           item.value = JSON.stringify(JSON.parse(item.value), null, 2); | ||||
|         } | ||||
|         newInputs[item.key] = item.value; | ||||
|       }); | ||||
|       setInputs(newInputs); | ||||
|       setOriginInputs(newInputs); | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   useEffect(() => { | ||||
|     getOptions().then(); | ||||
|   }, []); | ||||
|  | ||||
|   const updateOption = async (key, value) => { | ||||
|     setLoading(true); | ||||
|     if (key.endsWith('Enabled')) { | ||||
|       value = inputs[key] === 'true' ? 'false' : 'true'; | ||||
|     } | ||||
|     const res = await API.put('/api/option/', { | ||||
|       key, | ||||
|       value | ||||
|     }); | ||||
|     const { success, message } = res.data; | ||||
|     if (success) { | ||||
|       setInputs((inputs) => ({ ...inputs, [key]: value })); | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
|     setLoading(false); | ||||
|   }; | ||||
|  | ||||
|   const handleInputChange = async (e, { name, value }) => { | ||||
|     if (name.endsWith('Enabled')) { | ||||
|       await updateOption(name, value); | ||||
|     } else { | ||||
|       setInputs((inputs) => ({ ...inputs, [name]: value })); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   const submitConfig = async (group) => { | ||||
|     switch (group) { | ||||
|       case 'monitor': | ||||
|         if (originInputs['AutomaticDisableChannelEnabled'] !== inputs.AutomaticDisableChannelEnabled) { | ||||
|           await updateOption('AutomaticDisableChannelEnabled', inputs.AutomaticDisableChannelEnabled); | ||||
|         } | ||||
|         if (originInputs['ChannelDisableThreshold'] !== inputs.ChannelDisableThreshold) { | ||||
|           await updateOption('ChannelDisableThreshold', inputs.ChannelDisableThreshold); | ||||
|         } | ||||
|         if (originInputs['QuotaRemindThreshold'] !== inputs.QuotaRemindThreshold) { | ||||
|           await updateOption('QuotaRemindThreshold', inputs.QuotaRemindThreshold); | ||||
|         } | ||||
|         break; | ||||
|       case 'ratio': | ||||
|         if (originInputs['ModelRatio'] !== inputs.ModelRatio) { | ||||
|           if (!verifyJSON(inputs.ModelRatio)) { | ||||
|             showError('模型倍率不是合法的 JSON 字符串'); | ||||
|             return; | ||||
|           } | ||||
|           await updateOption('ModelRatio', inputs.ModelRatio); | ||||
|         } | ||||
|         if (originInputs['GroupRatio'] !== inputs.GroupRatio) { | ||||
|           if (!verifyJSON(inputs.GroupRatio)) { | ||||
|             showError('分组倍率不是合法的 JSON 字符串'); | ||||
|             return; | ||||
|           } | ||||
|           await updateOption('GroupRatio', inputs.GroupRatio); | ||||
|         } | ||||
|         break; | ||||
|       case 'quota': | ||||
|         if (originInputs['QuotaForNewUser'] !== inputs.QuotaForNewUser) { | ||||
|           await updateOption('QuotaForNewUser', inputs.QuotaForNewUser); | ||||
|         } | ||||
|         if (originInputs['QuotaForInvitee'] !== inputs.QuotaForInvitee) { | ||||
|           await updateOption('QuotaForInvitee', inputs.QuotaForInvitee); | ||||
|         } | ||||
|         if (originInputs['QuotaForInviter'] !== inputs.QuotaForInviter) { | ||||
|           await updateOption('QuotaForInviter', inputs.QuotaForInviter); | ||||
|         } | ||||
|         if (originInputs['PreConsumedQuota'] !== inputs.PreConsumedQuota) { | ||||
|           await updateOption('PreConsumedQuota', inputs.PreConsumedQuota); | ||||
|         } | ||||
|         break; | ||||
|       case 'general': | ||||
|         if (originInputs['TopUpLink'] !== inputs.TopUpLink) { | ||||
|           await updateOption('TopUpLink', inputs.TopUpLink); | ||||
|         } | ||||
|         if (originInputs['ChatLink'] !== inputs.ChatLink) { | ||||
|           await updateOption('ChatLink', inputs.ChatLink); | ||||
|         } | ||||
|         if (originInputs['QuotaPerUnit'] !== inputs.QuotaPerUnit) { | ||||
|           await updateOption('QuotaPerUnit', inputs.QuotaPerUnit); | ||||
|         } | ||||
|         break; | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   return ( | ||||
|     <Grid columns={1}> | ||||
|       <Grid.Column> | ||||
|         <Form loading={loading}> | ||||
|           <Header as='h3'> | ||||
|             通用设置 | ||||
|           </Header> | ||||
|           <Form.Group widths={3}> | ||||
|             <Form.Input | ||||
|               label='充值链接' | ||||
|               name='TopUpLink' | ||||
|               onChange={handleInputChange} | ||||
|               autoComplete='new-password' | ||||
|               value={inputs.TopUpLink} | ||||
|               type='link' | ||||
|               placeholder='例如发卡网站的购买链接' | ||||
|             /> | ||||
|             <Form.Input | ||||
|               label='聊天页面链接' | ||||
|               name='ChatLink' | ||||
|               onChange={handleInputChange} | ||||
|               autoComplete='new-password' | ||||
|               value={inputs.ChatLink} | ||||
|               type='link' | ||||
|               placeholder='例如 ChatGPT Next Web 的部署地址' | ||||
|             /> | ||||
|             <Form.Input | ||||
|               label='单位美元额度' | ||||
|               name='QuotaPerUnit' | ||||
|               onChange={handleInputChange} | ||||
|               autoComplete='new-password' | ||||
|               value={inputs.QuotaPerUnit} | ||||
|               type='number' | ||||
|               step='0.01' | ||||
|               placeholder='一单位货币能兑换的额度' | ||||
|             /> | ||||
|           </Form.Group> | ||||
|           <Form.Group inline> | ||||
|             <Form.Checkbox | ||||
|               checked={inputs.LogConsumeEnabled === 'true'} | ||||
|               label='启用额度消费日志记录' | ||||
|               name='LogConsumeEnabled' | ||||
|               onChange={handleInputChange} | ||||
|             /> | ||||
|             <Form.Checkbox | ||||
|               checked={inputs.DisplayInCurrencyEnabled === 'true'} | ||||
|               label='以货币形式显示额度' | ||||
|               name='DisplayInCurrencyEnabled' | ||||
|               onChange={handleInputChange} | ||||
|             /> | ||||
|           </Form.Group> | ||||
|           <Form.Button onClick={() => { | ||||
|             submitConfig('general').then(); | ||||
|           }}>保存通用设置</Form.Button> | ||||
|           <Divider /> | ||||
|           <Header as='h3'> | ||||
|             监控设置 | ||||
|           </Header> | ||||
|           <Form.Group widths={3}> | ||||
|             <Form.Input | ||||
|               label='最长响应时间' | ||||
|               name='ChannelDisableThreshold' | ||||
|               onChange={handleInputChange} | ||||
|               autoComplete='new-password' | ||||
|               value={inputs.ChannelDisableThreshold} | ||||
|               type='number' | ||||
|               min='0' | ||||
|               placeholder='单位秒,当运行通道全部测试时,超过此时间将自动禁用通道' | ||||
|             /> | ||||
|             <Form.Input | ||||
|               label='额度提醒阈值' | ||||
|               name='QuotaRemindThreshold' | ||||
|               onChange={handleInputChange} | ||||
|               autoComplete='new-password' | ||||
|               value={inputs.QuotaRemindThreshold} | ||||
|               type='number' | ||||
|               min='0' | ||||
|               placeholder='低于此额度时将发送邮件提醒用户' | ||||
|             /> | ||||
|           </Form.Group> | ||||
|           <Form.Group inline> | ||||
|             <Form.Checkbox | ||||
|               checked={inputs.AutomaticDisableChannelEnabled === 'true'} | ||||
|               label='失败时自动禁用通道' | ||||
|               name='AutomaticDisableChannelEnabled' | ||||
|               onChange={handleInputChange} | ||||
|             /> | ||||
|           </Form.Group> | ||||
|           <Form.Button onClick={() => { | ||||
|             submitConfig('monitor').then(); | ||||
|           }}>保存监控设置</Form.Button> | ||||
|           <Divider /> | ||||
|           <Header as='h3'> | ||||
|             额度设置 | ||||
|           </Header> | ||||
|           <Form.Group widths={4}> | ||||
|             <Form.Input | ||||
|               label='新用户初始额度' | ||||
|               name='QuotaForNewUser' | ||||
|               onChange={handleInputChange} | ||||
|               autoComplete='new-password' | ||||
|               value={inputs.QuotaForNewUser} | ||||
|               type='number' | ||||
|               min='0' | ||||
|               placeholder='例如:100' | ||||
|             /> | ||||
|             <Form.Input | ||||
|               label='请求预扣费额度' | ||||
|               name='PreConsumedQuota' | ||||
|               onChange={handleInputChange} | ||||
|               autoComplete='new-password' | ||||
|               value={inputs.PreConsumedQuota} | ||||
|               type='number' | ||||
|               min='0' | ||||
|               placeholder='请求结束后多退少补' | ||||
|             /> | ||||
|             <Form.Input | ||||
|               label='邀请新用户奖励额度' | ||||
|               name='QuotaForInviter' | ||||
|               onChange={handleInputChange} | ||||
|               autoComplete='new-password' | ||||
|               value={inputs.QuotaForInviter} | ||||
|               type='number' | ||||
|               min='0' | ||||
|               placeholder='例如:2000' | ||||
|             /> | ||||
|             <Form.Input | ||||
|               label='新用户使用邀请码奖励额度' | ||||
|               name='QuotaForInvitee' | ||||
|               onChange={handleInputChange} | ||||
|               autoComplete='new-password' | ||||
|               value={inputs.QuotaForInvitee} | ||||
|               type='number' | ||||
|               min='0' | ||||
|               placeholder='例如:1000' | ||||
|             /> | ||||
|           </Form.Group> | ||||
|           <Form.Button onClick={() => { | ||||
|             submitConfig('quota').then(); | ||||
|           }}>保存额度设置</Form.Button> | ||||
|           <Divider /> | ||||
|           <Header as='h3'> | ||||
|             倍率设置 | ||||
|           </Header> | ||||
|           <Form.Group widths='equal'> | ||||
|             <Form.TextArea | ||||
|               label='模型倍率' | ||||
|               name='ModelRatio' | ||||
|               onChange={handleInputChange} | ||||
|               style={{ minHeight: 250, fontFamily: 'JetBrains Mono, Consolas' }} | ||||
|               autoComplete='new-password' | ||||
|               value={inputs.ModelRatio} | ||||
|               placeholder='为一个 JSON 文本,键为模型名称,值为倍率' | ||||
|             /> | ||||
|           </Form.Group> | ||||
|           <Form.Group widths='equal'> | ||||
|             <Form.TextArea | ||||
|               label='分组倍率' | ||||
|               name='GroupRatio' | ||||
|               onChange={handleInputChange} | ||||
|               style={{ minHeight: 250, fontFamily: 'JetBrains Mono, Consolas' }} | ||||
|               autoComplete='new-password' | ||||
|               value={inputs.GroupRatio} | ||||
|               placeholder='为一个 JSON 文本,键为分组名称,值为倍率' | ||||
|             /> | ||||
|           </Form.Group> | ||||
|           <Form.Button onClick={() => { | ||||
|             submitConfig('ratio').then(); | ||||
|           }}>保存倍率设置</Form.Button> | ||||
|         </Form> | ||||
|       </Grid.Column> | ||||
|     </Grid> | ||||
|   ); | ||||
| }; | ||||
|  | ||||
| export default OperationSetting; | ||||
| @@ -1,5 +1,5 @@ | ||||
| import React, { useEffect, useState } from 'react'; | ||||
| import { Button, Divider, Form, Grid, Header, Modal } from 'semantic-ui-react'; | ||||
| import { Button, Divider, Form, Grid, Header, Message, Modal } from 'semantic-ui-react'; | ||||
| import { API, showError, showSuccess } from '../helpers'; | ||||
| import { marked } from 'marked'; | ||||
|  | ||||
| @@ -8,17 +8,19 @@ const OtherSetting = () => { | ||||
|     Footer: '', | ||||
|     Notice: '', | ||||
|     About: '', | ||||
|     SystemName: '', | ||||
|     Logo: '', | ||||
|     HomePageContent: '' | ||||
|   }); | ||||
|   let originInputs = {}; | ||||
|   let [loading, setLoading] = useState(false); | ||||
|   const [showUpdateModal, setShowUpdateModal] = useState(false); | ||||
|   const [updateData, setUpdateData] = useState({ | ||||
|     tag_name: '', | ||||
|     content: '', | ||||
|     content: '' | ||||
|   }); | ||||
|  | ||||
|   const getOptions = async () => { | ||||
|     const res = await API.get('/api/option'); | ||||
|     const res = await API.get('/api/option/'); | ||||
|     const { success, message, data } = res.data; | ||||
|     if (success) { | ||||
|       let newInputs = {}; | ||||
| @@ -28,7 +30,6 @@ const OtherSetting = () => { | ||||
|         } | ||||
|       }); | ||||
|       setInputs(newInputs); | ||||
|       originInputs = newInputs; | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
| @@ -40,9 +41,9 @@ const OtherSetting = () => { | ||||
|  | ||||
|   const updateOption = async (key, value) => { | ||||
|     setLoading(true); | ||||
|     const res = await API.put('/api/option', { | ||||
|     const res = await API.put('/api/option/', { | ||||
|       key, | ||||
|       value, | ||||
|       value | ||||
|     }); | ||||
|     const { success, message } = res.data; | ||||
|     if (success) { | ||||
| @@ -65,10 +66,22 @@ const OtherSetting = () => { | ||||
|     await updateOption('Footer', inputs.Footer); | ||||
|   }; | ||||
|  | ||||
|   const submitSystemName = async () => { | ||||
|     await updateOption('SystemName', inputs.SystemName); | ||||
|   }; | ||||
|  | ||||
|   const submitLogo = async () => { | ||||
|     await updateOption('Logo', inputs.Logo); | ||||
|   }; | ||||
|  | ||||
|   const submitAbout = async () => { | ||||
|     await updateOption('About', inputs.About); | ||||
|   }; | ||||
|  | ||||
|   const submitOption = async (key) => { | ||||
|     await updateOption(key, inputs[key]); | ||||
|   }; | ||||
|  | ||||
|   const openGitHubRelease = () => { | ||||
|     window.location = | ||||
|       'https://github.com/songquanpeng/one-api/releases/latest'; | ||||
| @@ -84,7 +97,7 @@ const OtherSetting = () => { | ||||
|     } else { | ||||
|       setUpdateData({ | ||||
|         tag_name: tag_name, | ||||
|         content: marked.parse(body), | ||||
|         content: marked.parse(body) | ||||
|       }); | ||||
|       setShowUpdateModal(true); | ||||
|     } | ||||
| @@ -109,10 +122,42 @@ const OtherSetting = () => { | ||||
|           <Form.Button onClick={submitNotice}>保存公告</Form.Button> | ||||
|           <Divider /> | ||||
|           <Header as='h3'>个性化设置</Header> | ||||
|           <Form.Group widths='equal'> | ||||
|             <Form.Input | ||||
|               label='系统名称' | ||||
|               placeholder='在此输入系统名称' | ||||
|               value={inputs.SystemName} | ||||
|               name='SystemName' | ||||
|               onChange={handleInputChange} | ||||
|             /> | ||||
|           </Form.Group> | ||||
|           <Form.Button onClick={submitSystemName}>设置系统名称</Form.Button> | ||||
|           <Form.Group widths='equal'> | ||||
|             <Form.Input | ||||
|               label='Logo 图片地址' | ||||
|               placeholder='在此输入 Logo 图片地址' | ||||
|               value={inputs.Logo} | ||||
|               name='Logo' | ||||
|               type='url' | ||||
|               onChange={handleInputChange} | ||||
|             /> | ||||
|           </Form.Group> | ||||
|           <Form.Button onClick={submitLogo}>设置 Logo</Form.Button> | ||||
|           <Form.Group widths='equal'> | ||||
|             <Form.TextArea | ||||
|               label='首页内容' | ||||
|               placeholder='在此输入首页内容,支持 Markdown & HTML 代码,设置后首页的状态信息将不再显示。如果输入的是一个链接,则会使用该链接作为 iframe 的 src 属性,这允许你设置任意网页作为首页。' | ||||
|               value={inputs.HomePageContent} | ||||
|               name='HomePageContent' | ||||
|               onChange={handleInputChange} | ||||
|               style={{ minHeight: 150, fontFamily: 'JetBrains Mono, Consolas' }} | ||||
|             /> | ||||
|           </Form.Group> | ||||
|           <Form.Button onClick={() => submitOption('HomePageContent')}>保存首页内容</Form.Button> | ||||
|           <Form.Group widths='equal'> | ||||
|             <Form.TextArea | ||||
|               label='关于' | ||||
|               placeholder='在此输入新的关于内容,支持 Markdown & HTML 代码' | ||||
|               placeholder='在此输入新的关于内容,支持 Markdown & HTML 代码。如果输入的是一个链接,则会使用该链接作为 iframe 的 src 属性,这允许你设置任意网页作为关于页面。' | ||||
|               value={inputs.About} | ||||
|               name='About' | ||||
|               onChange={handleInputChange} | ||||
| @@ -120,6 +165,7 @@ const OtherSetting = () => { | ||||
|             /> | ||||
|           </Form.Group> | ||||
|           <Form.Button onClick={submitAbout}>保存关于</Form.Button> | ||||
|           <Message>移除 One API 的版权标识必须首先获得授权,项目维护需要花费大量精力,如果本项目对你有意义,请主动支持本项目。</Message> | ||||
|           <Form.Group widths='equal'> | ||||
|             <Form.Input | ||||
|               label='页脚' | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| import React, { useEffect, useState } from 'react'; | ||||
| import { Button, Divider, Form, Header, Image, Modal } from 'semantic-ui-react'; | ||||
| import { Button, Divider, Form, Header, Image, Message, Modal } from 'semantic-ui-react'; | ||||
| import { Link } from 'react-router-dom'; | ||||
| import { API, copy, showError, showInfo, showSuccess } from '../helpers'; | ||||
| import { API, copy, showError, showInfo, showNotice, showSuccess } from '../helpers'; | ||||
| import Turnstile from 'react-turnstile'; | ||||
|  | ||||
| const PersonalSetting = () => { | ||||
| @@ -34,6 +34,29 @@ const PersonalSetting = () => { | ||||
|     setInputs((inputs) => ({ ...inputs, [name]: value })); | ||||
|   }; | ||||
|  | ||||
|   const generateAccessToken = async () => { | ||||
|     const res = await API.get('/api/user/token'); | ||||
|     const { success, message, data } = res.data; | ||||
|     if (success) { | ||||
|       await copy(data); | ||||
|       showSuccess(`令牌已重置并已复制到剪贴板:${data}`); | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   const getAffLink = async () => { | ||||
|     const res = await API.get('/api/user/aff'); | ||||
|     const { success, message, data } = res.data; | ||||
|     if (success) { | ||||
|       let link = `${window.location.origin}/register?aff=${data}`; | ||||
|       await copy(link); | ||||
|       showNotice(`邀请链接已复制到剪切板:${link}`); | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   const bindWeChat = async () => { | ||||
|     if (inputs.wechat_verification_code === '') return; | ||||
|     const res = await API.get( | ||||
| @@ -92,18 +115,27 @@ const PersonalSetting = () => { | ||||
|   return ( | ||||
|     <div style={{ lineHeight: '40px' }}> | ||||
|       <Header as='h3'>通用设置</Header> | ||||
|       <Message> | ||||
|         注意,此处生成的令牌用于系统管理,而非用于请求 OpenAI 相关的服务,请知悉。 | ||||
|       </Message> | ||||
|       <Button as={Link} to={`/user/edit/`}> | ||||
|         更新个人信息 | ||||
|       </Button> | ||||
|       <Button onClick={generateAccessToken}>生成系统访问令牌</Button> | ||||
|       <Button onClick={getAffLink}>复制邀请链接</Button> | ||||
|       <Divider /> | ||||
|       <Header as='h3'>账号绑定</Header> | ||||
|       <Button | ||||
|         onClick={() => { | ||||
|           setShowWeChatBindModal(true); | ||||
|         }} | ||||
|       > | ||||
|         绑定微信账号 | ||||
|       </Button> | ||||
|       { | ||||
|         status.wechat_login && ( | ||||
|           <Button | ||||
|             onClick={() => { | ||||
|               setShowWeChatBindModal(true); | ||||
|             }} | ||||
|           > | ||||
|             绑定微信账号 | ||||
|           </Button> | ||||
|         ) | ||||
|       } | ||||
|       <Modal | ||||
|         onClose={() => setShowWeChatBindModal(false)} | ||||
|         onOpen={() => setShowWeChatBindModal(true)} | ||||
| @@ -133,7 +165,11 @@ const PersonalSetting = () => { | ||||
|           </Modal.Description> | ||||
|         </Modal.Content> | ||||
|       </Modal> | ||||
|       <Button onClick={openGitHubOAuth}>绑定 GitHub 账号</Button> | ||||
|       { | ||||
|         status.github_oauth && ( | ||||
|           <Button onClick={openGitHubOAuth}>绑定 GitHub 账号</Button> | ||||
|         ) | ||||
|       } | ||||
|       <Button | ||||
|         onClick={() => { | ||||
|           setShowEmailBindModal(true); | ||||
|   | ||||
							
								
								
									
										304
									
								
								web/src/components/RedemptionsTable.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										304
									
								
								web/src/components/RedemptionsTable.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,304 @@ | ||||
| import React, { useEffect, useState } from 'react'; | ||||
| import { Button, Form, Label, Message, Pagination, Table } from 'semantic-ui-react'; | ||||
| import { Link } from 'react-router-dom'; | ||||
| import { API, copy, showError, showInfo, showSuccess, showWarning, timestamp2string } from '../helpers'; | ||||
|  | ||||
| import { ITEMS_PER_PAGE } from '../constants'; | ||||
| import { renderQuota } from '../helpers/render'; | ||||
|  | ||||
| function renderTimestamp(timestamp) { | ||||
|   return ( | ||||
|     <> | ||||
|       {timestamp2string(timestamp)} | ||||
|     </> | ||||
|   ); | ||||
| } | ||||
|  | ||||
| function renderStatus(status) { | ||||
|   switch (status) { | ||||
|     case 1: | ||||
|       return <Label basic color='green'>未使用</Label>; | ||||
|     case 2: | ||||
|       return <Label basic color='red'> 已禁用 </Label>; | ||||
|     case 3: | ||||
|       return <Label basic color='grey'> 已使用 </Label>; | ||||
|     default: | ||||
|       return <Label basic color='black'> 未知状态 </Label>; | ||||
|   } | ||||
| } | ||||
|  | ||||
| const RedemptionsTable = () => { | ||||
|   const [redemptions, setRedemptions] = useState([]); | ||||
|   const [loading, setLoading] = useState(true); | ||||
|   const [activePage, setActivePage] = useState(1); | ||||
|   const [searchKeyword, setSearchKeyword] = useState(''); | ||||
|   const [searching, setSearching] = useState(false); | ||||
|  | ||||
|   const loadRedemptions = async (startIdx) => { | ||||
|     const res = await API.get(`/api/redemption/?p=${startIdx}`); | ||||
|     const { success, message, data } = res.data; | ||||
|     if (success) { | ||||
|       if (startIdx === 0) { | ||||
|         setRedemptions(data); | ||||
|       } else { | ||||
|         let newRedemptions = redemptions; | ||||
|         newRedemptions.push(...data); | ||||
|         setRedemptions(newRedemptions); | ||||
|       } | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
|     setLoading(false); | ||||
|   }; | ||||
|  | ||||
|   const onPaginationChange = (e, { activePage }) => { | ||||
|     (async () => { | ||||
|       if (activePage === Math.ceil(redemptions.length / ITEMS_PER_PAGE) + 1) { | ||||
|         // In this case we have to load more data and then append them. | ||||
|         await loadRedemptions(activePage - 1); | ||||
|       } | ||||
|       setActivePage(activePage); | ||||
|     })(); | ||||
|   }; | ||||
|  | ||||
|   useEffect(() => { | ||||
|     loadRedemptions(0) | ||||
|       .then() | ||||
|       .catch((reason) => { | ||||
|         showError(reason); | ||||
|       }); | ||||
|   }, []); | ||||
|  | ||||
|   const manageRedemption = async (id, action, idx) => { | ||||
|     let data = { id }; | ||||
|     let res; | ||||
|     switch (action) { | ||||
|       case 'delete': | ||||
|         res = await API.delete(`/api/redemption/${id}/`); | ||||
|         break; | ||||
|       case 'enable': | ||||
|         data.status = 1; | ||||
|         res = await API.put('/api/redemption/?status_only=true', data); | ||||
|         break; | ||||
|       case 'disable': | ||||
|         data.status = 2; | ||||
|         res = await API.put('/api/redemption/?status_only=true', data); | ||||
|         break; | ||||
|     } | ||||
|     const { success, message } = res.data; | ||||
|     if (success) { | ||||
|       showSuccess('操作成功完成!'); | ||||
|       let redemption = res.data.data; | ||||
|       let newRedemptions = [...redemptions]; | ||||
|       let realIdx = (activePage - 1) * ITEMS_PER_PAGE + idx; | ||||
|       if (action === 'delete') { | ||||
|         newRedemptions[realIdx].deleted = true; | ||||
|       } else { | ||||
|         newRedemptions[realIdx].status = redemption.status; | ||||
|       } | ||||
|       setRedemptions(newRedemptions); | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   const searchRedemptions = async () => { | ||||
|     if (searchKeyword === '') { | ||||
|       // if keyword is blank, load files instead. | ||||
|       await loadRedemptions(0); | ||||
|       setActivePage(1); | ||||
|       return; | ||||
|     } | ||||
|     setSearching(true); | ||||
|     const res = await API.get(`/api/redemption/search?keyword=${searchKeyword}`); | ||||
|     const { success, message, data } = res.data; | ||||
|     if (success) { | ||||
|       setRedemptions(data); | ||||
|       setActivePage(1); | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
|     setSearching(false); | ||||
|   }; | ||||
|  | ||||
|   const handleKeywordChange = async (e, { value }) => { | ||||
|     setSearchKeyword(value.trim()); | ||||
|   }; | ||||
|  | ||||
|   const sortRedemption = (key) => { | ||||
|     if (redemptions.length === 0) return; | ||||
|     setLoading(true); | ||||
|     let sortedRedemptions = [...redemptions]; | ||||
|     sortedRedemptions.sort((a, b) => { | ||||
|       return ('' + a[key]).localeCompare(b[key]); | ||||
|     }); | ||||
|     if (sortedRedemptions[0].id === redemptions[0].id) { | ||||
|       sortedRedemptions.reverse(); | ||||
|     } | ||||
|     setRedemptions(sortedRedemptions); | ||||
|     setLoading(false); | ||||
|   }; | ||||
|  | ||||
|   return ( | ||||
|     <> | ||||
|       <Form onSubmit={searchRedemptions}> | ||||
|         <Form.Input | ||||
|           icon='search' | ||||
|           fluid | ||||
|           iconPosition='left' | ||||
|           placeholder='搜索兑换码的 ID 和名称 ...' | ||||
|           value={searchKeyword} | ||||
|           loading={searching} | ||||
|           onChange={handleKeywordChange} | ||||
|         /> | ||||
|       </Form> | ||||
|  | ||||
|       <Table basic compact size='small'> | ||||
|         <Table.Header> | ||||
|           <Table.Row> | ||||
|             <Table.HeaderCell | ||||
|               style={{ cursor: 'pointer' }} | ||||
|               onClick={() => { | ||||
|                 sortRedemption('id'); | ||||
|               }} | ||||
|             > | ||||
|               ID | ||||
|             </Table.HeaderCell> | ||||
|             <Table.HeaderCell | ||||
|               style={{ cursor: 'pointer' }} | ||||
|               onClick={() => { | ||||
|                 sortRedemption('name'); | ||||
|               }} | ||||
|             > | ||||
|               名称 | ||||
|             </Table.HeaderCell> | ||||
|             <Table.HeaderCell | ||||
|               style={{ cursor: 'pointer' }} | ||||
|               onClick={() => { | ||||
|                 sortRedemption('status'); | ||||
|               }} | ||||
|             > | ||||
|               状态 | ||||
|             </Table.HeaderCell> | ||||
|             <Table.HeaderCell | ||||
|               style={{ cursor: 'pointer' }} | ||||
|               onClick={() => { | ||||
|                 sortRedemption('quota'); | ||||
|               }} | ||||
|             > | ||||
|               额度 | ||||
|             </Table.HeaderCell> | ||||
|             <Table.HeaderCell | ||||
|               style={{ cursor: 'pointer' }} | ||||
|               onClick={() => { | ||||
|                 sortRedemption('created_time'); | ||||
|               }} | ||||
|             > | ||||
|               创建时间 | ||||
|             </Table.HeaderCell> | ||||
|             <Table.HeaderCell | ||||
|               style={{ cursor: 'pointer' }} | ||||
|               onClick={() => { | ||||
|                 sortRedemption('redeemed_time'); | ||||
|               }} | ||||
|             > | ||||
|               兑换时间 | ||||
|             </Table.HeaderCell> | ||||
|             <Table.HeaderCell>操作</Table.HeaderCell> | ||||
|           </Table.Row> | ||||
|         </Table.Header> | ||||
|  | ||||
|         <Table.Body> | ||||
|           {redemptions | ||||
|             .slice( | ||||
|               (activePage - 1) * ITEMS_PER_PAGE, | ||||
|               activePage * ITEMS_PER_PAGE | ||||
|             ) | ||||
|             .map((redemption, idx) => { | ||||
|               if (redemption.deleted) return <></>; | ||||
|               return ( | ||||
|                 <Table.Row key={redemption.id}> | ||||
|                   <Table.Cell>{redemption.id}</Table.Cell> | ||||
|                   <Table.Cell>{redemption.name ? redemption.name : '无'}</Table.Cell> | ||||
|                   <Table.Cell>{renderStatus(redemption.status)}</Table.Cell> | ||||
|                   <Table.Cell>{renderQuota(redemption.quota)}</Table.Cell> | ||||
|                   <Table.Cell>{renderTimestamp(redemption.created_time)}</Table.Cell> | ||||
|                   <Table.Cell>{redemption.redeemed_time ? renderTimestamp(redemption.redeemed_time) : "尚未兑换"} </Table.Cell> | ||||
|                   <Table.Cell> | ||||
|                     <div> | ||||
|                       <Button | ||||
|                         size={'small'} | ||||
|                         positive | ||||
|                         onClick={async () => { | ||||
|                           if (await copy(redemption.key)) { | ||||
|                             showSuccess('已复制到剪贴板!'); | ||||
|                           } else { | ||||
|                             showWarning('无法复制到剪贴板,请手动复制,已将兑换码填入搜索框。') | ||||
|                             setSearchKeyword(redemption.key); | ||||
|                           } | ||||
|                         }} | ||||
|                       > | ||||
|                         复制 | ||||
|                       </Button> | ||||
|                       <Button | ||||
|                         size={'small'} | ||||
|                         negative | ||||
|                         onClick={() => { | ||||
|                           manageRedemption(redemption.id, 'delete', idx); | ||||
|                         }} | ||||
|                       > | ||||
|                         删除 | ||||
|                       </Button> | ||||
|                       <Button | ||||
|                         size={'small'} | ||||
|                         disabled={redemption.status === 3}  // used | ||||
|                         onClick={() => { | ||||
|                           manageRedemption( | ||||
|                             redemption.id, | ||||
|                             redemption.status === 1 ? 'disable' : 'enable', | ||||
|                             idx | ||||
|                           ); | ||||
|                         }} | ||||
|                       > | ||||
|                         {redemption.status === 1 ? '禁用' : '启用'} | ||||
|                       </Button> | ||||
|                       <Button | ||||
|                         size={'small'} | ||||
|                         as={Link} | ||||
|                         to={'/redemption/edit/' + redemption.id} | ||||
|                       > | ||||
|                         编辑 | ||||
|                       </Button> | ||||
|                     </div> | ||||
|                   </Table.Cell> | ||||
|                 </Table.Row> | ||||
|               ); | ||||
|             })} | ||||
|         </Table.Body> | ||||
|  | ||||
|         <Table.Footer> | ||||
|           <Table.Row> | ||||
|             <Table.HeaderCell colSpan='8'> | ||||
|               <Button size='small' as={Link} to='/redemption/add' loading={loading}> | ||||
|                 添加新的兑换码 | ||||
|               </Button> | ||||
|               <Pagination | ||||
|                 floated='right' | ||||
|                 activePage={activePage} | ||||
|                 onPageChange={onPaginationChange} | ||||
|                 size='small' | ||||
|                 siblingRange={1} | ||||
|                 totalPages={ | ||||
|                   Math.ceil(redemptions.length / ITEMS_PER_PAGE) + | ||||
|                   (redemptions.length % ITEMS_PER_PAGE === 0 ? 1 : 0) | ||||
|                 } | ||||
|               /> | ||||
|             </Table.HeaderCell> | ||||
|           </Table.Row> | ||||
|         </Table.Footer> | ||||
|       </Table> | ||||
|     </> | ||||
|   ); | ||||
| }; | ||||
|  | ||||
| export default RedemptionsTable; | ||||
| @@ -9,7 +9,7 @@ import { | ||||
|   Segment, | ||||
| } from 'semantic-ui-react'; | ||||
| import { Link, useNavigate } from 'react-router-dom'; | ||||
| import { API, showError, showInfo, showSuccess } from '../helpers'; | ||||
| import { API, getLogo, showError, showInfo, showSuccess } from '../helpers'; | ||||
| import Turnstile from 'react-turnstile'; | ||||
|  | ||||
| const RegisterForm = () => { | ||||
| @@ -26,6 +26,11 @@ const RegisterForm = () => { | ||||
|   const [turnstileSiteKey, setTurnstileSiteKey] = useState(''); | ||||
|   const [turnstileToken, setTurnstileToken] = useState(''); | ||||
|   const [loading, setLoading] = useState(false); | ||||
|   const logo = getLogo(); | ||||
|   let affCode = new URLSearchParams(window.location.search).get('aff'); | ||||
|   if (affCode) { | ||||
|     localStorage.setItem('aff', affCode); | ||||
|   } | ||||
|  | ||||
|   useEffect(() => { | ||||
|     let status = localStorage.getItem('status'); | ||||
| @@ -62,6 +67,10 @@ const RegisterForm = () => { | ||||
|         return; | ||||
|       } | ||||
|       setLoading(true); | ||||
|       if (!affCode) { | ||||
|         affCode = localStorage.getItem('aff'); | ||||
|       } | ||||
|       inputs.aff_code = affCode; | ||||
|       const res = await API.post( | ||||
|         `/api/user/register?turnstile=${turnstileToken}`, | ||||
|         inputs | ||||
| @@ -100,7 +109,7 @@ const RegisterForm = () => { | ||||
|     <Grid textAlign='center' style={{ marginTop: '48px' }}> | ||||
|       <Grid.Column style={{ maxWidth: 450 }}> | ||||
|         <Header as='h2' color='' textAlign='center'> | ||||
|           <Image src='/logo.png' /> 新用户注册 | ||||
|           <Image src={logo} /> 新用户注册 | ||||
|         </Header> | ||||
|         <Form size='large'> | ||||
|           <Segment> | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| import React, { useEffect, useState } from 'react'; | ||||
| import { Divider, Form, Grid, Header, Message } from 'semantic-ui-react'; | ||||
| import { API, removeTrailingSlash, showError } from '../helpers'; | ||||
| import { API, removeTrailingSlash, showError, verifyJSON } from '../helpers'; | ||||
|  | ||||
| const SystemSetting = () => { | ||||
|   let [inputs, setInputs] = useState({ | ||||
| @@ -12,7 +12,9 @@ const SystemSetting = () => { | ||||
|     GitHubClientSecret: '', | ||||
|     Notice: '', | ||||
|     SMTPServer: '', | ||||
|     SMTPPort: '', | ||||
|     SMTPAccount: '', | ||||
|     SMTPFrom: '', | ||||
|     SMTPToken: '', | ||||
|     ServerAddress: '', | ||||
|     Footer: '', | ||||
| @@ -25,11 +27,11 @@ const SystemSetting = () => { | ||||
|     TurnstileSecretKey: '', | ||||
|     RegisterEnabled: '', | ||||
|   }); | ||||
|   let originInputs = {}; | ||||
|   const [originInputs, setOriginInputs] = useState({}); | ||||
|   let [loading, setLoading] = useState(false); | ||||
|  | ||||
|   const getOptions = async () => { | ||||
|     const res = await API.get('/api/option'); | ||||
|     const res = await API.get('/api/option/'); | ||||
|     const { success, message, data } = res.data; | ||||
|     if (success) { | ||||
|       let newInputs = {}; | ||||
| @@ -37,7 +39,7 @@ const SystemSetting = () => { | ||||
|         newInputs[item.key] = item.value; | ||||
|       }); | ||||
|       setInputs(newInputs); | ||||
|       originInputs = newInputs; | ||||
|       setOriginInputs(newInputs); | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
| @@ -62,9 +64,9 @@ const SystemSetting = () => { | ||||
|       default: | ||||
|         break; | ||||
|     } | ||||
|     const res = await API.put('/api/option', { | ||||
|     const res = await API.put('/api/option/', { | ||||
|       key, | ||||
|       value, | ||||
|       value | ||||
|     }); | ||||
|     const { success, message } = res.data; | ||||
|     if (success) { | ||||
| @@ -106,6 +108,15 @@ const SystemSetting = () => { | ||||
|     if (originInputs['SMTPAccount'] !== inputs.SMTPAccount) { | ||||
|       await updateOption('SMTPAccount', inputs.SMTPAccount); | ||||
|     } | ||||
|     if (originInputs['SMTPFrom'] !== inputs.SMTPFrom) { | ||||
|       await updateOption('SMTPFrom', inputs.SMTPFrom); | ||||
|     } | ||||
|     if ( | ||||
|       originInputs['SMTPPort'] !== inputs.SMTPPort && | ||||
|       inputs.SMTPPort !== '' | ||||
|     ) { | ||||
|       await updateOption('SMTPPort', inputs.SMTPPort); | ||||
|     } | ||||
|     if ( | ||||
|       originInputs['SMTPToken'] !== inputs.SMTPToken && | ||||
|       inputs.SMTPToken !== '' | ||||
| @@ -237,24 +248,42 @@ const SystemSetting = () => { | ||||
|               label='SMTP 服务器地址' | ||||
|               name='SMTPServer' | ||||
|               onChange={handleInputChange} | ||||
|               autoComplete='off' | ||||
|               autoComplete='new-password' | ||||
|               value={inputs.SMTPServer} | ||||
|               placeholder='例如:smtp.qq.com' | ||||
|             /> | ||||
|             <Form.Input | ||||
|               label='SMTP 端口' | ||||
|               name='SMTPPort' | ||||
|               onChange={handleInputChange} | ||||
|               autoComplete='new-password' | ||||
|               value={inputs.SMTPPort} | ||||
|               placeholder='默认: 587' | ||||
|             /> | ||||
|             <Form.Input | ||||
|               label='SMTP 账户' | ||||
|               name='SMTPAccount' | ||||
|               onChange={handleInputChange} | ||||
|               autoComplete='off' | ||||
|               autoComplete='new-password' | ||||
|               value={inputs.SMTPAccount} | ||||
|               placeholder='通常是邮箱地址' | ||||
|             /> | ||||
|           </Form.Group> | ||||
|           <Form.Group widths={3}> | ||||
|             <Form.Input | ||||
|               label='SMTP 发送者邮箱' | ||||
|               name='SMTPFrom' | ||||
|               onChange={handleInputChange} | ||||
|               autoComplete='new-password' | ||||
|               value={inputs.SMTPFrom} | ||||
|               placeholder='通常和邮箱地址保持一致' | ||||
|             /> | ||||
|             <Form.Input | ||||
|               label='SMTP 访问凭证' | ||||
|               name='SMTPToken' | ||||
|               onChange={handleInputChange} | ||||
|               type='password' | ||||
|               autoComplete='off' | ||||
|               autoComplete='new-password' | ||||
|               value={inputs.SMTPToken} | ||||
|               placeholder='敏感信息不会发送到前端显示' | ||||
|             /> | ||||
| @@ -281,7 +310,7 @@ const SystemSetting = () => { | ||||
|               label='GitHub Client ID' | ||||
|               name='GitHubClientId' | ||||
|               onChange={handleInputChange} | ||||
|               autoComplete='off' | ||||
|               autoComplete='new-password' | ||||
|               value={inputs.GitHubClientId} | ||||
|               placeholder='输入你注册的 GitHub OAuth APP 的 ID' | ||||
|             /> | ||||
| @@ -290,7 +319,7 @@ const SystemSetting = () => { | ||||
|               name='GitHubClientSecret' | ||||
|               onChange={handleInputChange} | ||||
|               type='password' | ||||
|               autoComplete='off' | ||||
|               autoComplete='new-password' | ||||
|               value={inputs.GitHubClientSecret} | ||||
|               placeholder='敏感信息不会发送到前端显示' | ||||
|             /> | ||||
| @@ -318,7 +347,7 @@ const SystemSetting = () => { | ||||
|               name='WeChatServerAddress' | ||||
|               placeholder='例如:https://yourdomain.com' | ||||
|               onChange={handleInputChange} | ||||
|               autoComplete='off' | ||||
|               autoComplete='new-password' | ||||
|               value={inputs.WeChatServerAddress} | ||||
|             /> | ||||
|             <Form.Input | ||||
| @@ -326,7 +355,7 @@ const SystemSetting = () => { | ||||
|               name='WeChatServerToken' | ||||
|               type='password' | ||||
|               onChange={handleInputChange} | ||||
|               autoComplete='off' | ||||
|               autoComplete='new-password' | ||||
|               value={inputs.WeChatServerToken} | ||||
|               placeholder='敏感信息不会发送到前端显示' | ||||
|             /> | ||||
| @@ -334,7 +363,7 @@ const SystemSetting = () => { | ||||
|               label='微信公众号二维码图片链接' | ||||
|               name='WeChatAccountQRCodeImageURL' | ||||
|               onChange={handleInputChange} | ||||
|               autoComplete='off' | ||||
|               autoComplete='new-password' | ||||
|               value={inputs.WeChatAccountQRCodeImageURL} | ||||
|               placeholder='输入一个图片链接' | ||||
|             /> | ||||
| @@ -358,7 +387,7 @@ const SystemSetting = () => { | ||||
|               label='Turnstile Site Key' | ||||
|               name='TurnstileSiteKey' | ||||
|               onChange={handleInputChange} | ||||
|               autoComplete='off' | ||||
|               autoComplete='new-password' | ||||
|               value={inputs.TurnstileSiteKey} | ||||
|               placeholder='输入你注册的 Turnstile Site Key' | ||||
|             /> | ||||
| @@ -367,7 +396,7 @@ const SystemSetting = () => { | ||||
|               name='TurnstileSecretKey' | ||||
|               onChange={handleInputChange} | ||||
|               type='password' | ||||
|               autoComplete='off' | ||||
|               autoComplete='new-password' | ||||
|               value={inputs.TurnstileSecretKey} | ||||
|               placeholder='敏感信息不会发送到前端显示' | ||||
|             /> | ||||
|   | ||||
| @@ -1,9 +1,10 @@ | ||||
| import React, { useEffect, useState } from 'react'; | ||||
| import { Button, Form, Label, Message, Pagination, Table } from 'semantic-ui-react'; | ||||
| import { Button, Form, Label, Modal, Pagination, Popup, Table } from 'semantic-ui-react'; | ||||
| import { Link } from 'react-router-dom'; | ||||
| import { API, copy, showError, showInfo, showSuccess, showWarning, timestamp2string } from '../helpers'; | ||||
| import { API, copy, showError, showSuccess, showWarning, timestamp2string } from '../helpers'; | ||||
|  | ||||
| import { ITEMS_PER_PAGE } from '../constants'; | ||||
| import { renderQuota } from '../helpers/render'; | ||||
|  | ||||
| function renderTimestamp(timestamp) { | ||||
|   return ( | ||||
| @@ -34,6 +35,8 @@ const TokensTable = () => { | ||||
|   const [activePage, setActivePage] = useState(1); | ||||
|   const [searchKeyword, setSearchKeyword] = useState(''); | ||||
|   const [searching, setSearching] = useState(false); | ||||
|   const [showTopUpModal, setShowTopUpModal] = useState(false); | ||||
|   const [targetTokenIdx, setTargetTokenIdx] = useState(0); | ||||
|  | ||||
|   const loadTokens = async (startIdx) => { | ||||
|     const res = await API.get(`/api/token/?p=${startIdx}`); | ||||
| @@ -62,6 +65,11 @@ const TokensTable = () => { | ||||
|     })(); | ||||
|   }; | ||||
|  | ||||
|   const refresh = async () => { | ||||
|     setLoading(true); | ||||
|     await loadTokens(0); | ||||
|   } | ||||
|  | ||||
|   useEffect(() => { | ||||
|     loadTokens(0) | ||||
|       .then() | ||||
| @@ -79,11 +87,11 @@ const TokensTable = () => { | ||||
|         break; | ||||
|       case 'enable': | ||||
|         data.status = 1; | ||||
|         res = await API.put('/api/token/', data); | ||||
|         res = await API.put('/api/token/?status_only=true', data); | ||||
|         break; | ||||
|       case 'disable': | ||||
|         data.status = 2; | ||||
|         res = await API.put('/api/token/', data); | ||||
|         res = await API.put('/api/token/?status_only=true', data); | ||||
|         break; | ||||
|     } | ||||
|     const { success, message } = res.data; | ||||
| @@ -147,24 +155,16 @@ const TokensTable = () => { | ||||
|           icon='search' | ||||
|           fluid | ||||
|           iconPosition='left' | ||||
|           placeholder='搜索令牌的 ID 和名称 ...' | ||||
|           placeholder='搜索令牌的名称 ...' | ||||
|           value={searchKeyword} | ||||
|           loading={searching} | ||||
|           onChange={handleKeywordChange} | ||||
|         /> | ||||
|       </Form> | ||||
|  | ||||
|       <Table basic> | ||||
|       <Table basic compact size='small'> | ||||
|         <Table.Header> | ||||
|           <Table.Row> | ||||
|             <Table.HeaderCell | ||||
|               style={{ cursor: 'pointer' }} | ||||
|               onClick={() => { | ||||
|                 sortToken('id'); | ||||
|               }} | ||||
|             > | ||||
|               ID | ||||
|             </Table.HeaderCell> | ||||
|             <Table.HeaderCell | ||||
|               style={{ cursor: 'pointer' }} | ||||
|               onClick={() => { | ||||
| @@ -184,10 +184,10 @@ const TokensTable = () => { | ||||
|             <Table.HeaderCell | ||||
|               style={{ cursor: 'pointer' }} | ||||
|               onClick={() => { | ||||
|                 sortToken('remain_times'); | ||||
|                 sortToken('remain_quota'); | ||||
|               }} | ||||
|             > | ||||
|               剩余次数 | ||||
|               额度 | ||||
|             </Table.HeaderCell> | ||||
|             <Table.HeaderCell | ||||
|               style={{ cursor: 'pointer' }} | ||||
| @@ -197,14 +197,6 @@ const TokensTable = () => { | ||||
|             > | ||||
|               创建时间 | ||||
|             </Table.HeaderCell> | ||||
|             <Table.HeaderCell | ||||
|               style={{ cursor: 'pointer' }} | ||||
|               onClick={() => { | ||||
|                 sortToken('accessed_time'); | ||||
|               }} | ||||
|             > | ||||
|               访问时间 | ||||
|             </Table.HeaderCell> | ||||
|             <Table.HeaderCell | ||||
|               style={{ cursor: 'pointer' }} | ||||
|               onClick={() => { | ||||
| @@ -227,38 +219,47 @@ const TokensTable = () => { | ||||
|               if (token.deleted) return <></>; | ||||
|               return ( | ||||
|                 <Table.Row key={token.id}> | ||||
|                   <Table.Cell>{token.id}</Table.Cell> | ||||
|                   <Table.Cell>{token.name ? token.name : '无'}</Table.Cell> | ||||
|                   <Table.Cell>{renderStatus(token.status)}</Table.Cell> | ||||
|                   <Table.Cell>{token.remain_times === -1 ? "无限制" : token.remain_times}</Table.Cell> | ||||
|                   <Table.Cell>{token.unlimited_quota ? '无限制' : renderQuota(token.remain_quota, 2)}</Table.Cell> | ||||
|                   <Table.Cell>{renderTimestamp(token.created_time)}</Table.Cell> | ||||
|                   <Table.Cell>{renderTimestamp(token.accessed_time)}</Table.Cell> | ||||
|                   <Table.Cell>{token.expired_time === -1 ? "永不过期" : renderTimestamp(token.expired_time)}</Table.Cell> | ||||
|                   <Table.Cell>{token.expired_time === -1 ? '永不过期' : renderTimestamp(token.expired_time)}</Table.Cell> | ||||
|                   <Table.Cell> | ||||
|                     <div> | ||||
|                       <Button | ||||
|                         size={'small'} | ||||
|                         positive | ||||
|                         onClick={async () => { | ||||
|                           if (await copy(token.key)) { | ||||
|                           let key = "sk-" + token.key; | ||||
|                           if (await copy(key)) { | ||||
|                             showSuccess('已复制到剪贴板!'); | ||||
|                           } else { | ||||
|                             showWarning('无法复制到剪贴板,请手动复制,已将令牌填入搜索框。') | ||||
|                             setSearchKeyword(token.key); | ||||
|                             showWarning('无法复制到剪贴板,请手动复制,已将令牌填入搜索框。'); | ||||
|                             setSearchKeyword(key); | ||||
|                           } | ||||
|                         }} | ||||
|                       > | ||||
|                         复制 | ||||
|                       </Button> | ||||
|                       <Button | ||||
|                         size={'small'} | ||||
|                         negative | ||||
|                         onClick={() => { | ||||
|                           manageToken(token.id, 'delete', idx); | ||||
|                         }} | ||||
|                       <Popup | ||||
|                         trigger={ | ||||
|                           <Button size='small' negative> | ||||
|                             删除 | ||||
|                           </Button> | ||||
|                         } | ||||
|                         on='click' | ||||
|                         flowing | ||||
|                         hoverable | ||||
|                       > | ||||
|                         删除 | ||||
|                       </Button> | ||||
|                         <Button | ||||
|                           negative | ||||
|                           onClick={() => { | ||||
|                             manageToken(token.id, 'delete', idx); | ||||
|                           }} | ||||
|                         > | ||||
|                           删除令牌 {token.name} | ||||
|                         </Button> | ||||
|                       </Popup> | ||||
|                       <Button | ||||
|                         size={'small'} | ||||
|                         onClick={() => { | ||||
| @@ -287,10 +288,11 @@ const TokensTable = () => { | ||||
|  | ||||
|         <Table.Footer> | ||||
|           <Table.Row> | ||||
|             <Table.HeaderCell colSpan='8'> | ||||
|             <Table.HeaderCell colSpan='7'> | ||||
|               <Button size='small' as={Link} to='/token/add' loading={loading}> | ||||
|                 添加新的令牌 | ||||
|               </Button> | ||||
|               <Button size='small' onClick={refresh} loading={loading}>刷新</Button> | ||||
|               <Pagination | ||||
|                 floated='right' | ||||
|                 activePage={activePage} | ||||
|   | ||||
| @@ -1,9 +1,10 @@ | ||||
| import React, { useEffect, useState } from 'react'; | ||||
| import { Button, Form, Label, Pagination, Table } from 'semantic-ui-react'; | ||||
| import { Button, Form, Label, Pagination, Popup, Table } from 'semantic-ui-react'; | ||||
| import { Link } from 'react-router-dom'; | ||||
| import { API, showError, showSuccess } from '../helpers'; | ||||
|  | ||||
| import { ITEMS_PER_PAGE } from '../constants'; | ||||
| import { renderGroup, renderNumber, renderQuota, renderText } from '../helpers/render'; | ||||
|  | ||||
| function renderRole(role) { | ||||
|   switch (role) { | ||||
| @@ -64,7 +65,7 @@ const UsersTable = () => { | ||||
|     (async () => { | ||||
|       const res = await API.post('/api/user/manage', { | ||||
|         username, | ||||
|         action, | ||||
|         action | ||||
|       }); | ||||
|       const { success, message } = res.data; | ||||
|       if (success) { | ||||
| @@ -155,9 +156,17 @@ const UsersTable = () => { | ||||
|         /> | ||||
|       </Form> | ||||
|  | ||||
|       <Table basic> | ||||
|       <Table basic compact size='small'> | ||||
|         <Table.Header> | ||||
|           <Table.Row> | ||||
|             <Table.HeaderCell | ||||
|               style={{ cursor: 'pointer' }} | ||||
|               onClick={() => { | ||||
|                 sortUser('id'); | ||||
|               }} | ||||
|             > | ||||
|               ID | ||||
|             </Table.HeaderCell> | ||||
|             <Table.HeaderCell | ||||
|               style={{ cursor: 'pointer' }} | ||||
|               onClick={() => { | ||||
| @@ -169,10 +178,10 @@ const UsersTable = () => { | ||||
|             <Table.HeaderCell | ||||
|               style={{ cursor: 'pointer' }} | ||||
|               onClick={() => { | ||||
|                 sortUser('display_name'); | ||||
|                 sortUser('group'); | ||||
|               }} | ||||
|             > | ||||
|               显示名称 | ||||
|               分组 | ||||
|             </Table.HeaderCell> | ||||
|             <Table.HeaderCell | ||||
|               style={{ cursor: 'pointer' }} | ||||
| @@ -182,6 +191,14 @@ const UsersTable = () => { | ||||
|             > | ||||
|               邮箱地址 | ||||
|             </Table.HeaderCell> | ||||
|             <Table.HeaderCell | ||||
|               style={{ cursor: 'pointer' }} | ||||
|               onClick={() => { | ||||
|                 sortUser('quota'); | ||||
|               }} | ||||
|             > | ||||
|               统计信息 | ||||
|             </Table.HeaderCell> | ||||
|             <Table.HeaderCell | ||||
|               style={{ cursor: 'pointer' }} | ||||
|               onClick={() => { | ||||
| @@ -212,9 +229,25 @@ const UsersTable = () => { | ||||
|               if (user.deleted) return <></>; | ||||
|               return ( | ||||
|                 <Table.Row key={user.id}> | ||||
|                   <Table.Cell>{user.username}</Table.Cell> | ||||
|                   <Table.Cell>{user.display_name}</Table.Cell> | ||||
|                   <Table.Cell>{user.email ? user.email : '无'}</Table.Cell> | ||||
|                   <Table.Cell>{user.id}</Table.Cell> | ||||
|                   <Table.Cell> | ||||
|                     <Popup | ||||
|                       content={user.email ? user.email : '未绑定邮箱地址'} | ||||
|                       key={user.display_name} | ||||
|                       header={user.display_name ? user.display_name : user.username} | ||||
|                       trigger={<span>{renderText(user.username, 10)}</span>} | ||||
|                       hoverable | ||||
|                     /> | ||||
|                   </Table.Cell> | ||||
|                   <Table.Cell>{renderGroup(user.group)}</Table.Cell> | ||||
|                   <Table.Cell> | ||||
|                     {user.email ? <Popup hoverable content={user.email} trigger={<span>{renderText(user.email, 24)}</span>} /> : '无'} | ||||
|                   </Table.Cell> | ||||
|                   <Table.Cell> | ||||
|                     <Popup content='剩余额度' trigger={<Label>{renderQuota(user.quota)}</Label>} /> | ||||
|                     <Popup content='已用额度' trigger={<Label>{renderQuota(user.used_quota)}</Label>} /> | ||||
|                     <Popup content='请求次数' trigger={<Label>{renderNumber(user.request_count)}</Label>} /> | ||||
|                   </Table.Cell> | ||||
|                   <Table.Cell>{renderRole(user.role)}</Table.Cell> | ||||
|                   <Table.Cell>{renderStatus(user.status)}</Table.Cell> | ||||
|                   <Table.Cell> | ||||
| @@ -225,6 +258,7 @@ const UsersTable = () => { | ||||
|                         onClick={() => { | ||||
|                           manageUser(user.username, 'promote', idx); | ||||
|                         }} | ||||
|                         disabled={user.role === 100} | ||||
|                       > | ||||
|                         提升 | ||||
|                       </Button> | ||||
| @@ -234,18 +268,29 @@ const UsersTable = () => { | ||||
|                         onClick={() => { | ||||
|                           manageUser(user.username, 'demote', idx); | ||||
|                         }} | ||||
|                         disabled={user.role === 100} | ||||
|                       > | ||||
|                         降级 | ||||
|                       </Button> | ||||
|                       <Button | ||||
|                         size={'small'} | ||||
|                         negative | ||||
|                         onClick={() => { | ||||
|                           manageUser(user.username, 'delete', idx); | ||||
|                         }} | ||||
|                       <Popup | ||||
|                         trigger={ | ||||
|                           <Button size='small' negative disabled={user.role === 100}> | ||||
|                             删除 | ||||
|                           </Button> | ||||
|                         } | ||||
|                         on='click' | ||||
|                         flowing | ||||
|                         hoverable | ||||
|                       > | ||||
|                         删除 | ||||
|                       </Button> | ||||
|                         <Button | ||||
|                           negative | ||||
|                           onClick={() => { | ||||
|                             manageUser(user.username, 'delete', idx); | ||||
|                           }} | ||||
|                         > | ||||
|                           删除用户 {user.username} | ||||
|                         </Button> | ||||
|                       </Popup> | ||||
|                       <Button | ||||
|                         size={'small'} | ||||
|                         onClick={() => { | ||||
| @@ -255,6 +300,7 @@ const UsersTable = () => { | ||||
|                             idx | ||||
|                           ); | ||||
|                         }} | ||||
|                         disabled={user.role === 100} | ||||
|                       > | ||||
|                         {user.status === 1 ? '禁用' : '启用'} | ||||
|                       </Button> | ||||
| @@ -274,7 +320,7 @@ const UsersTable = () => { | ||||
|  | ||||
|         <Table.Footer> | ||||
|           <Table.Row> | ||||
|             <Table.HeaderCell colSpan='6'> | ||||
|             <Table.HeaderCell colSpan='8'> | ||||
|               <Button size='small' as={Link} to='/user/add' loading={loading}> | ||||
|                 添加新的用户 | ||||
|               </Button> | ||||
|   | ||||
| @@ -1,10 +1,13 @@ | ||||
| export const CHANNEL_OPTIONS = [ | ||||
|   { key: 1, text: 'OpenAI', value: 1, color: 'green' }, | ||||
|   { key: 2, text: 'API2D', value: 2, color: 'blue' }, | ||||
|   { key: 8, text: '自定义', value: 8, color: 'pink' }, | ||||
|   { key: 3, text: 'Azure', value: 3, color: 'olive' }, | ||||
|   { key: 2, text: 'API2D', value: 2, color: 'blue' }, | ||||
|   { key: 4, text: 'CloseAI', value: 4, color: 'teal' }, | ||||
|   { key: 5, text: 'OpenAI-SB', value: 5, color: 'brown' }, | ||||
|   { key: 6, text: 'OpenAI Max', value: 6, color: 'violet' }, | ||||
|   { key: 7, text: 'OhMyGPT', value: 7, color: 'purple' }, | ||||
|   { key: 8, text: '自定义', value: 8, color: 'pink' } | ||||
|   { key: 9, text: 'AI.LS', value: 9, color: 'yellow' }, | ||||
|   { key: 10, text: 'AI Proxy', value: 10, color: 'purple' }, | ||||
|   { key: 12, text: 'API2GPT', value: 12, color: 'blue' } | ||||
| ]; | ||||
|   | ||||
							
								
								
									
										58
									
								
								web/src/helpers/render.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										58
									
								
								web/src/helpers/render.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,58 @@ | ||||
| import { Label } from 'semantic-ui-react'; | ||||
|  | ||||
| export function renderText(text, limit) { | ||||
|   if (text.length > limit) { | ||||
|     return text.slice(0, limit - 3) + '...'; | ||||
|   } | ||||
|   return text; | ||||
| } | ||||
|  | ||||
| export function renderGroup(group) { | ||||
|   if (group === '') { | ||||
|     return <Label>default</Label>; | ||||
|   } | ||||
|   let groups = group.split(','); | ||||
|   groups.sort(); | ||||
|   return <> | ||||
|     {groups.map((group) => { | ||||
|       if (group === 'vip' || group === 'pro') { | ||||
|         return <Label color='yellow'>{group}</Label>; | ||||
|       } else if (group === 'svip' || group === 'premium') { | ||||
|         return <Label color='red'>{group}</Label>; | ||||
|       } | ||||
|       return <Label>{group}</Label>; | ||||
|     })} | ||||
|   </>; | ||||
| } | ||||
|  | ||||
| export function renderNumber(num) { | ||||
|   if (num >= 1000000000) { | ||||
|     return (num / 1000000000).toFixed(1) + 'B'; | ||||
|   } else if (num >= 1000000) { | ||||
|     return (num / 1000000).toFixed(1) + 'M'; | ||||
|   } else if (num >= 10000) { | ||||
|     return (num / 1000).toFixed(1) + 'k'; | ||||
|   } else { | ||||
|     return num; | ||||
|   } | ||||
| } | ||||
|  | ||||
| export function renderQuota(quota, digits = 2) { | ||||
|   let quotaPerUnit = localStorage.getItem('quota_per_unit'); | ||||
|   let displayInCurrency = localStorage.getItem('display_in_currency'); | ||||
|   quotaPerUnit = parseFloat(quotaPerUnit); | ||||
|   displayInCurrency = displayInCurrency === 'true'; | ||||
|   if (displayInCurrency) { | ||||
|     return '$' + (quota / quotaPerUnit).toFixed(digits); | ||||
|   } | ||||
|   return renderNumber(quota); | ||||
| } | ||||
|  | ||||
| export function renderQuotaWithPrompt(quota, digits) { | ||||
|   let displayInCurrency = localStorage.getItem('display_in_currency'); | ||||
|   displayInCurrency = displayInCurrency === 'true'; | ||||
|   if (displayInCurrency) { | ||||
|     return `(等价金额:${renderQuota(quota, digits)})`; | ||||
|   } | ||||
|   return ''; | ||||
| } | ||||
| @@ -15,6 +15,22 @@ export function isRoot() { | ||||
|   return user.role >= 100; | ||||
| } | ||||
|  | ||||
| export function getSystemName() { | ||||
|   let system_name = localStorage.getItem('system_name'); | ||||
|   if (!system_name) return 'One API'; | ||||
|   return system_name; | ||||
| } | ||||
|  | ||||
| export function getLogo() { | ||||
|   let logo = localStorage.getItem('logo'); | ||||
|   if (!logo) return '/logo.png'; | ||||
|   return logo | ||||
| } | ||||
|  | ||||
| export function getFooterHTML() { | ||||
|   return localStorage.getItem('footer_html'); | ||||
| } | ||||
|  | ||||
| export async function copy(text) { | ||||
|   let okay = true; | ||||
|   try { | ||||
| @@ -54,14 +70,18 @@ export function showError(error) { | ||||
|   console.error(error); | ||||
|   if (error.message) { | ||||
|     if (error.name === 'AxiosError') { | ||||
|       switch (error.message) { | ||||
|         case 'Request failed with status code 429': | ||||
|       switch (error.response.status) { | ||||
|         case 401: | ||||
|           // toast.error('错误:未登录或登录已过期,请重新登录!', showErrorOptions); | ||||
|           window.location.href = '/login?expired=true'; | ||||
|           break; | ||||
|         case 429: | ||||
|           toast.error('错误:请求次数过多,请稍后再试!', showErrorOptions); | ||||
|           break; | ||||
|         case 'Request failed with status code 500': | ||||
|         case 500: | ||||
|           toast.error('错误:服务器内部错误,请联系管理员!', showErrorOptions); | ||||
|           break; | ||||
|         case 'Request failed with status code 405': | ||||
|         case 405: | ||||
|           toast.info('本站仅作演示之用,无服务端!'); | ||||
|           break; | ||||
|         default: | ||||
| @@ -139,4 +159,22 @@ export function timestamp2string(timestamp) { | ||||
|     ':' + | ||||
|     second | ||||
|   ); | ||||
| } | ||||
| } | ||||
|  | ||||
| export function downloadTextAsFile(text, filename) { | ||||
|   let blob = new Blob([text], { type: 'text/plain;charset=utf-8' }); | ||||
|   let url = URL.createObjectURL(blob); | ||||
|   let a = document.createElement('a'); | ||||
|   a.href = url; | ||||
|   a.download = filename; | ||||
|   a.click(); | ||||
| } | ||||
|  | ||||
| export const verifyJSON = (str) => { | ||||
|   try { | ||||
|     JSON.parse(str); | ||||
|   } catch (e) { | ||||
|     return false; | ||||
|   } | ||||
|   return true; | ||||
| }; | ||||
| @@ -5,6 +5,11 @@ body { | ||||
|     font-family: Lato, 'Helvetica Neue', Arial, Helvetica, "Microsoft YaHei", sans-serif; | ||||
|     -webkit-font-smoothing: antialiased; | ||||
|     -moz-osx-font-smoothing: grayscale; | ||||
|     scrollbar-width: none; | ||||
| } | ||||
|  | ||||
| body::-webkit-scrollbar { | ||||
|     display: none; | ||||
| } | ||||
|  | ||||
| code { | ||||
|   | ||||
| @@ -5,18 +5,24 @@ import { marked } from 'marked'; | ||||
|  | ||||
| const About = () => { | ||||
|   const [about, setAbout] = useState(''); | ||||
|   const [aboutLoaded, setAboutLoaded] = useState(false); | ||||
|  | ||||
|   const displayAbout = async () => { | ||||
|     setAbout(localStorage.getItem('about') || ''); | ||||
|     const res = await API.get('/api/about'); | ||||
|     const { success, message, data } = res.data; | ||||
|     if (success) { | ||||
|       let HTMLAbout = marked.parse(data); | ||||
|       localStorage.setItem('about', HTMLAbout); | ||||
|       setAbout(HTMLAbout); | ||||
|       let aboutContent = data; | ||||
|       if (!data.startsWith('https://')) { | ||||
|         aboutContent = marked.parse(data); | ||||
|       } | ||||
|       setAbout(aboutContent); | ||||
|       localStorage.setItem('about', aboutContent); | ||||
|     } else { | ||||
|       showError(message); | ||||
|       setAbout('加载关于内容失败...'); | ||||
|     } | ||||
|     setAboutLoaded(true); | ||||
|   }; | ||||
|  | ||||
|   useEffect(() => { | ||||
| @@ -25,20 +31,27 @@ const About = () => { | ||||
|  | ||||
|   return ( | ||||
|     <> | ||||
|       <Segment> | ||||
|         { | ||||
|           about === '' ? <> | ||||
|       { | ||||
|         aboutLoaded && about === '' ? <> | ||||
|           <Segment> | ||||
|             <Header as='h3'>关于</Header> | ||||
|             <p>可在设置页面设置关于内容,支持 HTML & Markdown</p> | ||||
|             项目仓库地址: | ||||
|             <a href="https://github.com/songquanpeng/one-api"> | ||||
|             <a href='https://github.com/songquanpeng/one-api'> | ||||
|               https://github.com/songquanpeng/one-api | ||||
|             </a> | ||||
|           </> : <> | ||||
|             <div dangerouslySetInnerHTML={{ __html: about}}></div> | ||||
|           </> | ||||
|         } | ||||
|       </Segment> | ||||
|           </Segment> | ||||
|         </> : <> | ||||
|           { | ||||
|             about.startsWith('https://') ? <iframe | ||||
|               src={about} | ||||
|               style={{ width: '100%', height: '100vh', border: 'none' }} | ||||
|             /> : <Segment> | ||||
|               <div style={{ fontSize: 'larger' }} dangerouslySetInnerHTML={{ __html: about }}></div> | ||||
|             </Segment> | ||||
|           } | ||||
|         </> | ||||
|       } | ||||
|     </> | ||||
|   ); | ||||
| }; | ||||
|   | ||||
| @@ -1,95 +0,0 @@ | ||||
| import React, { useState } from 'react'; | ||||
| import { Button, Form, Header, Segment } from 'semantic-ui-react'; | ||||
| import { API, showError, showSuccess } from '../../helpers'; | ||||
| import { CHANNEL_OPTIONS } from '../../constants'; | ||||
|  | ||||
| const AddChannel = () => { | ||||
|   const originInputs = { | ||||
|     name: '', | ||||
|     type: 1, | ||||
|     key: '', | ||||
|     base_url: '', | ||||
|   }; | ||||
|   const [inputs, setInputs] = useState(originInputs); | ||||
|   const { name, type, key } = inputs; | ||||
|  | ||||
|   const handleInputChange = (e, { name, value }) => { | ||||
|     setInputs((inputs) => ({ ...inputs, [name]: value })); | ||||
|   }; | ||||
|  | ||||
|   const submit = async () => { | ||||
|     if (inputs.name === '' || inputs.key === '') return; | ||||
|     if (inputs.base_url.endsWith('/')) { | ||||
|       inputs.base_url = inputs.base_url.slice(0, inputs.base_url.length - 1); | ||||
|     } | ||||
|     const res = await API.post(`/api/channel/`, inputs); | ||||
|     const { success, message } = res.data; | ||||
|     if (success) { | ||||
|       showSuccess('渠道创建成功!'); | ||||
|       setInputs(originInputs); | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   return ( | ||||
|     <> | ||||
|       <Segment> | ||||
|         <Header as='h3'>创建新的渠道</Header> | ||||
|         <Form autoComplete='off'> | ||||
|           <Form.Field> | ||||
|             <Form.Select | ||||
|               label='类型' | ||||
|               name='type' | ||||
|               options={CHANNEL_OPTIONS} | ||||
|               value={inputs.type} | ||||
|               onChange={handleInputChange} | ||||
|             /> | ||||
|           </Form.Field> | ||||
|           { | ||||
|             type === 8 && ( | ||||
|               <Form.Field> | ||||
|                 <Form.Input | ||||
|                   label='Base URL' | ||||
|                   name='base_url' | ||||
|                   placeholder={'请输入自定义渠道的 Base URL'} | ||||
|                   onChange={handleInputChange} | ||||
|                   value={inputs.base_url} | ||||
|                   autoComplete='off' | ||||
|                 /> | ||||
|               </Form.Field> | ||||
|             ) | ||||
|           } | ||||
|           <Form.Field> | ||||
|             <Form.Input | ||||
|               label='名称' | ||||
|               name='name' | ||||
|               placeholder={'请输入名称'} | ||||
|               onChange={handleInputChange} | ||||
|               value={name} | ||||
|               autoComplete='off' | ||||
|               required | ||||
|             /> | ||||
|           </Form.Field> | ||||
|           <Form.Field> | ||||
|             <Form.Input | ||||
|               label='密钥' | ||||
|               name='key' | ||||
|               placeholder={'请输入密钥'} | ||||
|               onChange={handleInputChange} | ||||
|               value={key} | ||||
|               // type='password' | ||||
|               autoComplete='off' | ||||
|               required | ||||
|             /> | ||||
|           </Form.Field> | ||||
|           <Button type={'submit'} onClick={submit}> | ||||
|             提交 | ||||
|           </Button> | ||||
|         </Form> | ||||
|       </Segment> | ||||
|     </> | ||||
|   ); | ||||
| }; | ||||
|  | ||||
| export default AddChannel; | ||||
| @@ -1,19 +1,29 @@ | ||||
| import React, { useEffect, useState } from 'react'; | ||||
| import { Button, Form, Header, Segment } from 'semantic-ui-react'; | ||||
| import { Button, Form, Header, Message, Segment } from 'semantic-ui-react'; | ||||
| import { useParams } from 'react-router-dom'; | ||||
| import { API, showError, showSuccess } from '../../helpers'; | ||||
| import { API, showError, showInfo, showSuccess } from '../../helpers'; | ||||
| import { CHANNEL_OPTIONS } from '../../constants'; | ||||
|  | ||||
| const EditChannel = () => { | ||||
|   const params = useParams(); | ||||
|   const channelId = params.id; | ||||
|   const [loading, setLoading] = useState(true); | ||||
|   const [inputs, setInputs] = useState({ | ||||
|   const isEdit = channelId !== undefined; | ||||
|   const [loading, setLoading] = useState(isEdit); | ||||
|   const originInputs = { | ||||
|     name: '', | ||||
|     key: '', | ||||
|     type: 1, | ||||
|     key: '', | ||||
|     base_url: '', | ||||
|   }); | ||||
|     other: '', | ||||
|     models: [], | ||||
|     groups: ['default'] | ||||
|   }; | ||||
|   const [batch, setBatch] = useState(false); | ||||
|   const [inputs, setInputs] = useState(originInputs); | ||||
|   const [modelOptions, setModelOptions] = useState([]); | ||||
|   const [groupOptions, setGroupOptions] = useState([]); | ||||
|   const [basicModels, setBasicModels] = useState([]); | ||||
|   const [fullModels, setFullModels] = useState([]); | ||||
|   const handleInputChange = (e, { name, value }) => { | ||||
|     setInputs((inputs) => ({ ...inputs, [name]: value })); | ||||
|   }; | ||||
| @@ -22,25 +32,87 @@ const EditChannel = () => { | ||||
|     let res = await API.get(`/api/channel/${channelId}`); | ||||
|     const { success, message, data } = res.data; | ||||
|     if (success) { | ||||
|       data.password = ''; | ||||
|       if (data.models === "") { | ||||
|         data.models = [] | ||||
|       } else { | ||||
|         data.models = data.models.split(",") | ||||
|       } | ||||
|       if (data.group === "") { | ||||
|         data.groups = [] | ||||
|       } else { | ||||
|         data.groups = data.group.split(",") | ||||
|       } | ||||
|       setInputs(data); | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
|     setLoading(false); | ||||
|   }; | ||||
|  | ||||
|   const fetchModels = async () => { | ||||
|     try { | ||||
|       let res = await API.get(`/api/channel/models`); | ||||
|       setModelOptions(res.data.data.map((model) => ({ | ||||
|         key: model.id, | ||||
|         text: model.id, | ||||
|         value: model.id, | ||||
|       }))); | ||||
|       setFullModels(res.data.data.map((model) => model.id)); | ||||
|       setBasicModels(res.data.data.filter((model) => !model.id.startsWith("gpt-4")).map((model) => model.id)); | ||||
|     } catch (error) { | ||||
|       showError(error.message); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   const fetchGroups = async () => { | ||||
|     try { | ||||
|       let res = await API.get(`/api/group/`); | ||||
|       setGroupOptions(res.data.data.map((group) => ({ | ||||
|         key: group, | ||||
|         text: group, | ||||
|         value: group, | ||||
|       }))); | ||||
|     } catch (error) { | ||||
|       showError(error.message); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   useEffect(() => { | ||||
|     loadChannel().then(); | ||||
|     if (isEdit) { | ||||
|       loadChannel().then(); | ||||
|     } | ||||
|     fetchModels().then(); | ||||
|     fetchGroups().then(); | ||||
|   }, []); | ||||
|  | ||||
|   const submit = async () => { | ||||
|     if (inputs.base_url.endsWith('/')) { | ||||
|       inputs.base_url = inputs.base_url.slice(0, inputs.base_url.length - 1); | ||||
|     if (!isEdit && (inputs.name === '' || inputs.key === '')) { | ||||
|       showInfo('请填写渠道名称和渠道密钥!'); | ||||
|       return; | ||||
|     } | ||||
|     let localInputs = inputs; | ||||
|     if (localInputs.base_url.endsWith('/')) { | ||||
|       localInputs.base_url = localInputs.base_url.slice(0, localInputs.base_url.length - 1); | ||||
|     } | ||||
|     if (localInputs.type === 3 && localInputs.other === '') { | ||||
|       localInputs.other = '2023-03-15-preview'; | ||||
|     } | ||||
|     let res; | ||||
|     localInputs.models = localInputs.models.join(",") | ||||
|     localInputs.group = localInputs.groups.join(",") | ||||
|     if (isEdit) { | ||||
|       res = await API.put(`/api/channel/`, { ...localInputs, id: parseInt(channelId) }); | ||||
|     } else { | ||||
|       res = await API.post(`/api/channel/`, localInputs); | ||||
|     } | ||||
|     let res = await API.put(`/api/channel/`, { ...inputs, id: parseInt(channelId) }); | ||||
|     const { success, message } = res.data; | ||||
|     if (success) { | ||||
|       showSuccess('渠道更新成功!'); | ||||
|       if (isEdit) { | ||||
|         showSuccess('渠道更新成功!'); | ||||
|       } else { | ||||
|         showSuccess('渠道创建成功!'); | ||||
|         setInputs(originInputs); | ||||
|       } | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
| @@ -49,8 +121,8 @@ const EditChannel = () => { | ||||
|   return ( | ||||
|     <> | ||||
|       <Segment loading={loading}> | ||||
|         <Header as='h3'>更新渠道信息</Header> | ||||
|         <Form autoComplete='off'> | ||||
|         <Header as='h3'>{isEdit ? '更新渠道信息' : '创建新的渠道'}</Header> | ||||
|         <Form autoComplete='new-password'> | ||||
|           <Form.Field> | ||||
|             <Form.Select | ||||
|               label='类型' | ||||
| @@ -60,16 +132,61 @@ const EditChannel = () => { | ||||
|               onChange={handleInputChange} | ||||
|             /> | ||||
|           </Form.Field> | ||||
|           { | ||||
|             inputs.type === 3 && ( | ||||
|               <> | ||||
|                 <Message> | ||||
|                   注意,<strong>模型部署名称必须和模型名称保持一致</strong>,因为 One API 会把请求体中的 model | ||||
|                   参数替换为你的部署名称(模型名称中的点会被剔除),<a target='_blank' | ||||
|                                                                     href='https://github.com/songquanpeng/one-api/issues/133?notification_referrer_id=NT_kwDOAmJSYrM2NjIwMzI3NDgyOjM5OTk4MDUw#issuecomment-1571602271'>图片演示</a>。 | ||||
|                 </Message> | ||||
|                 <Form.Field> | ||||
|                   <Form.Input | ||||
|                     label='AZURE_OPENAI_ENDPOINT' | ||||
|                     name='base_url' | ||||
|                     placeholder={'请输入 AZURE_OPENAI_ENDPOINT,例如:https://docs-test-001.openai.azure.com'} | ||||
|                     onChange={handleInputChange} | ||||
|                     value={inputs.base_url} | ||||
|                     autoComplete='new-password' | ||||
|                   /> | ||||
|                 </Form.Field> | ||||
|                 <Form.Field> | ||||
|                   <Form.Input | ||||
|                     label='默认 API 版本' | ||||
|                     name='other' | ||||
|                     placeholder={'请输入默认 API 版本,例如:2023-03-15-preview,该配置可以被实际的请求查询参数所覆盖'} | ||||
|                     onChange={handleInputChange} | ||||
|                     value={inputs.other} | ||||
|                     autoComplete='new-password' | ||||
|                   /> | ||||
|                 </Form.Field> | ||||
|               </> | ||||
|             ) | ||||
|           } | ||||
|           { | ||||
|             inputs.type === 8 && ( | ||||
|               <Form.Field> | ||||
|                 <Form.Input | ||||
|                   label='Base URL' | ||||
|                   name='base_url' | ||||
|                   placeholder={'请输入新的自定义渠道的 Base URL'} | ||||
|                   placeholder={'请输入自定义渠道的 Base URL,例如:https://openai.justsong.cn'} | ||||
|                   onChange={handleInputChange} | ||||
|                   value={inputs.base_url} | ||||
|                   autoComplete='off' | ||||
|                   autoComplete='new-password' | ||||
|                 /> | ||||
|               </Form.Field> | ||||
|             ) | ||||
|           } | ||||
|           { | ||||
|             inputs.type !== 3 && inputs.type !== 8 && ( | ||||
|               <Form.Field> | ||||
|                 <Form.Input | ||||
|                   label='Base URL' | ||||
|                   name='base_url' | ||||
|                   placeholder={'请输入自定义 Base URL,格式为:https://domain.com,可不填,不填使用渠道默认值'} | ||||
|                   onChange={handleInputChange} | ||||
|                   value={inputs.base_url} | ||||
|                   autoComplete='new-password' | ||||
|                 /> | ||||
|               </Form.Field> | ||||
|             ) | ||||
| @@ -78,24 +195,97 @@ const EditChannel = () => { | ||||
|             <Form.Input | ||||
|               label='名称' | ||||
|               name='name' | ||||
|               placeholder={'请输入新的名称'} | ||||
|               placeholder={'请输入名称'} | ||||
|               onChange={handleInputChange} | ||||
|               value={inputs.name} | ||||
|               autoComplete='off' | ||||
|               autoComplete='new-password' | ||||
|             /> | ||||
|           </Form.Field> | ||||
|           <Form.Field> | ||||
|             <Form.Input | ||||
|               label='密钥' | ||||
|               name='key' | ||||
|               placeholder={'请输入新的密钥'} | ||||
|             <Form.Dropdown | ||||
|               label='分组' | ||||
|               placeholder={'请选择分组'} | ||||
|               name='groups' | ||||
|               fluid | ||||
|               multiple | ||||
|               selection | ||||
|               allowAdditions | ||||
|               additionLabel={'请在系统设置页面编辑分组倍率以添加新的分组:'} | ||||
|               onChange={handleInputChange} | ||||
|               value={inputs.key} | ||||
|               // type='password' | ||||
|               autoComplete='off' | ||||
|               value={inputs.groups} | ||||
|               autoComplete='new-password' | ||||
|               options={groupOptions} | ||||
|             /> | ||||
|           </Form.Field> | ||||
|           <Button onClick={submit}>提交</Button> | ||||
|           <Form.Field> | ||||
|             <Form.Dropdown | ||||
|               label='模型' | ||||
|               placeholder={'请选择该通道所支持的模型'} | ||||
|               name='models' | ||||
|               fluid | ||||
|               multiple | ||||
|               selection | ||||
|               onChange={handleInputChange} | ||||
|               value={inputs.models} | ||||
|               autoComplete='new-password' | ||||
|               options={modelOptions} | ||||
|             /> | ||||
|           </Form.Field> | ||||
|           <div style={{ lineHeight: '40px', marginBottom: '12px'}}> | ||||
|             <Button type={'button'} onClick={() => { | ||||
|               handleInputChange(null, { name: 'models', value: basicModels }); | ||||
|             }}>填入基础模型</Button> | ||||
|             <Button type={'button'} onClick={() => { | ||||
|               handleInputChange(null, { name: 'models', value: fullModels }); | ||||
|             }}>填入所有模型</Button> | ||||
|           </div> | ||||
|           { | ||||
|             inputs.type === 1 && ( | ||||
|               <Form.Field> | ||||
|                 <Form.Input | ||||
|                   label='代理' | ||||
|                   name='base_url' | ||||
|                   placeholder={'请输入 OpenAI API 代理地址,如果不需要请留空,格式为:https://api.openai.com'} | ||||
|                   onChange={handleInputChange} | ||||
|                   value={inputs.base_url} | ||||
|                   autoComplete='new-password' | ||||
|                 /> | ||||
|               </Form.Field> | ||||
|             ) | ||||
|           } | ||||
|           { | ||||
|             batch ? <Form.Field> | ||||
|               <Form.TextArea | ||||
|                 label='密钥' | ||||
|                 name='key' | ||||
|                 placeholder={'请输入密钥,一行一个'} | ||||
|                 onChange={handleInputChange} | ||||
|                 value={inputs.key} | ||||
|                 style={{ minHeight: 150, fontFamily: 'JetBrains Mono, Consolas' }} | ||||
|                 autoComplete='new-password' | ||||
|               /> | ||||
|             </Form.Field> : <Form.Field> | ||||
|               <Form.Input | ||||
|                 label='密钥' | ||||
|                 name='key' | ||||
|                 placeholder={'请输入密钥'} | ||||
|                 onChange={handleInputChange} | ||||
|                 value={inputs.key} | ||||
|                 autoComplete='new-password' | ||||
|               /> | ||||
|             </Form.Field> | ||||
|           } | ||||
|           { | ||||
|             !isEdit && ( | ||||
|               <Form.Checkbox | ||||
|                 checked={batch} | ||||
|                 label='批量创建' | ||||
|                 name='batch' | ||||
|                 onChange={() => setBatch(!batch)} | ||||
|               /> | ||||
|             ) | ||||
|           } | ||||
|           <Button positive onClick={submit}>提交</Button> | ||||
|         </Form> | ||||
|       </Segment> | ||||
|     </> | ||||
|   | ||||
							
								
								
									
										15
									
								
								web/src/pages/Chat/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								web/src/pages/Chat/index.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,15 @@ | ||||
| import React from 'react'; | ||||
|  | ||||
| const Chat = () => { | ||||
|   const chatLink = localStorage.getItem('chat_link'); | ||||
|  | ||||
|   return ( | ||||
|     <iframe | ||||
|       src={chatLink} | ||||
|       style={{ width: '100%', height: '85vh', border: 'none' }} | ||||
|     /> | ||||
|   ); | ||||
| }; | ||||
|  | ||||
|  | ||||
| export default Chat; | ||||
| @@ -1,10 +1,13 @@ | ||||
| import React, { useContext, useEffect } from 'react'; | ||||
| import React, { useContext, useEffect, useState } from 'react'; | ||||
| import { Card, Grid, Header, Segment } from 'semantic-ui-react'; | ||||
| import { API, showError, showNotice, timestamp2string } from '../../helpers'; | ||||
| import { StatusContext } from '../../context/Status'; | ||||
| import { marked } from 'marked'; | ||||
|  | ||||
| const Home = () => { | ||||
|   const [statusState, statusDispatch] = useContext(StatusContext); | ||||
|   const [homePageContentLoaded, setHomePageContentLoaded] = useState(false); | ||||
|   const [homePageContent, setHomePageContent] = useState(''); | ||||
|  | ||||
|   const displayNotice = async () => { | ||||
|     const res = await API.get('/api/notice'); | ||||
| @@ -20,6 +23,24 @@ const Home = () => { | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   const displayHomePageContent = async () => { | ||||
|     setHomePageContent(localStorage.getItem('home_page_content') || ''); | ||||
|     const res = await API.get('/api/home_page_content'); | ||||
|     const { success, message, data } = res.data; | ||||
|     if (success) { | ||||
|       let content = data; | ||||
|       if (!data.startsWith('https://')) { | ||||
|         content = marked.parse(data); | ||||
|       } | ||||
|       setHomePageContent(content); | ||||
|       localStorage.setItem('home_page_content', content); | ||||
|     } else { | ||||
|       showError(message); | ||||
|       setHomePageContent('加载首页内容失败...'); | ||||
|     } | ||||
|     setHomePageContentLoaded(true); | ||||
|   }; | ||||
|  | ||||
|   const getStartTimeString = () => { | ||||
|     const timestamp = statusState?.status?.start_time; | ||||
|     return timestamp2string(timestamp); | ||||
| @@ -27,70 +48,83 @@ const Home = () => { | ||||
|  | ||||
|   useEffect(() => { | ||||
|     displayNotice().then(); | ||||
|     displayHomePageContent().then(); | ||||
|   }, []); | ||||
|   return ( | ||||
|     <> | ||||
|       <Segment> | ||||
|         <Header as='h3'>系统状况</Header> | ||||
|         <Grid columns={2} stackable> | ||||
|           <Grid.Column> | ||||
|             <Card fluid> | ||||
|               <Card.Content> | ||||
|                 <Card.Header>系统信息</Card.Header> | ||||
|                 <Card.Meta>系统信息总览</Card.Meta> | ||||
|                 <Card.Description> | ||||
|                   <p>名称:{statusState?.status?.system_name}</p> | ||||
|                   <p>版本:{statusState?.status?.version}</p> | ||||
|                   <p> | ||||
|                     源码: | ||||
|                     <a | ||||
|                       href='https://github.com/songquanpeng/one-api' | ||||
|                       target='_blank' | ||||
|                     > | ||||
|                       https://github.com/songquanpeng/one-api | ||||
|                     </a> | ||||
|                   </p> | ||||
|                   <p>启动时间:{getStartTimeString()}</p> | ||||
|                 </Card.Description> | ||||
|               </Card.Content> | ||||
|             </Card> | ||||
|           </Grid.Column> | ||||
|           <Grid.Column> | ||||
|             <Card fluid> | ||||
|               <Card.Content> | ||||
|                 <Card.Header>系统配置</Card.Header> | ||||
|                 <Card.Meta>系统配置总览</Card.Meta> | ||||
|                 <Card.Description> | ||||
|                   <p> | ||||
|                     邮箱验证: | ||||
|                     {statusState?.status?.email_verification === true | ||||
|                       ? '已启用' | ||||
|                       : '未启用'} | ||||
|                   </p> | ||||
|                   <p> | ||||
|                     GitHub 身份验证: | ||||
|                     {statusState?.status?.github_oauth === true | ||||
|                       ? '已启用' | ||||
|                       : '未启用'} | ||||
|                   </p> | ||||
|                   <p> | ||||
|                     微信身份验证: | ||||
|                     {statusState?.status?.wechat_login === true | ||||
|                       ? '已启用' | ||||
|                       : '未启用'} | ||||
|                   </p> | ||||
|                   <p> | ||||
|                     Turnstile 用户校验: | ||||
|                     {statusState?.status?.turnstile_check === true | ||||
|                       ? '已启用' | ||||
|                       : '未启用'} | ||||
|                   </p> | ||||
|                 </Card.Description> | ||||
|               </Card.Content> | ||||
|             </Card> | ||||
|           </Grid.Column> | ||||
|         </Grid> | ||||
|       </Segment> | ||||
|       { | ||||
|         homePageContentLoaded && homePageContent === '' ? <> | ||||
|           <Segment> | ||||
|             <Header as='h3'>系统状况</Header> | ||||
|             <Grid columns={2} stackable> | ||||
|               <Grid.Column> | ||||
|                 <Card fluid> | ||||
|                   <Card.Content> | ||||
|                     <Card.Header>系统信息</Card.Header> | ||||
|                     <Card.Meta>系统信息总览</Card.Meta> | ||||
|                     <Card.Description> | ||||
|                       <p>名称:{statusState?.status?.system_name}</p> | ||||
|                       <p>版本:{statusState?.status?.version}</p> | ||||
|                       <p> | ||||
|                         源码: | ||||
|                         <a | ||||
|                           href='https://github.com/songquanpeng/one-api' | ||||
|                           target='_blank' | ||||
|                         > | ||||
|                           https://github.com/songquanpeng/one-api | ||||
|                         </a> | ||||
|                       </p> | ||||
|                       <p>启动时间:{getStartTimeString()}</p> | ||||
|                     </Card.Description> | ||||
|                   </Card.Content> | ||||
|                 </Card> | ||||
|               </Grid.Column> | ||||
|               <Grid.Column> | ||||
|                 <Card fluid> | ||||
|                   <Card.Content> | ||||
|                     <Card.Header>系统配置</Card.Header> | ||||
|                     <Card.Meta>系统配置总览</Card.Meta> | ||||
|                     <Card.Description> | ||||
|                       <p> | ||||
|                         邮箱验证: | ||||
|                         {statusState?.status?.email_verification === true | ||||
|                           ? '已启用' | ||||
|                           : '未启用'} | ||||
|                       </p> | ||||
|                       <p> | ||||
|                         GitHub 身份验证: | ||||
|                         {statusState?.status?.github_oauth === true | ||||
|                           ? '已启用' | ||||
|                           : '未启用'} | ||||
|                       </p> | ||||
|                       <p> | ||||
|                         微信身份验证: | ||||
|                         {statusState?.status?.wechat_login === true | ||||
|                           ? '已启用' | ||||
|                           : '未启用'} | ||||
|                       </p> | ||||
|                       <p> | ||||
|                         Turnstile 用户校验: | ||||
|                         {statusState?.status?.turnstile_check === true | ||||
|                           ? '已启用' | ||||
|                           : '未启用'} | ||||
|                       </p> | ||||
|                     </Card.Description> | ||||
|                   </Card.Content> | ||||
|                 </Card> | ||||
|               </Grid.Column> | ||||
|             </Grid> | ||||
|           </Segment> | ||||
|         </> : <> | ||||
|           { | ||||
|             homePageContent.startsWith('https://') ? <iframe | ||||
|               src={homePageContent} | ||||
|               style={{ width: '100%', height: '100vh', border: 'none' }} | ||||
|             /> : <div style={{ fontSize: 'larger' }} dangerouslySetInnerHTML={{ __html: homePageContent }}></div> | ||||
|           } | ||||
|         </> | ||||
|       } | ||||
|  | ||||
|     </> | ||||
|   ); | ||||
| }; | ||||
|   | ||||
							
								
								
									
										14
									
								
								web/src/pages/Log/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								web/src/pages/Log/index.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | ||||
| import React from 'react'; | ||||
| import { Header, Segment } from 'semantic-ui-react'; | ||||
| import LogsTable from '../../components/LogsTable'; | ||||
|  | ||||
| const Token = () => ( | ||||
|   <> | ||||
|     <Segment> | ||||
|       <Header as='h3'>额度明细</Header> | ||||
|       <LogsTable /> | ||||
|     </Segment> | ||||
|   </> | ||||
| ); | ||||
|  | ||||
| export default Token; | ||||
							
								
								
									
										122
									
								
								web/src/pages/Redemption/EditRedemption.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										122
									
								
								web/src/pages/Redemption/EditRedemption.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,122 @@ | ||||
| import React, { useEffect, useState } from 'react'; | ||||
| import { Button, Form, Header, Segment } from 'semantic-ui-react'; | ||||
| import { useParams } from 'react-router-dom'; | ||||
| import { API, downloadTextAsFile, showError, showSuccess } from '../../helpers'; | ||||
| import { renderQuota, renderQuotaWithPrompt } from '../../helpers/render'; | ||||
|  | ||||
| const EditRedemption = () => { | ||||
|   const params = useParams(); | ||||
|   const redemptionId = params.id; | ||||
|   const isEdit = redemptionId !== undefined; | ||||
|   const [loading, setLoading] = useState(isEdit); | ||||
|   const originInputs = { | ||||
|     name: '', | ||||
|     quota: 100000, | ||||
|     count: 1 | ||||
|   }; | ||||
|   const [inputs, setInputs] = useState(originInputs); | ||||
|   const { name, quota, count } = inputs; | ||||
|  | ||||
|   const handleInputChange = (e, { name, value }) => { | ||||
|     setInputs((inputs) => ({ ...inputs, [name]: value })); | ||||
|   }; | ||||
|  | ||||
|   const loadRedemption = async () => { | ||||
|     let res = await API.get(`/api/redemption/${redemptionId}`); | ||||
|     const { success, message, data } = res.data; | ||||
|     if (success) { | ||||
|       setInputs(data); | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
|     setLoading(false); | ||||
|   }; | ||||
|   useEffect(() => { | ||||
|     if (isEdit) { | ||||
|       loadRedemption().then(); | ||||
|     } | ||||
|   }, []); | ||||
|  | ||||
|   const submit = async () => { | ||||
|     if (!isEdit && inputs.name === '') return; | ||||
|     let localInputs = inputs; | ||||
|     localInputs.count = parseInt(localInputs.count); | ||||
|     localInputs.quota = parseInt(localInputs.quota); | ||||
|     let res; | ||||
|     if (isEdit) { | ||||
|       res = await API.put(`/api/redemption/`, { ...localInputs, id: parseInt(redemptionId) }); | ||||
|     } else { | ||||
|       res = await API.post(`/api/redemption/`, { | ||||
|         ...localInputs | ||||
|       }); | ||||
|     } | ||||
|     const { success, message, data } = res.data; | ||||
|     if (success) { | ||||
|       if (isEdit) { | ||||
|         showSuccess('兑换码更新成功!'); | ||||
|       } else { | ||||
|         showSuccess('兑换码创建成功!'); | ||||
|         setInputs(originInputs); | ||||
|       } | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
|     if (!isEdit && data) { | ||||
|       let text = ""; | ||||
|       for (let i = 0; i < data.length; i++) { | ||||
|         text += data[i] + "\n"; | ||||
|       } | ||||
|       downloadTextAsFile(text, `${inputs.name}.txt`); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   return ( | ||||
|     <> | ||||
|       <Segment loading={loading}> | ||||
|         <Header as='h3'>{isEdit ? '更新兑换码信息' : '创建新的兑换码'}</Header> | ||||
|         <Form autoComplete='new-password'> | ||||
|           <Form.Field> | ||||
|             <Form.Input | ||||
|               label='名称' | ||||
|               name='name' | ||||
|               placeholder={'请输入名称'} | ||||
|               onChange={handleInputChange} | ||||
|               value={name} | ||||
|               autoComplete='new-password' | ||||
|               required={!isEdit} | ||||
|             /> | ||||
|           </Form.Field> | ||||
|           <Form.Field> | ||||
|             <Form.Input | ||||
|               label={`额度${renderQuotaWithPrompt(quota)}`} | ||||
|               name='quota' | ||||
|               placeholder={'请输入单个兑换码中包含的额度'} | ||||
|               onChange={handleInputChange} | ||||
|               value={quota} | ||||
|               autoComplete='new-password' | ||||
|               type='number' | ||||
|             /> | ||||
|           </Form.Field> | ||||
|           { | ||||
|             !isEdit && <> | ||||
|               <Form.Field> | ||||
|                 <Form.Input | ||||
|                   label='生成数量' | ||||
|                   name='count' | ||||
|                   placeholder={'请输入生成数量'} | ||||
|                   onChange={handleInputChange} | ||||
|                   value={count} | ||||
|                   autoComplete='new-password' | ||||
|                   type='number' | ||||
|                 /> | ||||
|               </Form.Field> | ||||
|             </> | ||||
|           } | ||||
|           <Button positive onClick={submit}>提交</Button> | ||||
|         </Form> | ||||
|       </Segment> | ||||
|     </> | ||||
|   ); | ||||
| }; | ||||
|  | ||||
| export default EditRedemption; | ||||
							
								
								
									
										14
									
								
								web/src/pages/Redemption/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								web/src/pages/Redemption/index.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | ||||
| import React from 'react'; | ||||
| import { Segment, Header } from 'semantic-ui-react'; | ||||
| import RedemptionsTable from '../../components/RedemptionsTable'; | ||||
|  | ||||
| const Redemption = () => ( | ||||
|   <> | ||||
|     <Segment> | ||||
|       <Header as='h3'>管理兑换码</Header> | ||||
|       <RedemptionsTable/> | ||||
|     </Segment> | ||||
|   </> | ||||
| ); | ||||
|  | ||||
| export default Redemption; | ||||
| @@ -4,6 +4,7 @@ import SystemSetting from '../../components/SystemSetting'; | ||||
| import { isRoot } from '../../helpers'; | ||||
| import OtherSetting from '../../components/OtherSetting'; | ||||
| import PersonalSetting from '../../components/PersonalSetting'; | ||||
| import OperationSetting from '../../components/OperationSetting'; | ||||
|  | ||||
| const Setting = () => { | ||||
|   let panes = [ | ||||
| @@ -18,6 +19,14 @@ const Setting = () => { | ||||
|   ]; | ||||
|  | ||||
|   if (isRoot()) { | ||||
|     panes.push({ | ||||
|       menuItem: '运营设置', | ||||
|       render: () => ( | ||||
|         <Tab.Pane attached={false}> | ||||
|           <OperationSetting /> | ||||
|         </Tab.Pane> | ||||
|       ) | ||||
|     }); | ||||
|     panes.push({ | ||||
|       menuItem: '系统设置', | ||||
|       render: () => ( | ||||
|   | ||||
| @@ -1,7 +1,8 @@ | ||||
| import React, { useEffect, useState } from 'react'; | ||||
| import { Button, Form, Header, Segment } from 'semantic-ui-react'; | ||||
| import { Button, Form, Header, Message, Segment } from 'semantic-ui-react'; | ||||
| import { useParams } from 'react-router-dom'; | ||||
| import { API, showError, showSuccess, timestamp2string } from '../../helpers'; | ||||
| import { renderQuota, renderQuotaWithPrompt } from '../../helpers/render'; | ||||
|  | ||||
| const EditToken = () => { | ||||
|   const params = useParams(); | ||||
| @@ -10,11 +11,12 @@ const EditToken = () => { | ||||
|   const [loading, setLoading] = useState(isEdit); | ||||
|   const originInputs = { | ||||
|     name: '', | ||||
|     remain_times: -1, | ||||
|     expired_time: -1 | ||||
|     remain_quota: 0, | ||||
|     expired_time: -1, | ||||
|     unlimited_quota: false | ||||
|   }; | ||||
|   const [inputs, setInputs] = useState(originInputs); | ||||
|   const { name, remain_times, expired_time } = inputs; | ||||
|   const { name, remain_quota, expired_time, unlimited_quota } = inputs; | ||||
|  | ||||
|   const handleInputChange = (e, { name, value }) => { | ||||
|     setInputs((inputs) => ({ ...inputs, [name]: value })); | ||||
| @@ -35,6 +37,10 @@ const EditToken = () => { | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   const setUnlimitedQuota = () => { | ||||
|     setInputs({ ...inputs, unlimited_quota: !unlimited_quota }); | ||||
|   }; | ||||
|  | ||||
|   const loadToken = async () => { | ||||
|     let res = await API.get(`/api/token/${tokenId}`); | ||||
|     const { success, message, data } = res.data; | ||||
| @@ -57,7 +63,7 @@ const EditToken = () => { | ||||
|   const submit = async () => { | ||||
|     if (!isEdit && inputs.name === '') return; | ||||
|     let localInputs = inputs; | ||||
|     localInputs.remain_times = parseInt(localInputs.remain_times); | ||||
|     localInputs.remain_quota = parseInt(localInputs.remain_quota); | ||||
|     if (localInputs.expired_time !== -1) { | ||||
|       let time = Date.parse(localInputs.expired_time); | ||||
|       if (isNaN(time)) { | ||||
| @@ -88,8 +94,8 @@ const EditToken = () => { | ||||
|   return ( | ||||
|     <> | ||||
|       <Segment loading={loading}> | ||||
|         <Header as='h3'>{isEdit ? "更新令牌信息" : "创建新的令牌"}</Header> | ||||
|         <Form autoComplete='off'> | ||||
|         <Header as='h3'>{isEdit ? '更新令牌信息' : '创建新的令牌'}</Header> | ||||
|         <Form autoComplete='new-password'> | ||||
|           <Form.Field> | ||||
|             <Form.Input | ||||
|               label='名称' | ||||
| @@ -97,21 +103,10 @@ const EditToken = () => { | ||||
|               placeholder={'请输入名称'} | ||||
|               onChange={handleInputChange} | ||||
|               value={name} | ||||
|               autoComplete='off' | ||||
|               autoComplete='new-password' | ||||
|               required={!isEdit} | ||||
|             /> | ||||
|           </Form.Field> | ||||
|           <Form.Field> | ||||
|             <Form.Input | ||||
|               label='剩余次数' | ||||
|               name='remain_times' | ||||
|               placeholder={'请输入剩余次数,-1 表示无限制'} | ||||
|               onChange={handleInputChange} | ||||
|               value={remain_times} | ||||
|               autoComplete='off' | ||||
|               type='number' | ||||
|             /> | ||||
|           </Form.Field> | ||||
|           <Form.Field> | ||||
|             <Form.Input | ||||
|               label='过期时间' | ||||
| @@ -119,26 +114,44 @@ const EditToken = () => { | ||||
|               placeholder={'请输入过期时间,格式为 yyyy-MM-dd HH:mm:ss,-1 表示无限制'} | ||||
|               onChange={handleInputChange} | ||||
|               value={expired_time} | ||||
|               autoComplete='off' | ||||
|               autoComplete='new-password' | ||||
|               type='datetime-local' | ||||
|             /> | ||||
|           </Form.Field> | ||||
|           <div style={{ lineHeight: '40px' }}> | ||||
|             <Button type={'button'} onClick={() => { | ||||
|               setExpiredTime(0, 0, 0, 0); | ||||
|             }}>永不过期</Button> | ||||
|             <Button type={'button'} onClick={() => { | ||||
|               setExpiredTime(1, 0, 0, 0); | ||||
|             }}>一个月后过期</Button> | ||||
|             <Button type={'button'} onClick={() => { | ||||
|               setExpiredTime(0, 1, 0, 0); | ||||
|             }}>一天后过期</Button> | ||||
|             <Button type={'button'} onClick={() => { | ||||
|               setExpiredTime(0, 0, 1, 0); | ||||
|             }}>一小时后过期</Button> | ||||
|             <Button type={'button'} onClick={() => { | ||||
|               setExpiredTime(0, 0, 0, 1); | ||||
|             }}>一分钟后过期</Button> | ||||
|           </div> | ||||
|           <Message>注意,令牌的额度仅用于限制令牌本身的最大额度使用量,实际的使用受到账户的剩余额度限制。</Message> | ||||
|           <Form.Field> | ||||
|             <Form.Input | ||||
|               label={`额度${renderQuotaWithPrompt(remain_quota)}`} | ||||
|               name='remain_quota' | ||||
|               placeholder={'请输入额度'} | ||||
|               onChange={handleInputChange} | ||||
|               value={remain_quota} | ||||
|               autoComplete='new-password' | ||||
|               type='number' | ||||
|               disabled={unlimited_quota} | ||||
|             /> | ||||
|           </Form.Field> | ||||
|           <Button type={'button'} onClick={() => { | ||||
|             setExpiredTime(0, 0, 0, 0); | ||||
|           }}>永不过期</Button> | ||||
|           <Button type={'button'} onClick={() => { | ||||
|             setExpiredTime(1, 0, 0, 0); | ||||
|           }}>一个月后过期</Button> | ||||
|           <Button type={'button'} onClick={() => { | ||||
|             setExpiredTime(0, 1, 0, 0); | ||||
|           }}>一天后过期</Button> | ||||
|           <Button type={'button'} onClick={() => { | ||||
|             setExpiredTime(0, 0, 1, 0); | ||||
|           }}>一小时后过期</Button> | ||||
|           <Button type={'button'} onClick={() => { | ||||
|             setExpiredTime(0, 0, 0, 1); | ||||
|           }}>一分钟后过期</Button> | ||||
|           <Button onClick={submit}>提交</Button> | ||||
|             setUnlimitedQuota(); | ||||
|           }}>{unlimited_quota ? '取消无限额度' : '设置为无限额度'}</Button> | ||||
|           <Button positive onClick={submit}>提交</Button> | ||||
|         </Form> | ||||
|       </Segment> | ||||
|     </> | ||||
|   | ||||
							
								
								
									
										96
									
								
								web/src/pages/TopUp/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										96
									
								
								web/src/pages/TopUp/index.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,96 @@ | ||||
| import React, { useEffect, useState } from 'react'; | ||||
| import { Button, Form, Grid, Header, Segment, Statistic } from 'semantic-ui-react'; | ||||
| import { API, showError, showInfo, showSuccess } from '../../helpers'; | ||||
| import { renderQuota } from '../../helpers/render'; | ||||
|  | ||||
| const TopUp = () => { | ||||
|   const [redemptionCode, setRedemptionCode] = useState(''); | ||||
|   const [topUpLink, setTopUpLink] = useState(''); | ||||
|   const [userQuota, setUserQuota] = useState(0); | ||||
|  | ||||
|   const topUp = async () => { | ||||
|     if (redemptionCode === '') { | ||||
|       showInfo('请输入充值码!') | ||||
|       return; | ||||
|     } | ||||
|     const res = await API.post('/api/user/topup', { | ||||
|       key: redemptionCode | ||||
|     }); | ||||
|     const { success, message, data } = res.data; | ||||
|     if (success) { | ||||
|       showSuccess('充值成功!'); | ||||
|       setUserQuota((quota) => { | ||||
|         return quota + data; | ||||
|       }); | ||||
|       setRedemptionCode(''); | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   const openTopUpLink = () => { | ||||
|     if (!topUpLink) { | ||||
|       showError('超级管理员未设置充值链接!'); | ||||
|       return; | ||||
|     } | ||||
|     window.open(topUpLink, '_blank'); | ||||
|   }; | ||||
|  | ||||
|   const getUserQuota = async ()=>{ | ||||
|     let res  = await API.get(`/api/user/self`); | ||||
|     const {success, message, data} = res.data; | ||||
|     if (success) { | ||||
|       setUserQuota(data.quota); | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   useEffect(() => { | ||||
|     let status = localStorage.getItem('status'); | ||||
|     if (status) { | ||||
|       status = JSON.parse(status); | ||||
|       if (status.top_up_link) { | ||||
|         setTopUpLink(status.top_up_link); | ||||
|       } | ||||
|     } | ||||
|     getUserQuota().then(); | ||||
|   }, []); | ||||
|  | ||||
|   return ( | ||||
|     <Segment> | ||||
|       <Header as='h3'>充值额度</Header> | ||||
|       <Grid columns={2} stackable> | ||||
|         <Grid.Column> | ||||
|           <Form> | ||||
|             <Form.Input | ||||
|               placeholder='兑换码' | ||||
|               name='redemptionCode' | ||||
|               value={redemptionCode} | ||||
|               onChange={(e) => { | ||||
|                 setRedemptionCode(e.target.value); | ||||
|               }} | ||||
|             /> | ||||
|             <Button color='green' onClick={openTopUpLink}> | ||||
|               获取兑换码 | ||||
|             </Button> | ||||
|             <Button color='yellow' onClick={topUp}> | ||||
|               充值 | ||||
|             </Button> | ||||
|           </Form> | ||||
|         </Grid.Column> | ||||
|         <Grid.Column> | ||||
|           <Statistic.Group widths='one'> | ||||
|             <Statistic> | ||||
|               <Statistic.Value>{renderQuota(userQuota)}</Statistic.Value> | ||||
|               <Statistic.Label>剩余额度</Statistic.Label> | ||||
|             </Statistic> | ||||
|           </Statistic.Group> | ||||
|         </Grid.Column> | ||||
|       </Grid> | ||||
|     </Segment> | ||||
|   ); | ||||
| }; | ||||
|  | ||||
|  | ||||
| export default TopUp; | ||||
| @@ -65,7 +65,7 @@ const AddUser = () => { | ||||
|               required | ||||
|             /> | ||||
|           </Form.Field> | ||||
|           <Button type={'submit'} onClick={submit}> | ||||
|           <Button positive type={'submit'} onClick={submit}> | ||||
|             提交 | ||||
|           </Button> | ||||
|         </Form> | ||||
|   | ||||
| @@ -14,12 +14,27 @@ const EditUser = () => { | ||||
|     github_id: '', | ||||
|     wechat_id: '', | ||||
|     email: '', | ||||
|     quota: 0, | ||||
|     group: 'default' | ||||
|   }); | ||||
|   const { username, display_name, password, github_id, wechat_id, email } = | ||||
|   const [groupOptions, setGroupOptions] = useState([]); | ||||
|   const { username, display_name, password, github_id, wechat_id, email, quota, group } = | ||||
|     inputs; | ||||
|   const handleInputChange = (e, { name, value }) => { | ||||
|     setInputs((inputs) => ({ ...inputs, [name]: value })); | ||||
|   }; | ||||
|   const fetchGroups = async () => { | ||||
|     try { | ||||
|       let res = await API.get(`/api/group/`); | ||||
|       setGroupOptions(res.data.data.map((group) => ({ | ||||
|         key: group, | ||||
|         text: group, | ||||
|         value: group, | ||||
|       }))); | ||||
|     } catch (error) { | ||||
|       showError(error.message); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   const loadUser = async () => { | ||||
|     let res = undefined; | ||||
| @@ -39,12 +54,19 @@ const EditUser = () => { | ||||
|   }; | ||||
|   useEffect(() => { | ||||
|     loadUser().then(); | ||||
|     if (userId) { | ||||
|       fetchGroups().then(); | ||||
|     } | ||||
|   }, []); | ||||
|  | ||||
|   const submit = async () => { | ||||
|     let res = undefined; | ||||
|     if (userId) { | ||||
|       res = await API.put(`/api/user/`, { ...inputs, id: parseInt(userId) }); | ||||
|       let data = { ...inputs, id: parseInt(userId) }; | ||||
|       if (typeof data.quota === 'string') { | ||||
|         data.quota = parseInt(data.quota); | ||||
|       } | ||||
|       res = await API.put(`/api/user/`, data); | ||||
|     } else { | ||||
|       res = await API.put(`/api/user/self`, inputs); | ||||
|     } | ||||
| @@ -60,7 +82,7 @@ const EditUser = () => { | ||||
|     <> | ||||
|       <Segment loading={loading}> | ||||
|         <Header as='h3'>更新用户信息</Header> | ||||
|         <Form autoComplete='off'> | ||||
|         <Form autoComplete='new-password'> | ||||
|           <Form.Field> | ||||
|             <Form.Input | ||||
|               label='用户名' | ||||
| @@ -68,7 +90,7 @@ const EditUser = () => { | ||||
|               placeholder={'请输入新的用户名'} | ||||
|               onChange={handleInputChange} | ||||
|               value={username} | ||||
|               autoComplete='off' | ||||
|               autoComplete='new-password' | ||||
|             /> | ||||
|           </Form.Field> | ||||
|           <Form.Field> | ||||
| @@ -79,7 +101,7 @@ const EditUser = () => { | ||||
|               placeholder={'请输入新的密码'} | ||||
|               onChange={handleInputChange} | ||||
|               value={password} | ||||
|               autoComplete='off' | ||||
|               autoComplete='new-password' | ||||
|             /> | ||||
|           </Form.Field> | ||||
|           <Form.Field> | ||||
| @@ -89,15 +111,46 @@ const EditUser = () => { | ||||
|               placeholder={'请输入新的显示名称'} | ||||
|               onChange={handleInputChange} | ||||
|               value={display_name} | ||||
|               autoComplete='off' | ||||
|               autoComplete='new-password' | ||||
|             /> | ||||
|           </Form.Field> | ||||
|           { | ||||
|             userId && <> | ||||
|               <Form.Field> | ||||
|                 <Form.Dropdown | ||||
|                   label='分组' | ||||
|                   placeholder={'请选择分组'} | ||||
|                   name='group' | ||||
|                   fluid | ||||
|                   search | ||||
|                   selection | ||||
|                   allowAdditions | ||||
|                   additionLabel={'请在系统设置页面编辑分组倍率以添加新的分组:'} | ||||
|                   onChange={handleInputChange} | ||||
|                   value={inputs.group} | ||||
|                   autoComplete='new-password' | ||||
|                   options={groupOptions} | ||||
|                 /> | ||||
|               </Form.Field> | ||||
|               <Form.Field> | ||||
|                 <Form.Input | ||||
|                   label='剩余额度' | ||||
|                   name='quota' | ||||
|                   placeholder={'请输入新的剩余额度'} | ||||
|                   onChange={handleInputChange} | ||||
|                   value={quota} | ||||
|                   type={'number'} | ||||
|                   autoComplete='new-password' | ||||
|                 /> | ||||
|               </Form.Field> | ||||
|             </> | ||||
|           } | ||||
|           <Form.Field> | ||||
|             <Form.Input | ||||
|               label='已绑定的 GitHub 账户' | ||||
|               name='github_id' | ||||
|               value={github_id} | ||||
|               autoComplete='off' | ||||
|               autoComplete='new-password' | ||||
|               placeholder='此项只读,需要用户通过个人设置页面的相关绑定按钮进行绑定,不可直接修改' | ||||
|               readOnly | ||||
|             /> | ||||
| @@ -107,7 +160,7 @@ const EditUser = () => { | ||||
|               label='已绑定的微信账户' | ||||
|               name='wechat_id' | ||||
|               value={wechat_id} | ||||
|               autoComplete='off' | ||||
|               autoComplete='new-password' | ||||
|               placeholder='此项只读,需要用户通过个人设置页面的相关绑定按钮进行绑定,不可直接修改' | ||||
|               readOnly | ||||
|             /> | ||||
| @@ -117,12 +170,12 @@ const EditUser = () => { | ||||
|               label='已绑定的邮箱账户' | ||||
|               name='email' | ||||
|               value={email} | ||||
|               autoComplete='off' | ||||
|               autoComplete='new-password' | ||||
|               placeholder='此项只读,需要用户通过个人设置页面的相关绑定按钮进行绑定,不可直接修改' | ||||
|               readOnly | ||||
|             /> | ||||
|           </Form.Field> | ||||
|           <Button onClick={submit}>提交</Button> | ||||
|           <Button positive onClick={submit}>提交</Button> | ||||
|         </Form> | ||||
|       </Segment> | ||||
|     </> | ||||
|   | ||||
		Reference in New Issue
	
	Block a user