mirror of
				https://github.com/songquanpeng/one-api.git
				synced 2025-10-31 13:53:41 +08:00 
			
		
		
		
	Compare commits
	
		
			75 Commits
		
	
	
		
			v0.6.2-alp
			...
			v0.6.5-alp
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | ed70881a58 | ||
|  | 8b9fa3d6e4 | ||
|  | 8b9813d63b | ||
|  | dc7aaf2de5 | ||
|  | 065da8ef8c | ||
|  | e3cfb1fa52 | ||
|  | f89ae5ad58 | ||
|  | 06a3fc5421 | ||
|  | a9c464ec5a | ||
|  | 3f3c13c98c | ||
|  | 2ba28c72cb | ||
|  | 5e81e19bc8 | ||
|  | 96d7a99312 | ||
|  | 24be9de098 | ||
|  | 5b349efff9 | ||
|  | f76c46d648 | ||
|  | cdfdeea3b4 | ||
|  | 56ddbb842a | ||
|  | 99f81a267c | ||
|  | c243cd5535 | ||
|  | e96b173abe | ||
|  | 4ae311e964 | ||
|  | b14cb748d8 | ||
|  | ade19ba4a2 | ||
|  | 4d86d021c4 | ||
|  | 7a44adb5a7 | ||
|  | 9821bc7281 | ||
|  | 08831881f1 | ||
|  | 0eb2272bb7 | ||
|  | 704ec1a827 | ||
|  | 1d7470d6ad | ||
|  | 1185303346 | ||
|  | c212fcf8d7 | ||
|  | c285e000cc | ||
|  | d25ed4c009 | ||
|  | 7400885fbb | ||
|  | 11af81eb39 | ||
|  | 205aba694f | ||
|  | 8dac3afebc | ||
|  | a07791bf93 | ||
|  | 4bb662c0e4 | ||
|  | 4998d58319 | ||
|  | 190203cf8f | ||
|  | 6325c8e0b4 | ||
|  | b204f6d82b | ||
|  | 752639560f | ||
|  | 996f4d99dd | ||
|  | ebfee3b46c | ||
|  | 3e2e805d61 | ||
|  | 3edf7247c4 | ||
|  | 0926b6206b | ||
|  | 7cd57f3125 | ||
|  | 66efabd5ae | ||
|  | 8ede66a896 | ||
|  | b169173860 | ||
|  | f33555ae78 | ||
|  | c28ec10795 | ||
|  | e3767cbb07 | ||
|  | be9eb59fbb | ||
|  | 89e111ac69 | ||
|  | 2dcef85285 | ||
|  | 79d0cd378a | ||
|  | e99150bdb9 | ||
|  | a72e5fcc9e | ||
|  | 0710f8cd66 | ||
|  | 49cad7d4a5 | ||
|  | a90161cf00 | ||
|  | a45fc7d736 | ||
|  | 45940dcb12 | ||
|  | 969042b001 | ||
|  | 7e7369dbc4 | ||
|  | e54e647170 | ||
|  | 358920c858 | ||
|  | 1ea598c773 | ||
|  | 796be42487 | 
							
								
								
									
										7
									
								
								.github/workflows/docker-image-amd64-en.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										7
									
								
								.github/workflows/docker-image-amd64-en.yml
									
									
									
									
										vendored
									
									
								
							| @@ -20,6 +20,13 @@ jobs: | ||||
|       - name: Check out the repo | ||||
|         uses: actions/checkout@v3 | ||||
|  | ||||
|       - name: Check repository URL | ||||
|         run: | | ||||
|           REPO_URL=$(git config --get remote.origin.url) | ||||
|           if [[ $REPO_URL == *"pro" ]]; then | ||||
|             exit 1 | ||||
|           fi       | ||||
|  | ||||
|       - name: Save version info | ||||
|         run: | | ||||
|           git describe --tags > VERSION  | ||||
|   | ||||
							
								
								
									
										7
									
								
								.github/workflows/docker-image-amd64.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										7
									
								
								.github/workflows/docker-image-amd64.yml
									
									
									
									
										vendored
									
									
								
							| @@ -20,6 +20,13 @@ jobs: | ||||
|       - name: Check out the repo | ||||
|         uses: actions/checkout@v3 | ||||
|  | ||||
|       - name: Check repository URL | ||||
|         run: | | ||||
|           REPO_URL=$(git config --get remote.origin.url) | ||||
|           if [[ $REPO_URL == *"pro" ]]; then | ||||
|             exit 1 | ||||
|           fi         | ||||
|  | ||||
|       - name: Save version info | ||||
|         run: | | ||||
|           git describe --tags > VERSION  | ||||
|   | ||||
							
								
								
									
										7
									
								
								.github/workflows/docker-image-arm64.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										7
									
								
								.github/workflows/docker-image-arm64.yml
									
									
									
									
										vendored
									
									
								
							| @@ -21,6 +21,13 @@ jobs: | ||||
|       - name: Check out the repo | ||||
|         uses: actions/checkout@v3 | ||||
|  | ||||
|       - name: Check repository URL | ||||
|         run: | | ||||
|           REPO_URL=$(git config --get remote.origin.url) | ||||
|           if [[ $REPO_URL == *"pro" ]]; then | ||||
|             exit 1 | ||||
|           fi | ||||
|  | ||||
|       - name: Save version info | ||||
|         run: | | ||||
|           git describe --tags > VERSION  | ||||
|   | ||||
							
								
								
									
										6
									
								
								.github/workflows/linux-release.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.github/workflows/linux-release.yml
									
									
									
									
										vendored
									
									
								
							| @@ -20,6 +20,12 @@ jobs: | ||||
|         uses: actions/checkout@v3 | ||||
|         with: | ||||
|           fetch-depth: 0 | ||||
|       - name: Check repository URL | ||||
|         run: | | ||||
|           REPO_URL=$(git config --get remote.origin.url) | ||||
|           if [[ $REPO_URL == *"pro" ]]; then | ||||
|             exit 1 | ||||
|           fi | ||||
|       - uses: actions/setup-node@v3 | ||||
|         with: | ||||
|           node-version: 16 | ||||
|   | ||||
							
								
								
									
										6
									
								
								.github/workflows/macos-release.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.github/workflows/macos-release.yml
									
									
									
									
										vendored
									
									
								
							| @@ -20,6 +20,12 @@ jobs: | ||||
|         uses: actions/checkout@v3 | ||||
|         with: | ||||
|           fetch-depth: 0 | ||||
|       - name: Check repository URL | ||||
|         run: | | ||||
|           REPO_URL=$(git config --get remote.origin.url) | ||||
|           if [[ $REPO_URL == *"pro" ]]; then | ||||
|             exit 1 | ||||
|           fi | ||||
|       - uses: actions/setup-node@v3 | ||||
|         with: | ||||
|           node-version: 16 | ||||
|   | ||||
							
								
								
									
										6
									
								
								.github/workflows/windows-release.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.github/workflows/windows-release.yml
									
									
									
									
										vendored
									
									
								
							| @@ -23,6 +23,12 @@ jobs: | ||||
|         uses: actions/checkout@v3 | ||||
|         with: | ||||
|           fetch-depth: 0 | ||||
|       - name: Check repository URL | ||||
|         run: | | ||||
|           REPO_URL=$(git config --get remote.origin.url) | ||||
|           if [[ $REPO_URL == *"pro" ]]; then | ||||
|             exit 1 | ||||
|           fi | ||||
|       - uses: actions/setup-node@v3 | ||||
|         with: | ||||
|           node-version: 16 | ||||
|   | ||||
| @@ -12,6 +12,10 @@ WORKDIR /web/berry | ||||
| RUN npm install | ||||
| RUN DISABLE_ESLINT_PLUGIN='true' REACT_APP_VERSION=$(cat VERSION) npm run build | ||||
|  | ||||
| WORKDIR /web/air | ||||
| RUN npm install | ||||
| RUN DISABLE_ESLINT_PLUGIN='true' REACT_APP_VERSION=$(cat VERSION) npm run build | ||||
|  | ||||
| FROM golang AS builder2 | ||||
|  | ||||
| ENV GO111MODULE=on \ | ||||
|   | ||||
							
								
								
									
										14
									
								
								README.en.md
									
									
									
									
									
								
							
							
						
						
									
										14
									
								
								README.en.md
									
									
									
									
									
								
							| @@ -241,17 +241,19 @@ If the channel ID is not provided, load balancing will be used to distribute the | ||||
|     + Example: `SESSION_SECRET=random_string` | ||||
| 3. `SQL_DSN`: When set, the specified database will be used instead of SQLite. Please use MySQL version 8.0. | ||||
|     + Example: `SQL_DSN=root:123456@tcp(localhost:3306)/oneapi` | ||||
| 4. `FRONTEND_BASE_URL`: When set, the specified frontend address will be used instead of the backend address. | ||||
| 4. `LOG_SQL_DSN`: When set, a separate database will be used for the `logs` table; please use MySQL or PostgreSQL. | ||||
|     + Example: `LOG_SQL_DSN=root:123456@tcp(localhost:3306)/oneapi-logs` | ||||
| 5. `FRONTEND_BASE_URL`: When set, the specified frontend address will be used instead of the backend address. | ||||
|     + Example: `FRONTEND_BASE_URL=https://openai.justsong.cn` | ||||
| 5. `SYNC_FREQUENCY`: When set, the system will periodically sync configurations from the database, with the unit in seconds. If not set, no sync will happen. | ||||
| 6. `SYNC_FREQUENCY`: When set, the system will periodically sync configurations from the database, with the unit in seconds. If not set, no sync will happen. | ||||
|     + Example: `SYNC_FREQUENCY=60` | ||||
| 6. `NODE_TYPE`: When set, specifies the node type. Valid values are `master` and `slave`. If not set, it defaults to `master`. | ||||
| 7. `NODE_TYPE`: When set, specifies the node type. Valid values are `master` and `slave`. If not set, it defaults to `master`. | ||||
|     + Example: `NODE_TYPE=slave` | ||||
| 7. `CHANNEL_UPDATE_FREQUENCY`: When set, it periodically updates the channel balances, with the unit in minutes. If not set, no update will happen. | ||||
| 8. `CHANNEL_UPDATE_FREQUENCY`: When set, it periodically updates the channel balances, with the unit in minutes. If not set, no update will happen. | ||||
|     + Example: `CHANNEL_UPDATE_FREQUENCY=1440` | ||||
| 8. `CHANNEL_TEST_FREQUENCY`: When set, it periodically tests the channels, with the unit in minutes. If not set, no test will happen. | ||||
| 9. `CHANNEL_TEST_FREQUENCY`: When set, it periodically tests the channels, with the unit in minutes. If not set, no test will happen. | ||||
|     + Example: `CHANNEL_TEST_FREQUENCY=1440` | ||||
| 9. `POLLING_INTERVAL`: The time interval (in seconds) between requests when updating channel balances and testing channel availability. Default is no interval. | ||||
| 10. `POLLING_INTERVAL`: The time interval (in seconds) between requests when updating channel balances and testing channel availability. Default is no interval. | ||||
|     + Example: `POLLING_INTERVAL=5` | ||||
|  | ||||
| ### Command Line Parameters | ||||
|   | ||||
							
								
								
									
										13
									
								
								README.ja.md
									
									
									
									
									
								
							
							
						
						
									
										13
									
								
								README.ja.md
									
									
									
									
									
								
							| @@ -242,17 +242,18 @@ graph LR | ||||
|     + 例: `SESSION_SECRET=random_string` | ||||
| 3. `SQL_DSN`: 設定すると、SQLite の代わりに指定したデータベースが使用されます。MySQL バージョン 8.0 を使用してください。 | ||||
|     + 例: `SQL_DSN=root:123456@tcp(localhost:3306)/oneapi` | ||||
| 4. `FRONTEND_BASE_URL`: 設定されると、バックエンドアドレスではなく、指定されたフロントエンドアドレスが使われる。 | ||||
| 4. `LOG_SQL_DSN`: を設定すると、`logs`テーブルには独立したデータベースが使用されます。MySQLまたはPostgreSQLを使用してください。 | ||||
| 5. `FRONTEND_BASE_URL`: 設定されると、バックエンドアドレスではなく、指定されたフロントエンドアドレスが使われる。 | ||||
|     + 例: `FRONTEND_BASE_URL=https://openai.justsong.cn` | ||||
| 5. `SYNC_FREQUENCY`: 設定された場合、システムは定期的にデータベースからコンフィグを秒単位で同期する。設定されていない場合、同期は行われません。 | ||||
| 6. `SYNC_FREQUENCY`: 設定された場合、システムは定期的にデータベースからコンフィグを秒単位で同期する。設定されていない場合、同期は行われません。 | ||||
|     + 例: `SYNC_FREQUENCY=60` | ||||
| 6. `NODE_TYPE`: 設定すると、ノードのタイプを指定する。有効な値は `master` と `slave` である。設定されていない場合、デフォルトは `master`。 | ||||
| 7. `NODE_TYPE`: 設定すると、ノードのタイプを指定する。有効な値は `master` と `slave` である。設定されていない場合、デフォルトは `master`。 | ||||
|     + 例: `NODE_TYPE=slave` | ||||
| 7. `CHANNEL_UPDATE_FREQUENCY`: 設定すると、チャンネル残高を分単位で定期的に更新する。設定されていない場合、更新は行われません。 | ||||
| 8. `CHANNEL_UPDATE_FREQUENCY`: 設定すると、チャンネル残高を分単位で定期的に更新する。設定されていない場合、更新は行われません。 | ||||
|     + 例: `CHANNEL_UPDATE_FREQUENCY=1440` | ||||
| 8. `CHANNEL_TEST_FREQUENCY`: 設定すると、チャンネルを定期的にテストする。設定されていない場合、テストは行われません。 | ||||
| 9. `CHANNEL_TEST_FREQUENCY`: 設定すると、チャンネルを定期的にテストする。設定されていない場合、テストは行われません。 | ||||
|     + 例: `CHANNEL_TEST_FREQUENCY=1440` | ||||
| 9. `POLLING_INTERVAL`: チャネル残高の更新とチャネルの可用性をテストするときのリクエスト間の時間間隔 (秒)。デフォルトは間隔なし。 | ||||
| 10. `POLLING_INTERVAL`: チャネル残高の更新とチャネルの可用性をテストするときのリクエスト間の時間間隔 (秒)。デフォルトは間隔なし。 | ||||
|     + 例: `POLLING_INTERVAL=5` | ||||
|  | ||||
| ### コマンドラインパラメータ | ||||
|   | ||||
							
								
								
									
										48
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										48
									
								
								README.md
									
									
									
									
									
								
							| @@ -79,13 +79,15 @@ _✨ 通过标准的 OpenAI API 格式访问所有的大模型,开箱即用  | ||||
|    + [ ] [字节云雀大模型](https://www.volcengine.com/product/ark) (WIP) | ||||
|    + [x] [MINIMAX](https://api.minimax.chat/) | ||||
|    + [x] [Groq](https://wow.groq.com/) | ||||
|    + [x] [Ollama](https://github.com/ollama/ollama) | ||||
|    + [x] [零一万物](https://platform.lingyiwanwu.com/) | ||||
| 2. 支持配置镜像以及众多[第三方代理服务](https://iamazing.cn/page/openai-api-third-party-services)。 | ||||
| 3. 支持通过**负载均衡**的方式访问多个渠道。 | ||||
| 4. 支持 **stream 模式**,可以通过流式传输实现打字机效果。 | ||||
| 5. 支持**多机部署**,[详见此处](#多机部署)。 | ||||
| 6. 支持**令牌管理**,设置令牌的过期时间和额度。 | ||||
| 7. 支持**兑换码管理**,支持批量生成和导出兑换码,可使用兑换码为账户进行充值。 | ||||
| 8. 支持**通道管理**,批量创建通道。 | ||||
| 8. 支持**渠道管理**,批量创建渠道。 | ||||
| 9. 支持**用户分组**以及**渠道分组**,支持为不同分组设置不同的倍率。 | ||||
| 10. 支持渠道**设置模型列表**。 | ||||
| 11. 支持**查看额度明细**。 | ||||
| @@ -347,38 +349,40 @@ graph LR | ||||
|      + `SQL_MAX_OPEN_CONNS`:最大打开连接数,默认为 `1000`。 | ||||
|        + 如果报错 `Error 1040: Too many connections`,请适当减小该值。 | ||||
|      + `SQL_CONN_MAX_LIFETIME`:连接的最大生命周期,默认为 `60`,单位分钟。 | ||||
| 4. `FRONTEND_BASE_URL`:设置之后将重定向页面请求到指定的地址,仅限从服务器设置。 | ||||
| 4. `LOG_SQL_DSN`:设置之后将为 `logs` 表使用独立的数据库,请使用 MySQL 或 PostgreSQL。 | ||||
| 5. `FRONTEND_BASE_URL`:设置之后将重定向页面请求到指定的地址,仅限从服务器设置。 | ||||
|    + 例子:`FRONTEND_BASE_URL=https://openai.justsong.cn` | ||||
| 5. `MEMORY_CACHE_ENABLED`:启用内存缓存,会导致用户额度的更新存在一定的延迟,可选值为 `true` 和 `false`,未设置则默认为 `false`。 | ||||
| 6. `MEMORY_CACHE_ENABLED`:启用内存缓存,会导致用户额度的更新存在一定的延迟,可选值为 `true` 和 `false`,未设置则默认为 `false`。 | ||||
|    + 例子:`MEMORY_CACHE_ENABLED=true` | ||||
| 6. `SYNC_FREQUENCY`:在启用缓存的情况下与数据库同步配置的频率,单位为秒,默认为 `600` 秒。 | ||||
| 7. `SYNC_FREQUENCY`:在启用缓存的情况下与数据库同步配置的频率,单位为秒,默认为 `600` 秒。 | ||||
|    + 例子:`SYNC_FREQUENCY=60` | ||||
| 7. `NODE_TYPE`:设置之后将指定节点类型,可选值为 `master` 和 `slave`,未设置则默认为 `master`。 | ||||
| 8. `NODE_TYPE`:设置之后将指定节点类型,可选值为 `master` 和 `slave`,未设置则默认为 `master`。 | ||||
|    + 例子:`NODE_TYPE=slave` | ||||
| 8. `CHANNEL_UPDATE_FREQUENCY`:设置之后将定期更新渠道余额,单位为分钟,未设置则不进行更新。 | ||||
| 9. `CHANNEL_UPDATE_FREQUENCY`:设置之后将定期更新渠道余额,单位为分钟,未设置则不进行更新。 | ||||
|    + 例子:`CHANNEL_UPDATE_FREQUENCY=1440` | ||||
| 9. `CHANNEL_TEST_FREQUENCY`:设置之后将定期检查渠道,单位为分钟,未设置则不进行检查。 | ||||
| 10. `CHANNEL_TEST_FREQUENCY`:设置之后将定期检查渠道,单位为分钟,未设置则不进行检查。 | ||||
|    + 例子:`CHANNEL_TEST_FREQUENCY=1440` | ||||
| 10. `POLLING_INTERVAL`:批量更新渠道余额以及测试可用性时的请求间隔,单位为秒,默认无间隔。 | ||||
| 11. `POLLING_INTERVAL`:批量更新渠道余额以及测试可用性时的请求间隔,单位为秒,默认无间隔。 | ||||
|     + 例子:`POLLING_INTERVAL=5` | ||||
| 11. `BATCH_UPDATE_ENABLED`:启用数据库批量更新聚合,会导致用户额度的更新存在一定的延迟可选值为 `true` 和 `false`,未设置则默认为 `false`。 | ||||
| 12. `BATCH_UPDATE_ENABLED`:启用数据库批量更新聚合,会导致用户额度的更新存在一定的延迟可选值为 `true` 和 `false`,未设置则默认为 `false`。 | ||||
|     + 例子:`BATCH_UPDATE_ENABLED=true` | ||||
|     + 如果你遇到了数据库连接数过多的问题,可以尝试启用该选项。 | ||||
| 12. `BATCH_UPDATE_INTERVAL=5`:批量更新聚合的时间间隔,单位为秒,默认为 `5`。 | ||||
| 13. `BATCH_UPDATE_INTERVAL=5`:批量更新聚合的时间间隔,单位为秒,默认为 `5`。 | ||||
|     + 例子:`BATCH_UPDATE_INTERVAL=5` | ||||
| 13. 请求频率限制: | ||||
| 14. 请求频率限制: | ||||
|     + `GLOBAL_API_RATE_LIMIT`:全局 API 速率限制(除中继请求外),单 ip 三分钟内的最大请求数,默认为 `180`。 | ||||
|     + `GLOBAL_WEB_RATE_LIMIT`:全局 Web 速率限制,单 ip 三分钟内的最大请求数,默认为 `60`。 | ||||
| 14. 编码器缓存设置: | ||||
| 15. 编码器缓存设置: | ||||
|     + `TIKTOKEN_CACHE_DIR`:默认程序启动时会联网下载一些通用的词元的编码,如:`gpt-3.5-turbo`,在一些网络环境不稳定,或者离线情况,可能会导致启动有问题,可以配置此目录缓存数据,可迁移到离线环境。 | ||||
|     + `DATA_GYM_CACHE_DIR`:目前该配置作用与 `TIKTOKEN_CACHE_DIR` 一致,但是优先级没有它高。 | ||||
| 15. `RELAY_TIMEOUT`:中继超时设置,单位为秒,默认不设置超时时间。 | ||||
| 16. `SQLITE_BUSY_TIMEOUT`:SQLite 锁等待超时设置,单位为毫秒,默认 `3000`。 | ||||
| 17. `GEMINI_SAFETY_SETTING`:Gemini 的安全设置,默认 `BLOCK_NONE`。 | ||||
| 18. `THEME`:系统的主题设置,默认为 `default`,具体可选值参考[此处](./web/README.md)。 | ||||
| 19. `ENABLE_METRIC`:是否根据请求成功率禁用渠道,默认不开启,可选值为 `true` 和 `false`。 | ||||
| 20. `METRIC_QUEUE_SIZE`:请求成功率统计队列大小,默认为 `10`。 | ||||
| 21. `METRIC_SUCCESS_RATE_THRESHOLD`:请求成功率阈值,默认为 `0.8`。 | ||||
| 16. `RELAY_TIMEOUT`:中继超时设置,单位为秒,默认不设置超时时间。 | ||||
| 17. `SQLITE_BUSY_TIMEOUT`:SQLite 锁等待超时设置,单位为毫秒,默认 `3000`。 | ||||
| 18. `GEMINI_SAFETY_SETTING`:Gemini 的安全设置,默认 `BLOCK_NONE`。 | ||||
| 19. `THEME`:系统的主题设置,默认为 `default`,具体可选值参考[此处](./web/README.md)。 | ||||
| 20. `ENABLE_METRIC`:是否根据请求成功率禁用渠道,默认不开启,可选值为 `true` 和 `false`。 | ||||
| 21. `METRIC_QUEUE_SIZE`:请求成功率统计队列大小,默认为 `10`。 | ||||
| 22. `METRIC_SUCCESS_RATE_THRESHOLD`:请求成功率阈值,默认为 `0.8`。 | ||||
| 23. `INITIAL_ROOT_TOKEN`:如果设置了该值,则在系统首次启动时会自动创建一个值为该环境变量值的 root 用户令牌。 | ||||
|  | ||||
| ### 命令行参数 | ||||
| 1. `--port <port_number>`: 指定服务器监听的端口号,默认为 `3000`。 | ||||
| @@ -417,7 +421,7 @@ https://openai.justsong.cn | ||||
|    + 检查你的接口地址和 API Key 有没有填对。 | ||||
|    + 检查是否启用了 HTTPS,浏览器会拦截 HTTPS 域名下的 HTTP 请求。 | ||||
| 6. 报错:`当前分组负载已饱和,请稍后再试` | ||||
|    + 上游通道 429 了。 | ||||
|    + 上游渠道 429 了。 | ||||
| 7. 升级之后我的数据会丢失吗? | ||||
|    + 如果使用 MySQL,不会。 | ||||
|    + 如果使用 SQLite,需要按照我所给的部署命令挂载 volume 持久化 one-api.db 数据库文件,否则容器重启后数据会丢失。 | ||||
| @@ -425,8 +429,8 @@ https://openai.justsong.cn | ||||
|    + 一般情况下不需要,系统将在初始化的时候自动调整。 | ||||
|    + 如果需要的话,我会在更新日志中说明,并给出脚本。 | ||||
| 9. 手动修改数据库后报错:`数据库一致性已被破坏,请联系管理员`? | ||||
|    + 这是检测到 ability 表里有些记录的通道 id 是不存在的,这大概率是因为你删了 channel 表里的记录但是没有同步在 ability 表里清理无效的通道。 | ||||
|    + 对于每一个通道,其所支持的模型都需要有一个专门的 ability 表的记录,表示该通道支持该模型。 | ||||
|    + 这是检测到 ability 表里有些记录的渠道 id 是不存在的,这大概率是因为你删了 channel 表里的记录但是没有同步在 ability 表里清理无效的渠道。 | ||||
|    + 对于每一个渠道,其所支持的模型都需要有一个专门的 ability 表的记录,表示该渠道支持该模型。 | ||||
|  | ||||
| ## 相关项目 | ||||
| * [FastGPT](https://github.com/labring/FastGPT): 基于 LLM 大语言模型的知识库问答系统 | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| package config | ||||
|  | ||||
| import ( | ||||
| 	"github.com/songquanpeng/one-api/common/helper" | ||||
| 	"github.com/songquanpeng/one-api/common/env" | ||||
| 	"os" | ||||
| 	"strconv" | ||||
| 	"sync" | ||||
| @@ -76,14 +76,14 @@ var MessagePusherToken = "" | ||||
| var TurnstileSiteKey = "" | ||||
| var TurnstileSecretKey = "" | ||||
|  | ||||
| var QuotaForNewUser = 0 | ||||
| var QuotaForInviter = 0 | ||||
| var QuotaForInvitee = 0 | ||||
| var QuotaForNewUser int64 = 0 | ||||
| var QuotaForInviter int64 = 0 | ||||
| var QuotaForInvitee int64 = 0 | ||||
| var ChannelDisableThreshold = 5.0 | ||||
| var AutomaticDisableChannelEnabled = false | ||||
| var AutomaticEnableChannelEnabled = false | ||||
| var QuotaRemindThreshold = 1000 | ||||
| var PreConsumedQuota = 500 | ||||
| var QuotaRemindThreshold int64 = 1000 | ||||
| var PreConsumedQuota int64 = 500 | ||||
| var ApproximateTokenEnabled = false | ||||
| var RetryTimes = 0 | ||||
|  | ||||
| @@ -94,28 +94,29 @@ var IsMasterNode = os.Getenv("NODE_TYPE") != "slave" | ||||
| var requestInterval, _ = strconv.Atoi(os.Getenv("POLLING_INTERVAL")) | ||||
| var RequestInterval = time.Duration(requestInterval) * time.Second | ||||
|  | ||||
| var SyncFrequency = helper.GetOrDefaultEnvInt("SYNC_FREQUENCY", 10*60) // unit is second | ||||
| var SyncFrequency = env.Int("SYNC_FREQUENCY", 10*60) // unit is second | ||||
|  | ||||
| var BatchUpdateEnabled = false | ||||
| var BatchUpdateInterval = helper.GetOrDefaultEnvInt("BATCH_UPDATE_INTERVAL", 5) | ||||
| var BatchUpdateInterval = env.Int("BATCH_UPDATE_INTERVAL", 5) | ||||
|  | ||||
| var RelayTimeout = helper.GetOrDefaultEnvInt("RELAY_TIMEOUT", 0) // unit is second | ||||
| var RelayTimeout = env.Int("RELAY_TIMEOUT", 0) // unit is second | ||||
|  | ||||
| var GeminiSafetySetting = helper.GetOrDefaultEnvString("GEMINI_SAFETY_SETTING", "BLOCK_NONE") | ||||
| var GeminiSafetySetting = env.String("GEMINI_SAFETY_SETTING", "BLOCK_NONE") | ||||
|  | ||||
| var Theme = helper.GetOrDefaultEnvString("THEME", "default") | ||||
| var Theme = env.String("THEME", "default") | ||||
| var ValidThemes = map[string]bool{ | ||||
| 	"default": true, | ||||
| 	"berry":   true, | ||||
| 	"air":     true, | ||||
| } | ||||
|  | ||||
| // All duration's unit is seconds | ||||
| // Shouldn't larger then RateLimitKeyExpirationDuration | ||||
| var ( | ||||
| 	GlobalApiRateLimitNum            = helper.GetOrDefaultEnvInt("GLOBAL_API_RATE_LIMIT", 180) | ||||
| 	GlobalApiRateLimitNum            = env.Int("GLOBAL_API_RATE_LIMIT", 180) | ||||
| 	GlobalApiRateLimitDuration int64 = 3 * 60 | ||||
|  | ||||
| 	GlobalWebRateLimitNum            = helper.GetOrDefaultEnvInt("GLOBAL_WEB_RATE_LIMIT", 60) | ||||
| 	GlobalWebRateLimitNum            = env.Int("GLOBAL_WEB_RATE_LIMIT", 60) | ||||
| 	GlobalWebRateLimitDuration int64 = 3 * 60 | ||||
|  | ||||
| 	UploadRateLimitNum            = 10 | ||||
| @@ -130,8 +131,10 @@ var ( | ||||
|  | ||||
| var RateLimitKeyExpirationDuration = 20 * time.Minute | ||||
|  | ||||
| var EnableMetric = helper.GetOrDefaultEnvBool("ENABLE_METRIC", false) | ||||
| var MetricQueueSize = helper.GetOrDefaultEnvInt("METRIC_QUEUE_SIZE", 10) | ||||
| var MetricSuccessRateThreshold = helper.GetOrDefaultEnvFloat64("METRIC_SUCCESS_RATE_THRESHOLD", 0.8) | ||||
| var MetricSuccessChanSize = helper.GetOrDefaultEnvInt("METRIC_SUCCESS_CHAN_SIZE", 1024) | ||||
| var MetricFailChanSize = helper.GetOrDefaultEnvInt("METRIC_FAIL_CHAN_SIZE", 128) | ||||
| var EnableMetric = env.Bool("ENABLE_METRIC", false) | ||||
| var MetricQueueSize = env.Int("METRIC_QUEUE_SIZE", 10) | ||||
| var MetricSuccessRateThreshold = env.Float64("METRIC_SUCCESS_RATE_THRESHOLD", 0.8) | ||||
| var MetricSuccessChanSize = env.Int("METRIC_SUCCESS_CHAN_SIZE", 1024) | ||||
| var MetricFailChanSize = env.Int("METRIC_FAIL_CHAN_SIZE", 128) | ||||
|  | ||||
| var InitialRootToken = os.Getenv("INITIAL_ROOT_TOKEN") | ||||
|   | ||||
| @@ -69,6 +69,8 @@ const ( | ||||
| 	ChannelTypeMinimax | ||||
| 	ChannelTypeMistral | ||||
| 	ChannelTypeGroq | ||||
| 	ChannelTypeOllama | ||||
| 	ChannelTypeLingYiWanWu | ||||
|  | ||||
| 	ChannelTypeDummy | ||||
| ) | ||||
| @@ -104,6 +106,8 @@ var ChannelBaseURLs = []string{ | ||||
| 	"https://api.minimax.chat",                  // 27 | ||||
| 	"https://api.mistral.ai",                    // 28 | ||||
| 	"https://api.groq.com/openai",               // 29 | ||||
| 	"http://localhost:11434",                    // 30 | ||||
| 	"https://api.lingyiwanwu.com",               // 31 | ||||
| } | ||||
|  | ||||
| const ( | ||||
|   | ||||
							
								
								
									
										6
									
								
								common/conv/any.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								common/conv/any.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | ||||
| package conv | ||||
|  | ||||
| func AsString(v any) string { | ||||
| 	str, _ := v.(string) | ||||
| 	return str | ||||
| } | ||||
| @@ -1,9 +1,12 @@ | ||||
| package common | ||||
|  | ||||
| import "github.com/songquanpeng/one-api/common/helper" | ||||
| import ( | ||||
| 	"github.com/songquanpeng/one-api/common/env" | ||||
| ) | ||||
|  | ||||
| var UsingSQLite = false | ||||
| var UsingPostgreSQL = false | ||||
| var UsingMySQL = false | ||||
|  | ||||
| var SQLitePath = "one-api.db" | ||||
| var SQLiteBusyTimeout = helper.GetOrDefaultEnvInt("SQLITE_BUSY_TIMEOUT", 3000) | ||||
| var SQLiteBusyTimeout = env.Int("SQLITE_BUSY_TIMEOUT", 3000) | ||||
|   | ||||
							
								
								
									
										42
									
								
								common/env/helper.go
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										42
									
								
								common/env/helper.go
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,42 @@ | ||||
| package env | ||||
|  | ||||
| import ( | ||||
| 	"os" | ||||
| 	"strconv" | ||||
| ) | ||||
|  | ||||
| func Bool(env string, defaultValue bool) bool { | ||||
| 	if env == "" || os.Getenv(env) == "" { | ||||
| 		return defaultValue | ||||
| 	} | ||||
| 	return os.Getenv(env) == "true" | ||||
| } | ||||
|  | ||||
| func Int(env string, defaultValue int) int { | ||||
| 	if env == "" || os.Getenv(env) == "" { | ||||
| 		return defaultValue | ||||
| 	} | ||||
| 	num, err := strconv.Atoi(os.Getenv(env)) | ||||
| 	if err != nil { | ||||
| 		return defaultValue | ||||
| 	} | ||||
| 	return num | ||||
| } | ||||
|  | ||||
| func Float64(env string, defaultValue float64) float64 { | ||||
| 	if env == "" || os.Getenv(env) == "" { | ||||
| 		return defaultValue | ||||
| 	} | ||||
| 	num, err := strconv.ParseFloat(os.Getenv(env), 64) | ||||
| 	if err != nil { | ||||
| 		return defaultValue | ||||
| 	} | ||||
| 	return num | ||||
| } | ||||
|  | ||||
| func String(env string, defaultValue string) string { | ||||
| 	if env == "" || os.Getenv(env) == "" { | ||||
| 		return defaultValue | ||||
| 	} | ||||
| 	return os.Getenv(env) | ||||
| } | ||||
| @@ -3,12 +3,10 @@ package helper | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"github.com/google/uuid" | ||||
| 	"github.com/songquanpeng/one-api/common/logger" | ||||
| 	"html/template" | ||||
| 	"log" | ||||
| 	"math/rand" | ||||
| 	"net" | ||||
| 	"os" | ||||
| 	"os/exec" | ||||
| 	"runtime" | ||||
| 	"strconv" | ||||
| @@ -187,6 +185,10 @@ func GetTimeString() string { | ||||
| 	return fmt.Sprintf("%s%d", now.Format("20060102150405"), now.UnixNano()%1e9) | ||||
| } | ||||
|  | ||||
| func GenRequestID() string { | ||||
| 	return GetTimeString() + GetRandomNumberString(8) | ||||
| } | ||||
|  | ||||
| func Max(a int, b int) int { | ||||
| 	if a >= b { | ||||
| 		return a | ||||
| @@ -195,44 +197,6 @@ func Max(a int, b int) int { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func GetOrDefaultEnvBool(env string, defaultValue bool) bool { | ||||
| 	if env == "" || os.Getenv(env) == "" { | ||||
| 		return defaultValue | ||||
| 	} | ||||
| 	return os.Getenv(env) == "true" | ||||
| } | ||||
|  | ||||
| func GetOrDefaultEnvInt(env string, defaultValue int) int { | ||||
| 	if env == "" || os.Getenv(env) == "" { | ||||
| 		return defaultValue | ||||
| 	} | ||||
| 	num, err := strconv.Atoi(os.Getenv(env)) | ||||
| 	if err != nil { | ||||
| 		logger.SysError(fmt.Sprintf("failed to parse %s: %s, using default value: %d", env, err.Error(), defaultValue)) | ||||
| 		return defaultValue | ||||
| 	} | ||||
| 	return num | ||||
| } | ||||
|  | ||||
| func GetOrDefaultEnvFloat64(env string, defaultValue float64) float64 { | ||||
| 	if env == "" || os.Getenv(env) == "" { | ||||
| 		return defaultValue | ||||
| 	} | ||||
| 	num, err := strconv.ParseFloat(os.Getenv(env), 64) | ||||
| 	if err != nil { | ||||
| 		logger.SysError(fmt.Sprintf("failed to parse %s: %s, using default value: %f", env, err.Error(), defaultValue)) | ||||
| 		return defaultValue | ||||
| 	} | ||||
| 	return num | ||||
| } | ||||
|  | ||||
| func GetOrDefaultEnvString(env string, defaultValue string) string { | ||||
| 	if env == "" || os.Getenv(env) == "" { | ||||
| 		return defaultValue | ||||
| 	} | ||||
| 	return os.Getenv(env) | ||||
| } | ||||
|  | ||||
| func AssignOrDefault(value string, defaultValue string) string { | ||||
| 	if len(value) != 0 { | ||||
| 		return value | ||||
|   | ||||
| @@ -4,6 +4,8 @@ import ( | ||||
| 	"context" | ||||
| 	"fmt" | ||||
| 	"github.com/gin-gonic/gin" | ||||
| 	"github.com/songquanpeng/one-api/common/config" | ||||
| 	"github.com/songquanpeng/one-api/common/helper" | ||||
| 	"io" | ||||
| 	"log" | ||||
| 	"os" | ||||
| @@ -19,9 +21,6 @@ const ( | ||||
| 	loggerError = "ERR" | ||||
| ) | ||||
|  | ||||
| const maxLogCount = 1000000 | ||||
|  | ||||
| var logCount int | ||||
| var setupLogLock sync.Mutex | ||||
| var setupLogWorking bool | ||||
|  | ||||
| @@ -57,7 +56,9 @@ func SysError(s string) { | ||||
| } | ||||
|  | ||||
| func Debug(ctx context.Context, msg string) { | ||||
| 	logHelper(ctx, loggerDEBUG, msg) | ||||
| 	if config.DebugEnabled { | ||||
| 		logHelper(ctx, loggerDEBUG, msg) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func Info(ctx context.Context, msg string) { | ||||
| @@ -94,11 +95,12 @@ func logHelper(ctx context.Context, level string, msg string) { | ||||
| 		writer = gin.DefaultWriter | ||||
| 	} | ||||
| 	id := ctx.Value(RequestIdKey) | ||||
| 	if id == nil { | ||||
| 		id = helper.GenRequestID() | ||||
| 	} | ||||
| 	now := time.Now() | ||||
| 	_, _ = fmt.Fprintf(writer, "[%s] %v | %s | %s \n", level, now.Format("2006/01/02 - 15:04:05"), id, msg) | ||||
| 	logCount++ // we don't need accurate count, so no lock here | ||||
| 	if logCount > maxLogCount && !setupLogWorking { | ||||
| 		logCount = 0 | ||||
| 	if !setupLogWorking { | ||||
| 		setupLogWorking = true | ||||
| 		go func() { | ||||
| 			SetupLogger() | ||||
|   | ||||
| @@ -4,7 +4,6 @@ import ( | ||||
| 	"encoding/json" | ||||
| 	"github.com/songquanpeng/one-api/common/logger" | ||||
| 	"strings" | ||||
| 	"time" | ||||
| ) | ||||
|  | ||||
| const ( | ||||
| @@ -31,7 +30,7 @@ var ModelRatio = map[string]float64{ | ||||
| 	"gpt-4-0125-preview":      5,    // $0.01 / 1K tokens | ||||
| 	"gpt-4-turbo-preview":     5,    // $0.01 / 1K tokens | ||||
| 	"gpt-4-vision-preview":    5,    // $0.01 / 1K tokens | ||||
| 	"gpt-3.5-turbo":           0.75, // $0.0015 / 1K tokens | ||||
| 	"gpt-3.5-turbo":           0.25, // $0.0005 / 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 | ||||
| @@ -69,18 +68,25 @@ var ModelRatio = map[string]float64{ | ||||
| 	"claude-instant-1.2":       0.8 / 1000 * USD, | ||||
| 	"claude-2.0":               8.0 / 1000 * USD, | ||||
| 	"claude-2.1":               8.0 / 1000 * USD, | ||||
| 	"claude-3-haiku-20240229":  0.25 / 1000 * USD, | ||||
| 	"claude-3-haiku-20240307":  0.25 / 1000 * USD, | ||||
| 	"claude-3-sonnet-20240229": 3.0 / 1000 * USD, | ||||
| 	"claude-3-opus-20240229":   15.0 / 1000 * USD, | ||||
| 	// https://cloud.baidu.com/doc/WENXINWORKSHOP/s/hlrk4akp7 | ||||
| 	"ERNIE-Bot":         0.8572,     // ¥0.012 / 1k tokens | ||||
| 	"ERNIE-Bot-turbo":   0.5715,     // ¥0.008 / 1k tokens | ||||
| 	"ERNIE-Bot-4":       0.12 * RMB, // ¥0.12 / 1k tokens | ||||
| 	"ERNIE-Bot-8k":      0.024 * RMB, | ||||
| 	"Embedding-V1":      0.1429, // ¥0.002 / 1k tokens | ||||
| 	"PaLM-2":            1, | ||||
| 	"gemini-pro":        1, // $0.00025 / 1k characters -> $0.001 / 1k tokens | ||||
| 	"gemini-pro-vision": 1, // $0.00025 / 1k characters -> $0.001 / 1k tokens | ||||
| 	"ERNIE-Bot":       0.8572,     // ¥0.012 / 1k tokens | ||||
| 	"ERNIE-Bot-turbo": 0.5715,     // ¥0.008 / 1k tokens | ||||
| 	"ERNIE-Bot-4":     0.12 * RMB, // ¥0.12 / 1k tokens | ||||
| 	"ERNIE-Bot-8K":    0.024 * RMB, | ||||
| 	"Embedding-V1":    0.1429, // ¥0.002 / 1k tokens | ||||
| 	"bge-large-zh":    0.002 * RMB, | ||||
| 	"bge-large-en":    0.002 * RMB, | ||||
| 	"bge-large-8k":    0.002 * RMB, | ||||
| 	// https://ai.google.dev/pricing | ||||
| 	"PaLM-2":                    1, | ||||
| 	"gemini-pro":                1, // $0.00025 / 1k characters -> $0.001 / 1k tokens | ||||
| 	"gemini-pro-vision":         1, // $0.00025 / 1k characters -> $0.001 / 1k tokens | ||||
| 	"gemini-1.0-pro-vision-001": 1, | ||||
| 	"gemini-1.0-pro-001":        1, | ||||
| 	"gemini-1.5-pro":            1, | ||||
| 	// https://open.bigmodel.cn/pricing | ||||
| 	"glm-4":                     0.1 * RMB, | ||||
| 	"glm-4v":                    0.1 * RMB, | ||||
| @@ -130,6 +136,10 @@ var ModelRatio = map[string]float64{ | ||||
| 	"llama2-7b-2048":     0.1 / 1000 * USD, | ||||
| 	"mixtral-8x7b-32768": 0.27 / 1000 * USD, | ||||
| 	"gemma-7b-it":        0.1 / 1000 * USD, | ||||
| 	// https://platform.lingyiwanwu.com/docs#-计费单元 | ||||
| 	"yi-34b-chat-0205": 2.5 / 1000 * RMB, | ||||
| 	"yi-34b-chat-200k": 12.0 / 1000 * RMB, | ||||
| 	"yi-vl-plus":       6.0 / 1000 * RMB, | ||||
| } | ||||
|  | ||||
| var CompletionRatio = map[string]float64{} | ||||
| @@ -148,6 +158,26 @@ func init() { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func AddNewMissingRatio(oldRatio string) string { | ||||
| 	newRatio := make(map[string]float64) | ||||
| 	err := json.Unmarshal([]byte(oldRatio), &newRatio) | ||||
| 	if err != nil { | ||||
| 		logger.SysError("error unmarshalling old ratio: " + err.Error()) | ||||
| 		return oldRatio | ||||
| 	} | ||||
| 	for k, v := range DefaultModelRatio { | ||||
| 		if _, ok := newRatio[k]; !ok { | ||||
| 			newRatio[k] = v | ||||
| 		} | ||||
| 	} | ||||
| 	jsonBytes, err := json.Marshal(newRatio) | ||||
| 	if err != nil { | ||||
| 		logger.SysError("error marshalling new ratio: " + err.Error()) | ||||
| 		return oldRatio | ||||
| 	} | ||||
| 	return string(jsonBytes) | ||||
| } | ||||
|  | ||||
| func ModelRatio2JSONString() string { | ||||
| 	jsonBytes, err := json.Marshal(ModelRatio) | ||||
| 	if err != nil { | ||||
| @@ -197,7 +227,7 @@ func GetCompletionRatio(name string) float64 { | ||||
| 		return ratio | ||||
| 	} | ||||
| 	if strings.HasPrefix(name, "gpt-3.5") { | ||||
| 		if strings.HasSuffix(name, "0125") { | ||||
| 		if name == "gpt-3.5-turbo" || strings.HasSuffix(name, "0125") { | ||||
| 			// https://openai.com/blog/new-embedding-models-and-api-updates | ||||
| 			// Updated GPT-3.5 Turbo model and lower pricing | ||||
| 			return 3 | ||||
| @@ -205,15 +235,6 @@ func GetCompletionRatio(name string) float64 { | ||||
| 		if strings.HasSuffix(name, "1106") { | ||||
| 			return 2 | ||||
| 		} | ||||
| 		if name == "gpt-3.5-turbo" || name == "gpt-3.5-turbo-16k" { | ||||
| 			// TODO: clear this after 2023-12-11 | ||||
| 			now := time.Now() | ||||
| 			// https://platform.openai.com/docs/models/continuous-model-upgrades | ||||
| 			// if after 2023-12-11, use 2 | ||||
| 			if now.After(time.Date(2023, 12, 11, 0, 0, 0, 0, time.UTC)) { | ||||
| 				return 2 | ||||
| 			} | ||||
| 		} | ||||
| 		return 4.0 / 3.0 | ||||
| 	} | ||||
| 	if strings.HasPrefix(name, "gpt-4") { | ||||
| @@ -231,6 +252,9 @@ func GetCompletionRatio(name string) float64 { | ||||
| 	if strings.HasPrefix(name, "mistral-") { | ||||
| 		return 3 | ||||
| 	} | ||||
| 	if strings.HasPrefix(name, "gemini-") { | ||||
| 		return 3 | ||||
| 	} | ||||
| 	switch name { | ||||
| 	case "llama2-70b-4096": | ||||
| 		return 0.8 / 0.7 | ||||
|   | ||||
| @@ -5,7 +5,7 @@ import ( | ||||
| 	"github.com/songquanpeng/one-api/common/config" | ||||
| ) | ||||
|  | ||||
| func LogQuota(quota int) string { | ||||
| func LogQuota(quota int64) string { | ||||
| 	if config.DisplayInCurrencyEnabled { | ||||
| 		return fmt.Sprintf("$%.6f 额度", float64(quota)/config.QuotaPerUnit) | ||||
| 	} else { | ||||
|   | ||||
| @@ -8,8 +8,8 @@ import ( | ||||
| ) | ||||
|  | ||||
| func GetSubscription(c *gin.Context) { | ||||
| 	var remainQuota int | ||||
| 	var usedQuota int | ||||
| 	var remainQuota int64 | ||||
| 	var usedQuota int64 | ||||
| 	var err error | ||||
| 	var token *model.Token | ||||
| 	var expiredTime int64 | ||||
| @@ -60,7 +60,7 @@ func GetSubscription(c *gin.Context) { | ||||
| } | ||||
|  | ||||
| func GetUsage(c *gin.Context) { | ||||
| 	var quota int | ||||
| 	var quota int64 | ||||
| 	var err error | ||||
| 	var token *model.Token | ||||
| 	if config.DisplayTokenStatEnabled { | ||||
|   | ||||
| @@ -323,15 +323,14 @@ func updateAllChannelsBalance() error { | ||||
| } | ||||
|  | ||||
| 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 | ||||
| 	} | ||||
| 	//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": "", | ||||
|   | ||||
| @@ -30,7 +30,7 @@ import ( | ||||
|  | ||||
| func buildTestRequest() *relaymodel.GeneralOpenAIRequest { | ||||
| 	testRequest := &relaymodel.GeneralOpenAIRequest{ | ||||
| 		MaxTokens: 1, | ||||
| 		MaxTokens: 2, | ||||
| 		Stream:    false, | ||||
| 		Model:     "gpt-3.5-turbo", | ||||
| 	} | ||||
| @@ -178,7 +178,11 @@ func testChannels(notify bool, scope string) error { | ||||
| 			milliseconds := tok.Sub(tik).Milliseconds() | ||||
| 			if isChannelEnabled && milliseconds > disableThreshold { | ||||
| 				err = errors.New(fmt.Sprintf("响应时间 %.2fs 超过阈值 %.2fs", float64(milliseconds)/1000.0, float64(disableThreshold)/1000.0)) | ||||
| 				monitor.DisableChannel(channel.Id, channel.Name, err.Error()) | ||||
| 				if config.AutomaticDisableChannelEnabled { | ||||
| 					monitor.DisableChannel(channel.Id, channel.Name, err.Error()) | ||||
| 				} else { | ||||
| 					_ = message.Notify(message.ByAll, fmt.Sprintf("渠道 %s (%d)测试超时", channel.Name, channel.Id), "", err.Error()) | ||||
| 				} | ||||
| 			} | ||||
| 			if isChannelEnabled && util.ShouldDisableChannel(openaiErr, -1) { | ||||
| 				monitor.DisableChannel(channel.Id, channel.Name, err.Error()) | ||||
| @@ -193,7 +197,7 @@ func testChannels(notify bool, scope string) error { | ||||
| 		testAllChannelsRunning = false | ||||
| 		testAllChannelsLock.Unlock() | ||||
| 		if notify { | ||||
| 			err := message.Notify(message.ByAll, "通道测试完成", "", "通道测试完成,如果没有收到禁用通知,说明所有通道都正常") | ||||
| 			err := message.Notify(message.ByAll, "渠道测试完成", "", "渠道测试完成,如果没有收到禁用通知,说明所有渠道都正常") | ||||
| 			if err != nil { | ||||
| 				logger.SysError(fmt.Sprintf("failed to send email: %s", err.Error())) | ||||
| 			} | ||||
|   | ||||
| @@ -4,12 +4,14 @@ import ( | ||||
| 	"fmt" | ||||
| 	"github.com/gin-gonic/gin" | ||||
| 	"github.com/songquanpeng/one-api/common" | ||||
| 	"github.com/songquanpeng/one-api/model" | ||||
| 	"github.com/songquanpeng/one-api/relay/channel/openai" | ||||
| 	"github.com/songquanpeng/one-api/relay/constant" | ||||
| 	"github.com/songquanpeng/one-api/relay/helper" | ||||
| 	relaymodel "github.com/songquanpeng/one-api/relay/model" | ||||
| 	"github.com/songquanpeng/one-api/relay/util" | ||||
| 	"net/http" | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| // https://platform.openai.com/docs/api-reference/models/list | ||||
| @@ -120,9 +122,41 @@ func DashboardListModels(c *gin.Context) { | ||||
| } | ||||
|  | ||||
| func ListModels(c *gin.Context) { | ||||
| 	ctx := c.Request.Context() | ||||
| 	var availableModels []string | ||||
| 	if c.GetString("available_models") != "" { | ||||
| 		availableModels = strings.Split(c.GetString("available_models"), ",") | ||||
| 	} else { | ||||
| 		userId := c.GetInt("id") | ||||
| 		userGroup, _ := model.CacheGetUserGroup(userId) | ||||
| 		availableModels, _ = model.CacheGetGroupModels(ctx, userGroup) | ||||
| 	} | ||||
| 	modelSet := make(map[string]bool) | ||||
| 	for _, availableModel := range availableModels { | ||||
| 		modelSet[availableModel] = true | ||||
| 	} | ||||
| 	var availableOpenAIModels []OpenAIModels | ||||
| 	for _, model := range openAIModels { | ||||
| 		if _, ok := modelSet[model.Id]; ok { | ||||
| 			modelSet[model.Id] = false | ||||
| 			availableOpenAIModels = append(availableOpenAIModels, model) | ||||
| 		} | ||||
| 	} | ||||
| 	for modelName, ok := range modelSet { | ||||
| 		if ok { | ||||
| 			availableOpenAIModels = append(availableOpenAIModels, OpenAIModels{ | ||||
| 				Id:      modelName, | ||||
| 				Object:  "model", | ||||
| 				Created: 1626777600, | ||||
| 				OwnedBy: "custom", | ||||
| 				Root:    modelName, | ||||
| 				Parent:  nil, | ||||
| 			}) | ||||
| 		} | ||||
| 	} | ||||
| 	c.JSON(200, gin.H{ | ||||
| 		"object": "list", | ||||
| 		"data":   openAIModels, | ||||
| 		"data":   availableOpenAIModels, | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| @@ -142,3 +176,30 @@ func RetrieveModel(c *gin.Context) { | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func GetUserAvailableModels(c *gin.Context) { | ||||
| 	ctx := c.Request.Context() | ||||
| 	id := c.GetInt("id") | ||||
| 	userGroup, err := model.CacheGetUserGroup(id) | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	models, err := model.CacheGetGroupModels(ctx, userGroup) | ||||
| 	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":    models, | ||||
| 	}) | ||||
| 	return | ||||
| } | ||||
|   | ||||
| @@ -16,7 +16,10 @@ func GetAllTokens(c *gin.Context) { | ||||
| 	if p < 0 { | ||||
| 		p = 0 | ||||
| 	} | ||||
| 	tokens, err := model.GetAllUserTokens(userId, p*config.ItemsPerPage, config.ItemsPerPage) | ||||
|  | ||||
| 	order := c.Query("order") | ||||
| 	tokens, err := model.GetAllUserTokens(userId, p*config.ItemsPerPage, config.ItemsPerPage, order) | ||||
|  | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| @@ -127,6 +130,7 @@ func AddToken(c *gin.Context) { | ||||
| 		ExpiredTime:    token.ExpiredTime, | ||||
| 		RemainQuota:    token.RemainQuota, | ||||
| 		UnlimitedQuota: token.UnlimitedQuota, | ||||
| 		Models:         token.Models, | ||||
| 	} | ||||
| 	err = cleanToken.Insert() | ||||
| 	if err != nil { | ||||
| @@ -139,6 +143,7 @@ func AddToken(c *gin.Context) { | ||||
| 	c.JSON(http.StatusOK, gin.H{ | ||||
| 		"success": true, | ||||
| 		"message": "", | ||||
| 		"data":    cleanToken, | ||||
| 	}) | ||||
| 	return | ||||
| } | ||||
| @@ -212,6 +217,7 @@ func UpdateToken(c *gin.Context) { | ||||
| 		cleanToken.ExpiredTime = token.ExpiredTime | ||||
| 		cleanToken.RemainQuota = token.RemainQuota | ||||
| 		cleanToken.UnlimitedQuota = token.UnlimitedQuota | ||||
| 		cleanToken.Models = token.Models | ||||
| 	} | ||||
| 	err = cleanToken.Update() | ||||
| 	if err != nil { | ||||
|   | ||||
| @@ -180,24 +180,27 @@ func Register(c *gin.Context) { | ||||
| } | ||||
|  | ||||
| func GetAllUsers(c *gin.Context) { | ||||
| 	p, _ := strconv.Atoi(c.Query("p")) | ||||
| 	if p < 0 { | ||||
| 		p = 0 | ||||
| 	} | ||||
| 	users, err := model.GetAllUsers(p*config.ItemsPerPage, config.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":    users, | ||||
| 	}) | ||||
| 	return | ||||
|     p, _ := strconv.Atoi(c.Query("p")) | ||||
|     if p < 0 { | ||||
|         p = 0 | ||||
|     } | ||||
|      | ||||
|     order := c.DefaultQuery("order", "") | ||||
|     users, err := model.GetAllUsers(p*config.ItemsPerPage, config.ItemsPerPage, order) | ||||
| 	 | ||||
|     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":    users, | ||||
|     }) | ||||
| } | ||||
|  | ||||
| func SearchUsers(c *gin.Context) { | ||||
|   | ||||
| @@ -2,7 +2,7 @@ version: '3.4' | ||||
|  | ||||
| services: | ||||
|   one-api: | ||||
|     image: justsong/one-api:latest | ||||
|     image: "${REGISTRY:-docker.io}/justsong/one-api:latest" | ||||
|     container_name: one-api | ||||
|     restart: always | ||||
|     command: --log-dir /app/logs | ||||
| @@ -29,12 +29,12 @@ services: | ||||
|       retries: 3 | ||||
|  | ||||
|   redis: | ||||
|     image: redis:latest | ||||
|     image: "${REGISTRY:-docker.io}/redis:latest" | ||||
|     container_name: redis | ||||
|     restart: always | ||||
|  | ||||
|   db: | ||||
|     image: mysql:8.2.0 | ||||
|     image: "${REGISTRY:-docker.io}/mysql:8.2.0" | ||||
|     restart: always | ||||
|     container_name: mysql | ||||
|     volumes: | ||||
|   | ||||
							
								
								
									
										6
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										6
									
								
								go.mod
									
									
									
									
									
								
							| @@ -42,7 +42,8 @@ require ( | ||||
| 	github.com/gorilla/sessions v1.2.1 // indirect | ||||
| 	github.com/jackc/pgpassfile v1.0.0 // indirect | ||||
| 	github.com/jackc/pgservicefile v0.0.0-20221227161230-091c0ba34f0a // indirect | ||||
| 	github.com/jackc/pgx/v5 v5.3.1 // indirect | ||||
| 	github.com/jackc/pgx/v5 v5.5.4 // indirect | ||||
| 	github.com/jackc/puddle/v2 v2.2.1 // indirect | ||||
| 	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 | ||||
| @@ -58,8 +59,9 @@ require ( | ||||
| 	github.com/ugorji/go/codec v1.2.11 // indirect | ||||
| 	golang.org/x/arch v0.3.0 // indirect | ||||
| 	golang.org/x/net v0.17.0 // indirect | ||||
| 	golang.org/x/sync v0.1.0 // indirect | ||||
| 	golang.org/x/sys v0.15.0 // indirect | ||||
| 	golang.org/x/text v0.14.0 // indirect | ||||
| 	google.golang.org/protobuf v1.30.0 // indirect | ||||
| 	google.golang.org/protobuf v1.33.0 // indirect | ||||
| 	gopkg.in/yaml.v3 v3.0.1 // indirect | ||||
| ) | ||||
|   | ||||
							
								
								
									
										12
									
								
								go.sum
									
									
									
									
									
								
							
							
						
						
									
										12
									
								
								go.sum
									
									
									
									
									
								
							| @@ -73,8 +73,10 @@ github.com/jackc/pgpassfile v1.0.0 h1:/6Hmqy13Ss2zCq62VdNG8tM1wchn8zjSGOBJ6icpsI | ||||
| github.com/jackc/pgpassfile v1.0.0/go.mod h1:CEx0iS5ambNFdcRtxPj5JhEz+xB6uRky5eyVu/W2HEg= | ||||
| github.com/jackc/pgservicefile v0.0.0-20221227161230-091c0ba34f0a h1:bbPeKD0xmW/Y25WS6cokEszi5g+S0QxI/d45PkRi7Nk= | ||||
| github.com/jackc/pgservicefile v0.0.0-20221227161230-091c0ba34f0a/go.mod h1:5TJZWKEWniPve33vlWYSoGYefn3gLQRzjfDlhSJ9ZKM= | ||||
| github.com/jackc/pgx/v5 v5.3.1 h1:Fcr8QJ1ZeLi5zsPZqQeUZhNhxfkkKBOgJuYkJHoBOtU= | ||||
| github.com/jackc/pgx/v5 v5.3.1/go.mod h1:t3JDKnCBlYIc0ewLF0Q7B8MXmoIaBOZj/ic7iHozM/8= | ||||
| github.com/jackc/pgx/v5 v5.5.4 h1:Xp2aQS8uXButQdnCMWNmvx6UysWQQC+u1EoizjguY+8= | ||||
| github.com/jackc/pgx/v5 v5.5.4/go.mod h1:ez9gk+OAat140fv9ErkZDYFWmXLfV+++K0uAOiwgm1A= | ||||
| github.com/jackc/puddle/v2 v2.2.1 h1:RhxXJtFG022u4ibrCSMSiu5aOq1i77R3OHKNJj77OAk= | ||||
| github.com/jackc/puddle/v2 v2.2.1/go.mod h1:vriiEXHvEE654aYKXXjOvZM39qJ0q+azkZFrfEOc3H4= | ||||
| github.com/jinzhu/inflection v1.0.0 h1:K317FqzuhWc8YvSVlFMCCUb36O/S9MCKRDI7QkRKD/E= | ||||
| github.com/jinzhu/inflection v1.0.0/go.mod h1:h+uFLlag+Qp1Va5pdKtLDYj+kHp5pxUVkryuEj+Srlc= | ||||
| github.com/jinzhu/now v1.1.4/go.mod h1:d3SSVoowX0Lcu0IBviAWJpolVfI5UJVZZ7cO71lE/z8= | ||||
| @@ -157,6 +159,8 @@ golang.org/x/image v0.14.0/go.mod h1:HUYqC05R2ZcZ3ejNQsIHQDQiwWM4JBqmm6MKANTp4LE | ||||
| golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= | ||||
| golang.org/x/net v0.17.0 h1:pVaXccu2ozPjCXewfr1S7xza/zcXTity9cCdXQYSjIM= | ||||
| golang.org/x/net v0.17.0/go.mod h1:NxSsAGuq816PNPmqtQdLE42eU2Fs7NoRIZrHJAlaCOE= | ||||
| golang.org/x/sync v0.1.0 h1:wsuoTGHzEhffawBOhz5CYhcrV4IdKZbEyZjBMuTp12o= | ||||
| golang.org/x/sync v0.1.0/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| 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-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| @@ -177,8 +181,8 @@ golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543 h1:E7g+9GITq07hpfrRu66IV | ||||
| 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/go.mod h1:HV8QOd/L58Z+nl8r43ehVNZIU/HEI6OcFqwMG9pJV4I= | ||||
| 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= | ||||
| google.golang.org/protobuf v1.33.0 h1:uNO2rsAINq/JlFpSdYEKIZ0uKD/R9cpdv0T+yoGwGmI= | ||||
| google.golang.org/protobuf v1.33.0/go.mod h1:c6P6GXX6sHbq/GpV6MGZEdwhWPcYBgnhAHhKbcUYpos= | ||||
| 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= | ||||
|   | ||||
							
								
								
									
										34
									
								
								i18n/en.json
									
									
									
									
									
								
							
							
						
						
									
										34
									
								
								i18n/en.json
									
									
									
									
									
								
							| @@ -8,12 +8,12 @@ | ||||
|   "确认删除": "Confirm Delete", | ||||
|   "确认绑定": "Confirm Binding", | ||||
|   "您正在删除自己的帐户,将清空所有数据且不可恢复": "You are deleting your account, all data will be cleared and unrecoverable.", | ||||
|   "\"通道「%s」(#%d)已被禁用\"": "\"Channel %s (#%d) has been disabled\"", | ||||
|   "通道「%s」(#%d)已被禁用,原因:%s": "Channel %s (#%d) has been disabled, reason: %s", | ||||
|   "\"渠道「%s」(#%d)已被禁用\"": "\"Channel %s (#%d) has been disabled\"", | ||||
|   "渠道「%s」(#%d)已被禁用,原因:%s": "Channel %s (#%d) has been disabled, reason: %s", | ||||
|   "测试已在运行中": "Test is already running", | ||||
|   "响应时间 %.2fs 超过阈值 %.2fs": "Response time %.2fs exceeds threshold %.2fs", | ||||
|   "通道测试完成": "Channel test completed", | ||||
|   "通道测试完成,如果没有收到禁用通知,说明所有通道都正常": "Channel test completed, if you have not received the disable notification, it means that all channels are normal", | ||||
|   "渠道测试完成": "Channel test completed", | ||||
|   "渠道测试完成,如果没有收到禁用通知,说明所有渠道都正常": "Channel test completed, if you have not received the disable notification, it means that all channels are normal", | ||||
|   "无法连接至 GitHub 服务器,请稍后重试!": "Unable to connect to GitHub server, please try again later!", | ||||
|   "返回值非法,用户字段为空,请稍后重试!": "The return value is illegal, the user field is empty, please try again later!", | ||||
|   "管理员未开启通过 GitHub 登录以及注册": "The administrator did not turn on login and registration via GitHub", | ||||
| @@ -119,11 +119,11 @@ | ||||
|   " 个月 ": " M ", | ||||
|   " 年 ": " y ", | ||||
|   "未测试": "Not tested", | ||||
|   "通道 ${name} 测试成功,耗时 ${time.toFixed(2)} 秒。": "Channel ${name} test succeeded, time consumed ${time.toFixed(2)} s.", | ||||
|   "已成功开始测试所有通道,请刷新页面查看结果。": "All channels have been successfully tested, please refresh the page to view the results.", | ||||
|   "已成功开始测试所有已启用通道,请刷新页面查看结果。": "All enabled channels have been successfully tested, please refresh the page to view the results.", | ||||
|   "通道 ${name} 余额更新成功!": "Channel ${name} balance updated successfully!", | ||||
|   "已更新完毕所有已启用通道余额!": "The balance of all enabled channels has been updated!", | ||||
|   "渠道 ${name} 测试成功,耗时 ${time.toFixed(2)} 秒。": "Channel ${name} test succeeded, time consumed ${time.toFixed(2)} s.", | ||||
|   "已成功开始测试所有渠道,请刷新页面查看结果。": "All channels have been successfully tested, please refresh the page to view the results.", | ||||
|   "已成功开始测试所有已启用渠道,请刷新页面查看结果。": "All enabled channels have been successfully tested, please refresh the page to view the results.", | ||||
|   "渠道 ${name} 余额更新成功!": "Channel ${name} balance updated successfully!", | ||||
|   "已更新完毕所有已启用渠道余额!": "The balance of all enabled channels has been updated!", | ||||
|   "搜索渠道的 ID,名称和密钥 ...": "Search for channel ID, name and key ...", | ||||
|   "名称": "Name", | ||||
|   "分组": "Group", | ||||
| @@ -141,9 +141,9 @@ | ||||
|   "启用": "Enable", | ||||
|   "编辑": "Edit", | ||||
|   "添加新的渠道": "Add a new channel", | ||||
|   "测试所有通道": "Test all channels", | ||||
|   "测试所有已启用通道": "Test all enabled channels", | ||||
|   "更新所有已启用通道余额": "Update the balance of all enabled channels", | ||||
|   "测试所有渠道": "Test all channels", | ||||
|   "测试所有已启用渠道": "Test all enabled channels", | ||||
|   "更新所有已启用渠道余额": "Update the balance of all enabled channels", | ||||
|   "刷新": "Refresh", | ||||
|   "处理中...": "Processing...", | ||||
|   "绑定成功!": "Binding succeeded!", | ||||
| @@ -207,11 +207,11 @@ | ||||
|   "监控设置": "Monitoring Settings", | ||||
|   "最长响应时间": "Longest Response Time", | ||||
|   "单位秒": "Unit in seconds", | ||||
|   "当运行通道全部测试时": "When all operating channels are tested", | ||||
|   "超过此时间将自动禁用通道": "Channels will be automatically disabled if this time is exceeded", | ||||
|   "当运行渠道全部测试时": "When all operating channels are tested", | ||||
|   "超过此时间将自动禁用渠道": "Channels will be automatically disabled if this time is exceeded", | ||||
|   "额度提醒阈值": "Quota reminder threshold", | ||||
|   "低于此额度时将发送邮件提醒用户": "Email will be sent to remind users when the quota is below this", | ||||
|   "失败时自动禁用通道": "Automatically disable the channel when it fails", | ||||
|   "失败时自动禁用渠道": "Automatically disable the channel when it fails", | ||||
|   "保存监控设置": "Save Monitoring Settings", | ||||
|   "额度设置": "Quota Settings", | ||||
|   "新用户初始额度": "Initial quota for new users", | ||||
| @@ -405,7 +405,7 @@ | ||||
|   "镜像": "Mirror", | ||||
|   "请输入镜像站地址,格式为:https://domain.com,可不填,不填则使用渠道默认值": "Please enter the mirror site address, the format is: https://domain.com, it can be left blank, if left blank, the default value of the channel will be used", | ||||
|   "模型": "Model", | ||||
|   "请选择该通道所支持的模型": "Please select the model supported by the channel", | ||||
|   "请选择该渠道所支持的模型": "Please select the model supported by the channel", | ||||
|   "填入基础模型": "Fill in the basic model", | ||||
|   "填入所有模型": "Fill in all models", | ||||
|   "清除所有模型": "Clear all models", | ||||
| @@ -515,7 +515,7 @@ | ||||
|   "请输入自定义渠道的 Base URL": "Please enter the Base URL of the custom channel", | ||||
|   "Homepage URL 填": "Fill in the Homepage URL", | ||||
|   "Authorization callback URL 填": "Fill in the Authorization callback URL", | ||||
|   "请为通道命名": "Please name the channel", | ||||
|   "请为渠道命名": "Please name the channel", | ||||
|   "此项可选,用于修改请求体中的模型名称,为一个 JSON 字符串,键为请求中模型名称,值为要替换的模型名称,例如:": "This is optional, used to modify the model name in the request body, it's a JSON string, the key is the model name in the request, and the value is the model name to be replaced, for example:", | ||||
|   "模型重定向": "Model redirection", | ||||
|   "请输入渠道对应的鉴权密钥": "Please enter the authentication key corresponding to the channel", | ||||
|   | ||||
							
								
								
									
										25
									
								
								main.go
									
									
									
									
									
								
							
							
						
						
									
										25
									
								
								main.go
									
									
									
									
									
								
							| @@ -9,7 +9,6 @@ import ( | ||||
| 	"github.com/songquanpeng/one-api/common" | ||||
| 	"github.com/songquanpeng/one-api/common/config" | ||||
| 	"github.com/songquanpeng/one-api/common/logger" | ||||
| 	"github.com/songquanpeng/one-api/common/message" | ||||
| 	"github.com/songquanpeng/one-api/controller" | ||||
| 	"github.com/songquanpeng/one-api/middleware" | ||||
| 	"github.com/songquanpeng/one-api/model" | ||||
| @@ -31,11 +30,25 @@ func main() { | ||||
| 	if config.DebugEnabled { | ||||
| 		logger.SysLog("running in debug mode") | ||||
| 	} | ||||
| 	var err error | ||||
| 	// Initialize SQL Database | ||||
| 	err := model.InitDB() | ||||
| 	model.DB, err = model.InitDB("SQL_DSN") | ||||
| 	if err != nil { | ||||
| 		logger.FatalLog("failed to initialize database: " + err.Error()) | ||||
| 	} | ||||
| 	if os.Getenv("LOG_SQL_DSN") != "" { | ||||
| 		logger.SysLog("using secondary database for table logs") | ||||
| 		model.LOG_DB, err = model.InitDB("LOG_SQL_DSN") | ||||
| 		if err != nil { | ||||
| 			logger.FatalLog("failed to initialize secondary database: " + err.Error()) | ||||
| 		} | ||||
| 	} else { | ||||
| 		model.LOG_DB = model.DB | ||||
| 	} | ||||
| 	err = model.CreateRootAccountIfNeed() | ||||
| 	if err != nil { | ||||
| 		logger.FatalLog("database init error: " + err.Error()) | ||||
| 	} | ||||
| 	defer func() { | ||||
| 		err := model.CloseDB() | ||||
| 		if err != nil { | ||||
| @@ -65,13 +78,6 @@ func main() { | ||||
| 		go model.SyncOptions(config.SyncFrequency) | ||||
| 		go model.SyncChannelCache(config.SyncFrequency) | ||||
| 	} | ||||
| 	if os.Getenv("CHANNEL_UPDATE_FREQUENCY") != "" { | ||||
| 		frequency, err := strconv.Atoi(os.Getenv("CHANNEL_UPDATE_FREQUENCY")) | ||||
| 		if err != nil { | ||||
| 			logger.FatalLog("failed to parse CHANNEL_UPDATE_FREQUENCY: " + err.Error()) | ||||
| 		} | ||||
| 		go controller.AutomaticallyUpdateChannels(frequency) | ||||
| 	} | ||||
| 	if os.Getenv("CHANNEL_TEST_FREQUENCY") != "" { | ||||
| 		frequency, err := strconv.Atoi(os.Getenv("CHANNEL_TEST_FREQUENCY")) | ||||
| 		if err != nil { | ||||
| @@ -88,7 +94,6 @@ func main() { | ||||
| 		logger.SysLog("metric enabled, will disable channel if too much request failed") | ||||
| 	} | ||||
| 	openai.InitTokenEncoders() | ||||
| 	_ = message.SendMessage("One API", "", fmt.Sprintf("One API %s started", common.Version)) | ||||
|  | ||||
| 	// Initialize HTTP server | ||||
| 	server := gin.New() | ||||
|   | ||||
| @@ -1,6 +1,7 @@ | ||||
| package middleware | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"github.com/gin-contrib/sessions" | ||||
| 	"github.com/gin-gonic/gin" | ||||
| 	"github.com/songquanpeng/one-api/common" | ||||
| @@ -107,6 +108,19 @@ func TokenAuth() func(c *gin.Context) { | ||||
| 			abortWithMessage(c, http.StatusForbidden, "用户已被封禁") | ||||
| 			return | ||||
| 		} | ||||
| 		requestModel, err := getRequestModel(c) | ||||
| 		if err != nil { | ||||
| 			abortWithMessage(c, http.StatusBadRequest, err.Error()) | ||||
| 			return | ||||
| 		} | ||||
| 		c.Set("request_model", requestModel) | ||||
| 		if token.Models != nil && *token.Models != "" { | ||||
| 			c.Set("available_models", *token.Models) | ||||
| 			if requestModel != "" && !isModelInList(requestModel, *token.Models) { | ||||
| 				abortWithMessage(c, http.StatusForbidden, fmt.Sprintf("该令牌无权使用模型:%s", requestModel)) | ||||
| 				return | ||||
| 			} | ||||
| 		} | ||||
| 		c.Set("id", token.UserId) | ||||
| 		c.Set("token_id", token.Id) | ||||
| 		c.Set("token_name", token.Name) | ||||
|   | ||||
| @@ -2,14 +2,12 @@ package middleware | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"github.com/gin-gonic/gin" | ||||
| 	"github.com/songquanpeng/one-api/common" | ||||
| 	"github.com/songquanpeng/one-api/common/logger" | ||||
| 	"github.com/songquanpeng/one-api/model" | ||||
| 	"net/http" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
|  | ||||
| 	"github.com/gin-gonic/gin" | ||||
| ) | ||||
|  | ||||
| type ModelRequest struct { | ||||
| @@ -40,37 +38,11 @@ func Distribute() func(c *gin.Context) { | ||||
| 				return | ||||
| 			} | ||||
| 		} else { | ||||
| 			// Select a channel for the user | ||||
| 			var modelRequest ModelRequest | ||||
| 			err := common.UnmarshalBodyReusable(c, &modelRequest) | ||||
| 			requestModel := c.GetString("request_model") | ||||
| 			var err error | ||||
| 			channel, err = model.CacheGetRandomSatisfiedChannel(userGroup, requestModel, false) | ||||
| 			if err != nil { | ||||
| 				abortWithMessage(c, http.StatusBadRequest, "无效的请求") | ||||
| 				return | ||||
| 			} | ||||
| 			if strings.HasPrefix(c.Request.URL.Path, "/v1/moderations") { | ||||
| 				if modelRequest.Model == "" { | ||||
| 					modelRequest.Model = "text-moderation-stable" | ||||
| 				} | ||||
| 			} | ||||
| 			if strings.HasSuffix(c.Request.URL.Path, "embeddings") { | ||||
| 				if modelRequest.Model == "" { | ||||
| 					modelRequest.Model = c.Param("model") | ||||
| 				} | ||||
| 			} | ||||
| 			if strings.HasPrefix(c.Request.URL.Path, "/v1/images/generations") { | ||||
| 				if modelRequest.Model == "" { | ||||
| 					modelRequest.Model = "dall-e-2" | ||||
| 				} | ||||
| 			} | ||||
| 			if strings.HasPrefix(c.Request.URL.Path, "/v1/audio/transcriptions") || strings.HasPrefix(c.Request.URL.Path, "/v1/audio/translations") { | ||||
| 				if modelRequest.Model == "" { | ||||
| 					modelRequest.Model = "whisper-1" | ||||
| 				} | ||||
| 			} | ||||
| 			requestModel = modelRequest.Model | ||||
| 			channel, err = model.CacheGetRandomSatisfiedChannel(userGroup, modelRequest.Model, false) | ||||
| 			if err != nil { | ||||
| 				message := fmt.Sprintf("当前分组 %s 下对于模型 %s 无可用渠道", userGroup, modelRequest.Model) | ||||
| 				message := fmt.Sprintf("当前分组 %s 下对于模型 %s 无可用渠道", userGroup, requestModel) | ||||
| 				if channel != nil { | ||||
| 					logger.SysError(fmt.Sprintf("渠道不存在:%d", channel.Id)) | ||||
| 					message = "数据库一致性已被破坏,请联系管理员" | ||||
|   | ||||
| @@ -3,6 +3,7 @@ package middleware | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"github.com/gin-gonic/gin" | ||||
| 	"github.com/songquanpeng/one-api/common" | ||||
| 	"github.com/songquanpeng/one-api/common/logger" | ||||
| 	"net/http" | ||||
| 	"runtime/debug" | ||||
| @@ -12,11 +13,15 @@ func RelayPanicRecover() gin.HandlerFunc { | ||||
| 	return func(c *gin.Context) { | ||||
| 		defer func() { | ||||
| 			if err := recover(); err != nil { | ||||
| 				logger.SysError(fmt.Sprintf("panic detected: %v", err)) | ||||
| 				logger.SysError(fmt.Sprintf("stacktrace from panic: %s", string(debug.Stack()))) | ||||
| 				ctx := c.Request.Context() | ||||
| 				logger.Errorf(ctx, fmt.Sprintf("panic detected: %v", err)) | ||||
| 				logger.Errorf(ctx, fmt.Sprintf("stacktrace from panic: %s", string(debug.Stack()))) | ||||
| 				logger.Errorf(ctx, fmt.Sprintf("request: %s %s", c.Request.Method, c.Request.URL.Path)) | ||||
| 				body, _ := common.GetRequestBody(c) | ||||
| 				logger.Errorf(ctx, fmt.Sprintf("request body: %s", string(body))) | ||||
| 				c.JSON(http.StatusInternalServerError, gin.H{ | ||||
| 					"error": gin.H{ | ||||
| 						"message": fmt.Sprintf("Panic detected, error: %v. Please submit a issue here: https://github.com/songquanpeng/one-api", err), | ||||
| 						"message": fmt.Sprintf("Panic detected, error: %v. Please submit an issue with the related log here: https://github.com/songquanpeng/one-api", err), | ||||
| 						"type":    "one_api_panic", | ||||
| 					}, | ||||
| 				}) | ||||
|   | ||||
| @@ -9,7 +9,7 @@ import ( | ||||
|  | ||||
| func RequestId() func(c *gin.Context) { | ||||
| 	return func(c *gin.Context) { | ||||
| 		id := helper.GetTimeString() + helper.GetRandomNumberString(8) | ||||
| 		id := helper.GenRequestID() | ||||
| 		c.Set(logger.RequestIdKey, id) | ||||
| 		ctx := context.WithValue(c.Request.Context(), logger.RequestIdKey, id) | ||||
| 		c.Request = c.Request.WithContext(ctx) | ||||
|   | ||||
| @@ -1,9 +1,12 @@ | ||||
| package middleware | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"github.com/gin-gonic/gin" | ||||
| 	"github.com/songquanpeng/one-api/common" | ||||
| 	"github.com/songquanpeng/one-api/common/helper" | ||||
| 	"github.com/songquanpeng/one-api/common/logger" | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| func abortWithMessage(c *gin.Context, statusCode int, message string) { | ||||
| @@ -16,3 +19,42 @@ func abortWithMessage(c *gin.Context, statusCode int, message string) { | ||||
| 	c.Abort() | ||||
| 	logger.Error(c.Request.Context(), message) | ||||
| } | ||||
|  | ||||
| func getRequestModel(c *gin.Context) (string, error) { | ||||
| 	var modelRequest ModelRequest | ||||
| 	err := common.UnmarshalBodyReusable(c, &modelRequest) | ||||
| 	if err != nil { | ||||
| 		return "", fmt.Errorf("common.UnmarshalBodyReusable failed: %w", err) | ||||
| 	} | ||||
| 	if strings.HasPrefix(c.Request.URL.Path, "/v1/moderations") { | ||||
| 		if modelRequest.Model == "" { | ||||
| 			modelRequest.Model = "text-moderation-stable" | ||||
| 		} | ||||
| 	} | ||||
| 	if strings.HasSuffix(c.Request.URL.Path, "embeddings") { | ||||
| 		if modelRequest.Model == "" { | ||||
| 			modelRequest.Model = c.Param("model") | ||||
| 		} | ||||
| 	} | ||||
| 	if strings.HasPrefix(c.Request.URL.Path, "/v1/images/generations") { | ||||
| 		if modelRequest.Model == "" { | ||||
| 			modelRequest.Model = "dall-e-2" | ||||
| 		} | ||||
| 	} | ||||
| 	if strings.HasPrefix(c.Request.URL.Path, "/v1/audio/transcriptions") || strings.HasPrefix(c.Request.URL.Path, "/v1/audio/translations") { | ||||
| 		if modelRequest.Model == "" { | ||||
| 			modelRequest.Model = "whisper-1" | ||||
| 		} | ||||
| 	} | ||||
| 	return modelRequest.Model, nil | ||||
| } | ||||
|  | ||||
| func isModelInList(modelName string, models string) bool { | ||||
| 	modelList := strings.Split(models, ",") | ||||
| 	for _, model := range modelList { | ||||
| 		if modelName == model { | ||||
| 			return true | ||||
| 		} | ||||
| 	} | ||||
| 	return false | ||||
| } | ||||
|   | ||||
| @@ -1,7 +1,10 @@ | ||||
| package model | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
| 	"github.com/songquanpeng/one-api/common" | ||||
| 	"gorm.io/gorm" | ||||
| 	"sort" | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| @@ -13,7 +16,7 @@ type Ability struct { | ||||
| 	Priority  *int64 `json:"priority" gorm:"bigint;default:0;index"` | ||||
| } | ||||
|  | ||||
| func GetRandomSatisfiedChannel(group string, model string) (*Channel, error) { | ||||
| func GetRandomSatisfiedChannel(group string, model string, ignoreFirstPriority bool) (*Channel, error) { | ||||
| 	ability := Ability{} | ||||
| 	groupCol := "`group`" | ||||
| 	trueVal := "1" | ||||
| @@ -23,8 +26,13 @@ func GetRandomSatisfiedChannel(group string, model string) (*Channel, error) { | ||||
| 	} | ||||
|  | ||||
| 	var err error = nil | ||||
| 	maxPrioritySubQuery := DB.Model(&Ability{}).Select("MAX(priority)").Where(groupCol+" = ? and model = ? and enabled = "+trueVal, group, model) | ||||
| 	channelQuery := DB.Where(groupCol+" = ? and model = ? and enabled = "+trueVal+" and priority = (?)", group, model, maxPrioritySubQuery) | ||||
| 	var channelQuery *gorm.DB | ||||
| 	if ignoreFirstPriority { | ||||
| 		channelQuery = DB.Where(groupCol+" = ? and model = ? and enabled = "+trueVal, group, model) | ||||
| 	} else { | ||||
| 		maxPrioritySubQuery := DB.Model(&Ability{}).Select("MAX(priority)").Where(groupCol+" = ? and model = ? and enabled = "+trueVal, group, model) | ||||
| 		channelQuery = DB.Where(groupCol+" = ? and model = ? and enabled = "+trueVal+" and priority = (?)", group, model, maxPrioritySubQuery) | ||||
| 	} | ||||
| 	if common.UsingSQLite || common.UsingPostgreSQL { | ||||
| 		err = channelQuery.Order("RANDOM()").First(&ability).Error | ||||
| 	} else { | ||||
| @@ -82,3 +90,19 @@ func (channel *Channel) UpdateAbilities() error { | ||||
| func UpdateAbilityStatus(channelId int, status bool) error { | ||||
| 	return DB.Model(&Ability{}).Where("channel_id = ?", channelId).Select("enabled").Update("enabled", status).Error | ||||
| } | ||||
|  | ||||
| func GetGroupModels(ctx context.Context, group string) ([]string, error) { | ||||
| 	groupCol := "`group`" | ||||
| 	trueVal := "1" | ||||
| 	if common.UsingPostgreSQL { | ||||
| 		groupCol = `"group"` | ||||
| 		trueVal = "true" | ||||
| 	} | ||||
| 	var models []string | ||||
| 	err := DB.Model(&Ability{}).Distinct("model").Where(groupCol+" = ? and enabled = "+trueVal, group).Pluck("model", &models).Error | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	sort.Strings(models) | ||||
| 	return models, err | ||||
| } | ||||
|   | ||||
| @@ -1,6 +1,7 @@ | ||||
| package model | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
| 	"encoding/json" | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| @@ -20,6 +21,7 @@ var ( | ||||
| 	UserId2GroupCacheSeconds  = config.SyncFrequency | ||||
| 	UserId2QuotaCacheSeconds  = config.SyncFrequency | ||||
| 	UserId2StatusCacheSeconds = config.SyncFrequency | ||||
| 	GroupModelsCacheSeconds   = config.SyncFrequency | ||||
| ) | ||||
|  | ||||
| func CacheGetTokenByKey(key string) (*Token, error) { | ||||
| @@ -70,31 +72,42 @@ func CacheGetUserGroup(id int) (group string, err error) { | ||||
| 	return group, err | ||||
| } | ||||
|  | ||||
| func CacheGetUserQuota(id int) (quota int, err error) { | ||||
| func fetchAndUpdateUserQuota(ctx context.Context, id int) (quota int64, err error) { | ||||
| 	quota, err = GetUserQuota(id) | ||||
| 	if err != nil { | ||||
| 		return 0, err | ||||
| 	} | ||||
| 	err = common.RedisSet(fmt.Sprintf("user_quota:%d", id), fmt.Sprintf("%d", quota), time.Duration(UserId2QuotaCacheSeconds)*time.Second) | ||||
| 	if err != nil { | ||||
| 		logger.Error(ctx, "Redis set user quota error: "+err.Error()) | ||||
| 	} | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func CacheGetUserQuota(ctx context.Context, id int) (quota int64, err error) { | ||||
| 	if !common.RedisEnabled { | ||||
| 		return GetUserQuota(id) | ||||
| 	} | ||||
| 	quotaString, err := common.RedisGet(fmt.Sprintf("user_quota:%d", id)) | ||||
| 	if err != nil { | ||||
| 		quota, err = GetUserQuota(id) | ||||
| 		if err != nil { | ||||
| 			return 0, err | ||||
| 		} | ||||
| 		err = common.RedisSet(fmt.Sprintf("user_quota:%d", id), fmt.Sprintf("%d", quota), time.Duration(UserId2QuotaCacheSeconds)*time.Second) | ||||
| 		if err != nil { | ||||
| 			logger.SysError("Redis set user quota error: " + err.Error()) | ||||
| 		} | ||||
| 		return quota, err | ||||
| 		return fetchAndUpdateUserQuota(ctx, id) | ||||
| 	} | ||||
| 	quota, err = strconv.Atoi(quotaString) | ||||
| 	return quota, err | ||||
| 	quota, err = strconv.ParseInt(quotaString, 10, 64) | ||||
| 	if err != nil { | ||||
| 		return 0, nil | ||||
| 	} | ||||
| 	if quota <= config.PreConsumedQuota { // when user's quota is less than pre-consumed quota, we need to fetch from db | ||||
| 		logger.Infof(ctx, "user %d's cached quota is too low: %d, refreshing from db", quota, id) | ||||
| 		return fetchAndUpdateUserQuota(ctx, id) | ||||
| 	} | ||||
| 	return quota, nil | ||||
| } | ||||
|  | ||||
| func CacheUpdateUserQuota(id int) error { | ||||
| func CacheUpdateUserQuota(ctx context.Context, id int) error { | ||||
| 	if !common.RedisEnabled { | ||||
| 		return nil | ||||
| 	} | ||||
| 	quota, err := CacheGetUserQuota(id) | ||||
| 	quota, err := CacheGetUserQuota(ctx, id) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| @@ -102,7 +115,7 @@ func CacheUpdateUserQuota(id int) error { | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func CacheDecreaseUserQuota(id int, quota int) error { | ||||
| func CacheDecreaseUserQuota(id int, quota int64) error { | ||||
| 	if !common.RedisEnabled { | ||||
| 		return nil | ||||
| 	} | ||||
| @@ -134,6 +147,25 @@ func CacheIsUserEnabled(userId int) (bool, error) { | ||||
| 	return userEnabled, err | ||||
| } | ||||
|  | ||||
| func CacheGetGroupModels(ctx context.Context, group string) ([]string, error) { | ||||
| 	if !common.RedisEnabled { | ||||
| 		return GetGroupModels(ctx, group) | ||||
| 	} | ||||
| 	modelsStr, err := common.RedisGet(fmt.Sprintf("group_models:%s", group)) | ||||
| 	if err == nil { | ||||
| 		return strings.Split(modelsStr, ","), nil | ||||
| 	} | ||||
| 	models, err := GetGroupModels(ctx, group) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	err = common.RedisSet(fmt.Sprintf("group_models:%s", group), strings.Join(models, ","), time.Duration(GroupModelsCacheSeconds)*time.Second) | ||||
| 	if err != nil { | ||||
| 		logger.SysError("Redis set group models error: " + err.Error()) | ||||
| 	} | ||||
| 	return models, nil | ||||
| } | ||||
|  | ||||
| var group2model2channels map[string]map[string][]*Channel | ||||
| var channelSyncLock sync.RWMutex | ||||
|  | ||||
| @@ -193,7 +225,7 @@ func SyncChannelCache(frequency int) { | ||||
|  | ||||
| func CacheGetRandomSatisfiedChannel(group string, model string, ignoreFirstPriority bool) (*Channel, error) { | ||||
| 	if !config.MemoryCacheEnabled { | ||||
| 		return GetRandomSatisfiedChannel(group, model) | ||||
| 		return GetRandomSatisfiedChannel(group, model, ignoreFirstPriority) | ||||
| 	} | ||||
| 	channelSyncLock.RLock() | ||||
| 	defer channelSyncLock.RUnlock() | ||||
|   | ||||
| @@ -13,7 +13,7 @@ import ( | ||||
| type Channel struct { | ||||
| 	Id                 int     `json:"id"` | ||||
| 	Type               int     `json:"type" gorm:"default:0"` | ||||
| 	Key                string  `json:"key" gorm:"not null;index"` | ||||
| 	Key                string  `json:"key" gorm:"type:text"` | ||||
| 	Status             int     `json:"status" gorm:"default:1"` | ||||
| 	Name               string  `json:"name" gorm:"index"` | ||||
| 	Weight             *uint   `json:"weight" gorm:"default:0"` | ||||
| @@ -47,11 +47,7 @@ func GetAllChannels(startIdx int, num int, scope string) ([]*Channel, error) { | ||||
| } | ||||
|  | ||||
| func SearchChannels(keyword string) (channels []*Channel, err error) { | ||||
| 	keyCol := "`key`" | ||||
| 	if common.UsingPostgreSQL { | ||||
| 		keyCol = `"key"` | ||||
| 	} | ||||
| 	err = DB.Omit("key").Where("id = ? or name LIKE ? or "+keyCol+" = ?", helper.String2Int(keyword), keyword+"%", keyword).Find(&channels).Error | ||||
| 	err = DB.Omit("key").Where("id = ? or name LIKE ?", helper.String2Int(keyword), keyword+"%").Find(&channels).Error | ||||
| 	return channels, err | ||||
| } | ||||
|  | ||||
| @@ -182,7 +178,7 @@ func UpdateChannelStatusById(id int, status int) { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func UpdateChannelUsedQuota(id int, quota int) { | ||||
| func UpdateChannelUsedQuota(id int, quota int64) { | ||||
| 	if config.BatchUpdateEnabled { | ||||
| 		addNewRecord(BatchUpdateTypeChannelUsedQuota, id, quota) | ||||
| 		return | ||||
| @@ -190,7 +186,7 @@ func UpdateChannelUsedQuota(id int, quota int) { | ||||
| 	updateChannelUsedQuota(id, quota) | ||||
| } | ||||
|  | ||||
| func updateChannelUsedQuota(id int, quota int) { | ||||
| func updateChannelUsedQuota(id int, quota int64) { | ||||
| 	err := DB.Model(&Channel{}).Where("id = ?", id).Update("used_quota", gorm.Expr("used_quota + ?", quota)).Error | ||||
| 	if err != nil { | ||||
| 		logger.SysError("failed to update channel used quota: " + err.Error()) | ||||
|   | ||||
							
								
								
									
										30
									
								
								model/log.go
									
									
									
									
									
								
							
							
						
						
									
										30
									
								
								model/log.go
									
									
									
									
									
								
							| @@ -45,13 +45,13 @@ func RecordLog(userId int, logType int, content string) { | ||||
| 		Type:      logType, | ||||
| 		Content:   content, | ||||
| 	} | ||||
| 	err := DB.Create(log).Error | ||||
| 	err := LOG_DB.Create(log).Error | ||||
| 	if err != nil { | ||||
| 		logger.SysError("failed to record log: " + err.Error()) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func RecordConsumeLog(ctx context.Context, userId int, channelId int, promptTokens int, completionTokens int, modelName string, tokenName string, quota int, content string) { | ||||
| func RecordConsumeLog(ctx context.Context, userId int, channelId int, promptTokens int, completionTokens int, modelName string, tokenName string, quota int64, content string) { | ||||
| 	logger.Info(ctx, fmt.Sprintf("record consume log: userId=%d, channelId=%d, promptTokens=%d, completionTokens=%d, modelName=%s, tokenName=%s, quota=%d, content=%s", userId, channelId, promptTokens, completionTokens, modelName, tokenName, quota, content)) | ||||
| 	if !config.LogConsumeEnabled { | ||||
| 		return | ||||
| @@ -66,10 +66,10 @@ func RecordConsumeLog(ctx context.Context, userId int, channelId int, promptToke | ||||
| 		CompletionTokens: completionTokens, | ||||
| 		TokenName:        tokenName, | ||||
| 		ModelName:        modelName, | ||||
| 		Quota:            quota, | ||||
| 		Quota:            int(quota), | ||||
| 		ChannelId:        channelId, | ||||
| 	} | ||||
| 	err := DB.Create(log).Error | ||||
| 	err := LOG_DB.Create(log).Error | ||||
| 	if err != nil { | ||||
| 		logger.Error(ctx, "failed to record log: "+err.Error()) | ||||
| 	} | ||||
| @@ -78,9 +78,9 @@ func RecordConsumeLog(ctx context.Context, userId int, channelId int, promptToke | ||||
| func GetAllLogs(logType int, startTimestamp int64, endTimestamp int64, modelName string, username string, tokenName string, startIdx int, num int, channel int) (logs []*Log, err error) { | ||||
| 	var tx *gorm.DB | ||||
| 	if logType == LogTypeUnknown { | ||||
| 		tx = DB | ||||
| 		tx = LOG_DB | ||||
| 	} else { | ||||
| 		tx = DB.Where("type = ?", logType) | ||||
| 		tx = LOG_DB.Where("type = ?", logType) | ||||
| 	} | ||||
| 	if modelName != "" { | ||||
| 		tx = tx.Where("model_name = ?", modelName) | ||||
| @@ -107,9 +107,9 @@ func GetAllLogs(logType int, startTimestamp int64, endTimestamp int64, modelName | ||||
| func GetUserLogs(userId int, logType int, startTimestamp int64, endTimestamp int64, modelName string, tokenName string, startIdx int, num int) (logs []*Log, err error) { | ||||
| 	var tx *gorm.DB | ||||
| 	if logType == LogTypeUnknown { | ||||
| 		tx = DB.Where("user_id = ?", userId) | ||||
| 		tx = LOG_DB.Where("user_id = ?", userId) | ||||
| 	} else { | ||||
| 		tx = DB.Where("user_id = ? and type = ?", userId, logType) | ||||
| 		tx = LOG_DB.Where("user_id = ? and type = ?", userId, logType) | ||||
| 	} | ||||
| 	if modelName != "" { | ||||
| 		tx = tx.Where("model_name = ?", modelName) | ||||
| @@ -128,17 +128,17 @@ func GetUserLogs(userId int, logType int, startTimestamp int64, endTimestamp int | ||||
| } | ||||
|  | ||||
| func SearchAllLogs(keyword string) (logs []*Log, err error) { | ||||
| 	err = DB.Where("type = ? or content LIKE ?", keyword, keyword+"%").Order("id desc").Limit(config.MaxRecentItems).Find(&logs).Error | ||||
| 	err = LOG_DB.Where("type = ? or content LIKE ?", keyword, keyword+"%").Order("id desc").Limit(config.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(config.MaxRecentItems).Omit("id").Find(&logs).Error | ||||
| 	err = LOG_DB.Where("user_id = ? and type = ?", userId, keyword).Order("id desc").Limit(config.MaxRecentItems).Omit("id").Find(&logs).Error | ||||
| 	return logs, err | ||||
| } | ||||
|  | ||||
| func SumUsedQuota(logType int, startTimestamp int64, endTimestamp int64, modelName string, username string, tokenName string, channel int) (quota int) { | ||||
| 	tx := DB.Table("logs").Select("ifnull(sum(quota),0)") | ||||
| func SumUsedQuota(logType int, startTimestamp int64, endTimestamp int64, modelName string, username string, tokenName string, channel int) (quota int64) { | ||||
| 	tx := LOG_DB.Table("logs").Select("ifnull(sum(quota),0)") | ||||
| 	if username != "" { | ||||
| 		tx = tx.Where("username = ?", username) | ||||
| 	} | ||||
| @@ -162,7 +162,7 @@ func SumUsedQuota(logType int, startTimestamp int64, endTimestamp int64, modelNa | ||||
| } | ||||
|  | ||||
| func SumUsedToken(logType int, startTimestamp int64, endTimestamp int64, modelName string, username string, tokenName string) (token int) { | ||||
| 	tx := DB.Table("logs").Select("ifnull(sum(prompt_tokens),0) + ifnull(sum(completion_tokens),0)") | ||||
| 	tx := LOG_DB.Table("logs").Select("ifnull(sum(prompt_tokens),0) + ifnull(sum(completion_tokens),0)") | ||||
| 	if username != "" { | ||||
| 		tx = tx.Where("username = ?", username) | ||||
| 	} | ||||
| @@ -183,7 +183,7 @@ func SumUsedToken(logType int, startTimestamp int64, endTimestamp int64, modelNa | ||||
| } | ||||
|  | ||||
| func DeleteOldLog(targetTimestamp int64) (int64, error) { | ||||
| 	result := DB.Where("created_at < ?", targetTimestamp).Delete(&Log{}) | ||||
| 	result := LOG_DB.Where("created_at < ?", targetTimestamp).Delete(&Log{}) | ||||
| 	return result.RowsAffected, result.Error | ||||
| } | ||||
|  | ||||
| @@ -207,7 +207,7 @@ func SearchLogsByDayAndModel(userId, start, end int) (LogStatistics []*LogStatis | ||||
| 		groupSelect = "strftime('%Y-%m-%d', datetime(created_at, 'unixepoch')) as day" | ||||
| 	} | ||||
|  | ||||
| 	err = DB.Raw(` | ||||
| 	err = LOG_DB.Raw(` | ||||
| 		SELECT `+groupSelect+`, | ||||
| 		model_name, count(1) as request_count, | ||||
| 		sum(quota) as quota, | ||||
|   | ||||
| @@ -4,6 +4,7 @@ import ( | ||||
| 	"fmt" | ||||
| 	"github.com/songquanpeng/one-api/common" | ||||
| 	"github.com/songquanpeng/one-api/common/config" | ||||
| 	"github.com/songquanpeng/one-api/common/env" | ||||
| 	"github.com/songquanpeng/one-api/common/helper" | ||||
| 	"github.com/songquanpeng/one-api/common/logger" | ||||
| 	"gorm.io/driver/mysql" | ||||
| @@ -16,12 +17,13 @@ import ( | ||||
| ) | ||||
|  | ||||
| var DB *gorm.DB | ||||
| var LOG_DB *gorm.DB | ||||
|  | ||||
| func createRootAccountIfNeed() error { | ||||
| func CreateRootAccountIfNeed() error { | ||||
| 	var user User | ||||
| 	//if user.Status != util.UserStatusEnabled { | ||||
| 	if err := DB.First(&user).Error; err != nil { | ||||
| 		logger.SysLog("no user exists, create a root user for you: username is root, password is 123456") | ||||
| 		logger.SysLog("no user exists, creating a root user for you: username is root, password is 123456") | ||||
| 		hashedPassword, err := common.Password2Hash("123456") | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| @@ -33,16 +35,32 @@ func createRootAccountIfNeed() error { | ||||
| 			Status:      common.UserStatusEnabled, | ||||
| 			DisplayName: "Root User", | ||||
| 			AccessToken: helper.GetUUID(), | ||||
| 			Quota:       100000000, | ||||
| 			Quota:       500000000000000, | ||||
| 		} | ||||
| 		DB.Create(&rootUser) | ||||
| 		if config.InitialRootToken != "" { | ||||
| 			logger.SysLog("creating initial root token as requested") | ||||
| 			token := Token{ | ||||
| 				Id:             1, | ||||
| 				UserId:         rootUser.Id, | ||||
| 				Key:            config.InitialRootToken, | ||||
| 				Status:         common.TokenStatusEnabled, | ||||
| 				Name:           "Initial Root Token", | ||||
| 				CreatedTime:    helper.GetTimestamp(), | ||||
| 				AccessedTime:   helper.GetTimestamp(), | ||||
| 				ExpiredTime:    -1, | ||||
| 				RemainQuota:    500000000000000, | ||||
| 				UnlimitedQuota: true, | ||||
| 			} | ||||
| 			DB.Create(&token) | ||||
| 		} | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func chooseDB() (*gorm.DB, error) { | ||||
| 	if os.Getenv("SQL_DSN") != "" { | ||||
| 		dsn := os.Getenv("SQL_DSN") | ||||
| func chooseDB(envName string) (*gorm.DB, error) { | ||||
| 	if os.Getenv(envName) != "" { | ||||
| 		dsn := os.Getenv(envName) | ||||
| 		if strings.HasPrefix(dsn, "postgres://") { | ||||
| 			// Use PostgreSQL | ||||
| 			logger.SysLog("using PostgreSQL as database") | ||||
| @@ -56,6 +74,7 @@ func chooseDB() (*gorm.DB, error) { | ||||
| 		} | ||||
| 		// Use MySQL | ||||
| 		logger.SysLog("using MySQL as database") | ||||
| 		common.UsingMySQL = true | ||||
| 		return gorm.Open(mysql.Open(dsn), &gorm.Config{ | ||||
| 			PrepareStmt: true, // precompile SQL | ||||
| 		}) | ||||
| @@ -69,67 +88,78 @@ func chooseDB() (*gorm.DB, error) { | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func InitDB() (err error) { | ||||
| 	db, err := chooseDB() | ||||
| func InitDB(envName string) (db *gorm.DB, err error) { | ||||
| 	db, err = chooseDB(envName) | ||||
| 	if err == nil { | ||||
| 		if config.DebugSQLEnabled { | ||||
| 			db = db.Debug() | ||||
| 		} | ||||
| 		DB = db | ||||
| 		sqlDB, err := DB.DB() | ||||
| 		sqlDB, err := db.DB() | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 			return nil, err | ||||
| 		} | ||||
| 		sqlDB.SetMaxIdleConns(helper.GetOrDefaultEnvInt("SQL_MAX_IDLE_CONNS", 100)) | ||||
| 		sqlDB.SetMaxOpenConns(helper.GetOrDefaultEnvInt("SQL_MAX_OPEN_CONNS", 1000)) | ||||
| 		sqlDB.SetConnMaxLifetime(time.Second * time.Duration(helper.GetOrDefaultEnvInt("SQL_MAX_LIFETIME", 60))) | ||||
| 		sqlDB.SetMaxIdleConns(env.Int("SQL_MAX_IDLE_CONNS", 100)) | ||||
| 		sqlDB.SetMaxOpenConns(env.Int("SQL_MAX_OPEN_CONNS", 1000)) | ||||
| 		sqlDB.SetConnMaxLifetime(time.Second * time.Duration(env.Int("SQL_MAX_LIFETIME", 60))) | ||||
|  | ||||
| 		if !config.IsMasterNode { | ||||
| 			return nil | ||||
| 			return db, err | ||||
| 		} | ||||
| 		if common.UsingMySQL { | ||||
| 			_, _ = sqlDB.Exec("DROP INDEX idx_channels_key ON channels;") // TODO: delete this line when most users have upgraded | ||||
| 		} | ||||
| 		logger.SysLog("database migration started") | ||||
| 		err = db.AutoMigrate(&Channel{}) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 			return nil, err | ||||
| 		} | ||||
| 		err = db.AutoMigrate(&Token{}) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 			return nil, err | ||||
| 		} | ||||
| 		err = db.AutoMigrate(&User{}) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 			return nil, err | ||||
| 		} | ||||
| 		err = db.AutoMigrate(&Option{}) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 			return nil, err | ||||
| 		} | ||||
| 		err = db.AutoMigrate(&Redemption{}) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 			return nil, err | ||||
| 		} | ||||
| 		err = db.AutoMigrate(&Ability{}) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 			return nil, err | ||||
| 		} | ||||
| 		err = db.AutoMigrate(&Log{}) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 			return nil, err | ||||
| 		} | ||||
| 		logger.SysLog("database migrated") | ||||
| 		err = createRootAccountIfNeed() | ||||
| 		return err | ||||
| 		return db, err | ||||
| 	} else { | ||||
| 		logger.FatalLog(err) | ||||
| 	} | ||||
| 	return err | ||||
| 	return db, err | ||||
| } | ||||
|  | ||||
| func CloseDB() error { | ||||
| 	sqlDB, err := DB.DB() | ||||
| func closeDB(db *gorm.DB) error { | ||||
| 	sqlDB, err := db.DB() | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	err = sqlDB.Close() | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func CloseDB() error { | ||||
| 	if LOG_DB != DB { | ||||
| 		err := closeDB(LOG_DB) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
| 	return closeDB(DB) | ||||
| } | ||||
|   | ||||
| @@ -61,11 +61,11 @@ func InitOptionMap() { | ||||
| 	config.OptionMap["MessagePusherToken"] = "" | ||||
| 	config.OptionMap["TurnstileSiteKey"] = "" | ||||
| 	config.OptionMap["TurnstileSecretKey"] = "" | ||||
| 	config.OptionMap["QuotaForNewUser"] = strconv.Itoa(config.QuotaForNewUser) | ||||
| 	config.OptionMap["QuotaForInviter"] = strconv.Itoa(config.QuotaForInviter) | ||||
| 	config.OptionMap["QuotaForInvitee"] = strconv.Itoa(config.QuotaForInvitee) | ||||
| 	config.OptionMap["QuotaRemindThreshold"] = strconv.Itoa(config.QuotaRemindThreshold) | ||||
| 	config.OptionMap["PreConsumedQuota"] = strconv.Itoa(config.PreConsumedQuota) | ||||
| 	config.OptionMap["QuotaForNewUser"] = strconv.FormatInt(config.QuotaForNewUser, 10) | ||||
| 	config.OptionMap["QuotaForInviter"] = strconv.FormatInt(config.QuotaForInviter, 10) | ||||
| 	config.OptionMap["QuotaForInvitee"] = strconv.FormatInt(config.QuotaForInvitee, 10) | ||||
| 	config.OptionMap["QuotaRemindThreshold"] = strconv.FormatInt(config.QuotaRemindThreshold, 10) | ||||
| 	config.OptionMap["PreConsumedQuota"] = strconv.FormatInt(config.PreConsumedQuota, 10) | ||||
| 	config.OptionMap["ModelRatio"] = common.ModelRatio2JSONString() | ||||
| 	config.OptionMap["GroupRatio"] = common.GroupRatio2JSONString() | ||||
| 	config.OptionMap["CompletionRatio"] = common.CompletionRatio2JSONString() | ||||
| @@ -81,6 +81,9 @@ func InitOptionMap() { | ||||
| func loadOptionsFromDatabase() { | ||||
| 	options, _ := AllOption() | ||||
| 	for _, option := range options { | ||||
| 		if option.Key == "ModelRatio" { | ||||
| 			option.Value = common.AddNewMissingRatio(option.Value) | ||||
| 		} | ||||
| 		err := updateOptionMap(option.Key, option.Value) | ||||
| 		if err != nil { | ||||
| 			logger.SysError("failed to update option map: " + err.Error()) | ||||
| @@ -190,15 +193,15 @@ func updateOptionMap(key string, value string) (err error) { | ||||
| 	case "TurnstileSecretKey": | ||||
| 		config.TurnstileSecretKey = value | ||||
| 	case "QuotaForNewUser": | ||||
| 		config.QuotaForNewUser, _ = strconv.Atoi(value) | ||||
| 		config.QuotaForNewUser, _ = strconv.ParseInt(value, 10, 64) | ||||
| 	case "QuotaForInviter": | ||||
| 		config.QuotaForInviter, _ = strconv.Atoi(value) | ||||
| 		config.QuotaForInviter, _ = strconv.ParseInt(value, 10, 64) | ||||
| 	case "QuotaForInvitee": | ||||
| 		config.QuotaForInvitee, _ = strconv.Atoi(value) | ||||
| 		config.QuotaForInvitee, _ = strconv.ParseInt(value, 10, 64) | ||||
| 	case "QuotaRemindThreshold": | ||||
| 		config.QuotaRemindThreshold, _ = strconv.Atoi(value) | ||||
| 		config.QuotaRemindThreshold, _ = strconv.ParseInt(value, 10, 64) | ||||
| 	case "PreConsumedQuota": | ||||
| 		config.PreConsumedQuota, _ = strconv.Atoi(value) | ||||
| 		config.PreConsumedQuota, _ = strconv.ParseInt(value, 10, 64) | ||||
| 	case "RetryTimes": | ||||
| 		config.RetryTimes, _ = strconv.Atoi(value) | ||||
| 	case "ModelRatio": | ||||
|   | ||||
| @@ -14,7 +14,7 @@ type Redemption struct { | ||||
| 	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"` | ||||
| 	Quota        int64  `json:"quota" gorm:"bigint;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 | ||||
| @@ -42,7 +42,7 @@ func GetRedemptionById(id int) (*Redemption, error) { | ||||
| 	return &redemption, err | ||||
| } | ||||
|  | ||||
| func Redeem(key string, userId int) (quota int, err error) { | ||||
| func Redeem(key string, userId int) (quota int64, err error) { | ||||
| 	if key == "" { | ||||
| 		return 0, errors.New("未提供兑换码") | ||||
| 	} | ||||
|   | ||||
| @@ -12,23 +12,35 @@ import ( | ||||
| ) | ||||
|  | ||||
| type Token struct { | ||||
| 	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"` | ||||
| 	UsedQuota      int    `json:"used_quota" gorm:"default:0"` // used quota | ||||
| 	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    int64   `json:"remain_quota" gorm:"bigint;default:0"` | ||||
| 	UnlimitedQuota bool    `json:"unlimited_quota" gorm:"default:false"` | ||||
| 	UsedQuota      int64   `json:"used_quota" gorm:"bigint;default:0"` // used quota | ||||
| 	Models         *string `json:"models" gorm:"default:''"` | ||||
| } | ||||
|  | ||||
| func GetAllUserTokens(userId int, startIdx int, num int) ([]*Token, error) { | ||||
| func GetAllUserTokens(userId int, startIdx int, num int, order string) ([]*Token, error) { | ||||
| 	var tokens []*Token | ||||
| 	var err error | ||||
| 	err = DB.Where("user_id = ?", userId).Order("id desc").Limit(num).Offset(startIdx).Find(&tokens).Error | ||||
| 	query := DB.Where("user_id = ?", userId) | ||||
|  | ||||
| 	switch order { | ||||
| 	case "remain_quota": | ||||
| 		query = query.Order("unlimited_quota desc, remain_quota desc") | ||||
| 	case "used_quota": | ||||
| 		query = query.Order("used_quota desc") | ||||
| 	default: | ||||
| 		query = query.Order("id desc") | ||||
| 	} | ||||
|  | ||||
| 	err = query.Limit(num).Offset(startIdx).Find(&tokens).Error | ||||
| 	return tokens, err | ||||
| } | ||||
|  | ||||
| @@ -110,7 +122,7 @@ func (token *Token) Insert() error { | ||||
| // 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).Select("name", "status", "expired_time", "remain_quota", "unlimited_quota").Updates(token).Error | ||||
| 	err = DB.Model(token).Select("name", "status", "expired_time", "remain_quota", "unlimited_quota", "models").Updates(token).Error | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| @@ -138,7 +150,7 @@ func DeleteTokenById(id int, userId int) (err error) { | ||||
| 	return token.Delete() | ||||
| } | ||||
|  | ||||
| func IncreaseTokenQuota(id int, quota int) (err error) { | ||||
| func IncreaseTokenQuota(id int, quota int64) (err error) { | ||||
| 	if quota < 0 { | ||||
| 		return errors.New("quota 不能为负数!") | ||||
| 	} | ||||
| @@ -149,7 +161,7 @@ func IncreaseTokenQuota(id int, quota int) (err error) { | ||||
| 	return increaseTokenQuota(id, quota) | ||||
| } | ||||
|  | ||||
| func increaseTokenQuota(id int, quota int) (err error) { | ||||
| func increaseTokenQuota(id int, quota int64) (err error) { | ||||
| 	err = DB.Model(&Token{}).Where("id = ?", id).Updates( | ||||
| 		map[string]interface{}{ | ||||
| 			"remain_quota":  gorm.Expr("remain_quota + ?", quota), | ||||
| @@ -160,7 +172,7 @@ func increaseTokenQuota(id int, quota int) (err error) { | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func DecreaseTokenQuota(id int, quota int) (err error) { | ||||
| func DecreaseTokenQuota(id int, quota int64) (err error) { | ||||
| 	if quota < 0 { | ||||
| 		return errors.New("quota 不能为负数!") | ||||
| 	} | ||||
| @@ -171,7 +183,7 @@ func DecreaseTokenQuota(id int, quota int) (err error) { | ||||
| 	return decreaseTokenQuota(id, quota) | ||||
| } | ||||
|  | ||||
| func decreaseTokenQuota(id int, quota int) (err error) { | ||||
| func decreaseTokenQuota(id int, quota int64) (err error) { | ||||
| 	err = DB.Model(&Token{}).Where("id = ?", id).Updates( | ||||
| 		map[string]interface{}{ | ||||
| 			"remain_quota":  gorm.Expr("remain_quota - ?", quota), | ||||
| @@ -182,7 +194,7 @@ func decreaseTokenQuota(id int, quota int) (err error) { | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func PreConsumeTokenQuota(tokenId int, quota int) (err error) { | ||||
| func PreConsumeTokenQuota(tokenId int, quota int64) (err error) { | ||||
| 	if quota < 0 { | ||||
| 		return errors.New("quota 不能为负数!") | ||||
| 	} | ||||
| @@ -232,7 +244,7 @@ func PreConsumeTokenQuota(tokenId int, quota int) (err error) { | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func PostConsumeTokenQuota(tokenId int, quota int) (err error) { | ||||
| func PostConsumeTokenQuota(tokenId int, quota int64) (err error) { | ||||
| 	token, err := GetTokenById(tokenId) | ||||
| 	if quota > 0 { | ||||
| 		err = DecreaseUserQuota(token.UserId, quota) | ||||
|   | ||||
| @@ -26,9 +26,9 @@ type User struct { | ||||
| 	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! | ||||
| 	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 | ||||
| 	Quota            int64  `json:"quota" gorm:"bigint;default:0"` | ||||
| 	UsedQuota        int64  `json:"used_quota" gorm:"bigint;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"` | ||||
| @@ -40,9 +40,22 @@ func GetMaxUserId() int { | ||||
| 	return user.Id | ||||
| } | ||||
|  | ||||
| func GetAllUsers(startIdx int, num int) (users []*User, err error) { | ||||
| 	err = DB.Order("id desc").Limit(num).Offset(startIdx).Omit("password").Where("status != ?", common.UserStatusDeleted).Find(&users).Error | ||||
| 	return users, err | ||||
| func GetAllUsers(startIdx int, num int, order string) (users []*User, err error) { | ||||
|     query := DB.Limit(num).Offset(startIdx).Omit("password").Where("status != ?", common.UserStatusDeleted) | ||||
|      | ||||
|     switch order { | ||||
|     case "quota": | ||||
|         query = query.Order("quota desc") | ||||
|     case "used_quota": | ||||
|         query = query.Order("used_quota desc") | ||||
|     case "request_count": | ||||
|         query = query.Order("request_count desc") | ||||
|     default: | ||||
|         query = query.Order("id desc") | ||||
|     } | ||||
|      | ||||
|     err = query.Find(&users).Error | ||||
|     return users, err | ||||
| } | ||||
|  | ||||
| func SearchUsers(keyword string) (users []*User, err error) { | ||||
| @@ -274,12 +287,12 @@ func ValidateAccessToken(token string) (user *User) { | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func GetUserQuota(id int) (quota int, err error) { | ||||
| func GetUserQuota(id int) (quota int64, 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) { | ||||
| func GetUserUsedQuota(id int) (quota int64, err error) { | ||||
| 	err = DB.Model(&User{}).Where("id = ?", id).Select("used_quota").Find("a).Error | ||||
| 	return quota, err | ||||
| } | ||||
| @@ -299,7 +312,7 @@ func GetUserGroup(id int) (group string, err error) { | ||||
| 	return group, err | ||||
| } | ||||
|  | ||||
| func IncreaseUserQuota(id int, quota int) (err error) { | ||||
| func IncreaseUserQuota(id int, quota int64) (err error) { | ||||
| 	if quota < 0 { | ||||
| 		return errors.New("quota 不能为负数!") | ||||
| 	} | ||||
| @@ -310,12 +323,12 @@ func IncreaseUserQuota(id int, quota int) (err error) { | ||||
| 	return increaseUserQuota(id, quota) | ||||
| } | ||||
|  | ||||
| func increaseUserQuota(id int, quota int) (err error) { | ||||
| func increaseUserQuota(id int, quota int64) (err error) { | ||||
| 	err = DB.Model(&User{}).Where("id = ?", id).Update("quota", gorm.Expr("quota + ?", quota)).Error | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func DecreaseUserQuota(id int, quota int) (err error) { | ||||
| func DecreaseUserQuota(id int, quota int64) (err error) { | ||||
| 	if quota < 0 { | ||||
| 		return errors.New("quota 不能为负数!") | ||||
| 	} | ||||
| @@ -326,7 +339,7 @@ func DecreaseUserQuota(id int, quota int) (err error) { | ||||
| 	return decreaseUserQuota(id, quota) | ||||
| } | ||||
|  | ||||
| func decreaseUserQuota(id int, quota int) (err error) { | ||||
| func decreaseUserQuota(id int, quota int64) (err error) { | ||||
| 	err = DB.Model(&User{}).Where("id = ?", id).Update("quota", gorm.Expr("quota - ?", quota)).Error | ||||
| 	return err | ||||
| } | ||||
| @@ -336,7 +349,7 @@ func GetRootUserEmail() (email string) { | ||||
| 	return email | ||||
| } | ||||
|  | ||||
| func UpdateUserUsedQuotaAndRequestCount(id int, quota int) { | ||||
| func UpdateUserUsedQuotaAndRequestCount(id int, quota int64) { | ||||
| 	if config.BatchUpdateEnabled { | ||||
| 		addNewRecord(BatchUpdateTypeUsedQuota, id, quota) | ||||
| 		addNewRecord(BatchUpdateTypeRequestCount, id, 1) | ||||
| @@ -345,7 +358,7 @@ func UpdateUserUsedQuotaAndRequestCount(id int, quota int) { | ||||
| 	updateUserUsedQuotaAndRequestCount(id, quota, 1) | ||||
| } | ||||
|  | ||||
| func updateUserUsedQuotaAndRequestCount(id int, quota int, count int) { | ||||
| func updateUserUsedQuotaAndRequestCount(id int, quota int64, count int) { | ||||
| 	err := DB.Model(&User{}).Where("id = ?", id).Updates( | ||||
| 		map[string]interface{}{ | ||||
| 			"used_quota":    gorm.Expr("used_quota + ?", quota), | ||||
| @@ -357,7 +370,7 @@ func updateUserUsedQuotaAndRequestCount(id int, quota int, count int) { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func updateUserUsedQuota(id int, quota int) { | ||||
| func updateUserUsedQuota(id int, quota int64) { | ||||
| 	err := DB.Model(&User{}).Where("id = ?", id).Updates( | ||||
| 		map[string]interface{}{ | ||||
| 			"used_quota": gorm.Expr("used_quota + ?", quota), | ||||
|   | ||||
| @@ -16,12 +16,12 @@ const ( | ||||
| 	BatchUpdateTypeCount // if you add a new type, you need to add a new map and a new lock | ||||
| ) | ||||
|  | ||||
| var batchUpdateStores []map[int]int | ||||
| var batchUpdateStores []map[int]int64 | ||||
| var batchUpdateLocks []sync.Mutex | ||||
|  | ||||
| func init() { | ||||
| 	for i := 0; i < BatchUpdateTypeCount; i++ { | ||||
| 		batchUpdateStores = append(batchUpdateStores, make(map[int]int)) | ||||
| 		batchUpdateStores = append(batchUpdateStores, make(map[int]int64)) | ||||
| 		batchUpdateLocks = append(batchUpdateLocks, sync.Mutex{}) | ||||
| 	} | ||||
| } | ||||
| @@ -35,7 +35,7 @@ func InitBatchUpdater() { | ||||
| 	}() | ||||
| } | ||||
|  | ||||
| func addNewRecord(type_ int, id int, value int) { | ||||
| func addNewRecord(type_ int, id int, value int64) { | ||||
| 	batchUpdateLocks[type_].Lock() | ||||
| 	defer batchUpdateLocks[type_].Unlock() | ||||
| 	if _, ok := batchUpdateStores[type_][id]; !ok { | ||||
| @@ -50,7 +50,7 @@ func batchUpdate() { | ||||
| 	for i := 0; i < BatchUpdateTypeCount; i++ { | ||||
| 		batchUpdateLocks[i].Lock() | ||||
| 		store := batchUpdateStores[i] | ||||
| 		batchUpdateStores[i] = make(map[int]int) | ||||
| 		batchUpdateStores[i] = make(map[int]int64) | ||||
| 		batchUpdateLocks[i].Unlock() | ||||
| 		// TODO: maybe we can combine updates with same key? | ||||
| 		for key, value := range store { | ||||
| @@ -68,7 +68,7 @@ func batchUpdate() { | ||||
| 			case BatchUpdateTypeUsedQuota: | ||||
| 				updateUserUsedQuota(key, value) | ||||
| 			case BatchUpdateTypeRequestCount: | ||||
| 				updateUserRequestCount(key, value) | ||||
| 				updateUserRequestCount(key, int(value)) | ||||
| 			case BatchUpdateTypeChannelUsedQuota: | ||||
| 				updateChannelUsedQuota(key, value) | ||||
| 			} | ||||
|   | ||||
| @@ -31,17 +31,17 @@ func notifyRootUser(subject string, content string) { | ||||
| func DisableChannel(channelId int, channelName string, reason string) { | ||||
| 	model.UpdateChannelStatusById(channelId, common.ChannelStatusAutoDisabled) | ||||
| 	logger.SysLog(fmt.Sprintf("channel #%d has been disabled: %s", channelId, reason)) | ||||
| 	subject := fmt.Sprintf("通道「%s」(#%d)已被禁用", channelName, channelId) | ||||
| 	content := fmt.Sprintf("通道「%s」(#%d)已被禁用,原因:%s", channelName, channelId, reason) | ||||
| 	subject := fmt.Sprintf("渠道「%s」(#%d)已被禁用", channelName, channelId) | ||||
| 	content := fmt.Sprintf("渠道「%s」(#%d)已被禁用,原因:%s", channelName, channelId, reason) | ||||
| 	notifyRootUser(subject, content) | ||||
| } | ||||
|  | ||||
| func MetricDisableChannel(channelId int, successRate float64) { | ||||
| 	model.UpdateChannelStatusById(channelId, common.ChannelStatusAutoDisabled) | ||||
| 	logger.SysLog(fmt.Sprintf("channel #%d has been disabled due to low success rate: %.2f", channelId, successRate*100)) | ||||
| 	subject := fmt.Sprintf("通道 #%d 已被禁用", channelId) | ||||
| 	content := fmt.Sprintf("该渠道在最近 %d 次调用中成功率为 %.2f%%,低于阈值 %.2f%%,因此被系统自动禁用。", | ||||
| 		config.MetricQueueSize, successRate*100, config.MetricSuccessRateThreshold*100) | ||||
| 	subject := fmt.Sprintf("渠道 #%d 已被禁用", channelId) | ||||
| 	content := fmt.Sprintf("该渠道(#%d)在最近 %d 次调用中成功率为 %.2f%%,低于阈值 %.2f%%,因此被系统自动禁用。", | ||||
| 		channelId, config.MetricQueueSize, successRate*100, config.MetricSuccessRateThreshold*100) | ||||
| 	notifyRootUser(subject, content) | ||||
| } | ||||
|  | ||||
| @@ -49,7 +49,7 @@ func MetricDisableChannel(channelId int, successRate float64) { | ||||
| func EnableChannel(channelId int, channelName string) { | ||||
| 	model.UpdateChannelStatusById(channelId, common.ChannelStatusEnabled) | ||||
| 	logger.SysLog(fmt.Sprintf("channel #%d has been enabled", channelId)) | ||||
| 	subject := fmt.Sprintf("通道「%s」(#%d)已被启用", channelName, channelId) | ||||
| 	content := fmt.Sprintf("通道「%s」(#%d)已被启用", channelName, channelId) | ||||
| 	subject := fmt.Sprintf("渠道「%s」(#%d)已被启用", channelName, channelId) | ||||
| 	content := fmt.Sprintf("渠道「%s」(#%d)已被启用", channelName, channelId) | ||||
| 	notifyRootUser(subject, content) | ||||
| } | ||||
|   | ||||
| @@ -1,9 +1,10 @@ | ||||
| [//]: # (请按照以下格式关联 issue) | ||||
| [//]: # (请在提交 PR 前确认所提交的功能可用,附上截图即可,这将有助于项目维护者 review & merge 该 PR,谢谢) | ||||
| [//]: # (请在提交 PR 前确认所提交的功能可用,需要附上截图,谢谢) | ||||
| [//]: # (项目维护者一般仅在周末处理 PR,因此如若未能及时回复希望能理解) | ||||
| [//]: # (开发者交流群:910657413) | ||||
| [//]: # (请在提交 PR 之前删除上面的注释) | ||||
|  | ||||
| close #issue_number | ||||
|  | ||||
| 我已确认该 PR 已自测通过,相关截图如下: | ||||
| 我已确认该 PR 已自测通过,相关截图如下: | ||||
| (此处放上测试通过的截图,如果不涉及前端改动或从 UI 上无法看出,请放终端启动成功的截图) | ||||
|   | ||||
| @@ -32,6 +32,9 @@ func (a *Adaptor) GetRequestURL(meta *util.RelayMeta) (string, error) { | ||||
|  | ||||
| func (a *Adaptor) SetupRequestHeader(c *gin.Context, req *http.Request, meta *util.RelayMeta) error { | ||||
| 	channel.SetupCommonRequestHeader(c, req, meta) | ||||
| 	if meta.IsStream { | ||||
| 		req.Header.Set("Accept", "text/event-stream") | ||||
| 	} | ||||
| 	req.Header.Set("Authorization", "Bearer "+meta.APIKey) | ||||
| 	if meta.IsStream { | ||||
| 		req.Header.Set("X-DashScope-SSE", "enable") | ||||
|   | ||||
| @@ -48,6 +48,9 @@ func ConvertRequest(request model.GeneralOpenAIRequest) *ChatRequest { | ||||
| 			MaxTokens:         request.MaxTokens, | ||||
| 			Temperature:       request.Temperature, | ||||
| 			TopP:              request.TopP, | ||||
| 			TopK:              request.TopK, | ||||
| 			ResultFormat:      "message", | ||||
| 			Tools:             request.Tools, | ||||
| 		}, | ||||
| 	} | ||||
| } | ||||
| @@ -117,19 +120,11 @@ func embeddingResponseAli2OpenAI(response *EmbeddingResponse) *openai.EmbeddingR | ||||
| } | ||||
|  | ||||
| func responseAli2OpenAI(response *ChatResponse) *openai.TextResponse { | ||||
| 	choice := openai.TextResponseChoice{ | ||||
| 		Index: 0, | ||||
| 		Message: model.Message{ | ||||
| 			Role:    "assistant", | ||||
| 			Content: response.Output.Text, | ||||
| 		}, | ||||
| 		FinishReason: response.Output.FinishReason, | ||||
| 	} | ||||
| 	fullTextResponse := openai.TextResponse{ | ||||
| 		Id:      response.RequestId, | ||||
| 		Object:  "chat.completion", | ||||
| 		Created: helper.GetTimestamp(), | ||||
| 		Choices: []openai.TextResponseChoice{choice}, | ||||
| 		Choices: response.Output.Choices, | ||||
| 		Usage: model.Usage{ | ||||
| 			PromptTokens:     response.Usage.InputTokens, | ||||
| 			CompletionTokens: response.Usage.OutputTokens, | ||||
| @@ -140,10 +135,14 @@ func responseAli2OpenAI(response *ChatResponse) *openai.TextResponse { | ||||
| } | ||||
|  | ||||
| func streamResponseAli2OpenAI(aliResponse *ChatResponse) *openai.ChatCompletionsStreamResponse { | ||||
| 	if len(aliResponse.Output.Choices) == 0 { | ||||
| 		return nil | ||||
| 	} | ||||
| 	aliChoice := aliResponse.Output.Choices[0] | ||||
| 	var choice openai.ChatCompletionsStreamResponseChoice | ||||
| 	choice.Delta.Content = aliResponse.Output.Text | ||||
| 	if aliResponse.Output.FinishReason != "null" { | ||||
| 		finishReason := aliResponse.Output.FinishReason | ||||
| 	choice.Delta = aliChoice.Message | ||||
| 	if aliChoice.FinishReason != "null" { | ||||
| 		finishReason := aliChoice.FinishReason | ||||
| 		choice.FinishReason = &finishReason | ||||
| 	} | ||||
| 	response := openai.ChatCompletionsStreamResponse{ | ||||
| @@ -204,6 +203,9 @@ func StreamHandler(c *gin.Context, resp *http.Response) (*model.ErrorWithStatusC | ||||
| 				usage.TotalTokens = aliResponse.Usage.InputTokens + aliResponse.Usage.OutputTokens | ||||
| 			} | ||||
| 			response := streamResponseAli2OpenAI(&aliResponse) | ||||
| 			if response == nil { | ||||
| 				return true | ||||
| 			} | ||||
| 			//response.Choices[0].Delta.Content = strings.TrimPrefix(response.Choices[0].Delta.Content, lastResponseText) | ||||
| 			//lastResponseText = aliResponse.Output.Text | ||||
| 			jsonResponse, err := json.Marshal(response) | ||||
| @@ -226,6 +228,7 @@ func StreamHandler(c *gin.Context, resp *http.Response) (*model.ErrorWithStatusC | ||||
| } | ||||
|  | ||||
| func Handler(c *gin.Context, resp *http.Response) (*model.ErrorWithStatusCode, *model.Usage) { | ||||
| 	ctx := c.Request.Context() | ||||
| 	var aliResponse ChatResponse | ||||
| 	responseBody, err := io.ReadAll(resp.Body) | ||||
| 	if err != nil { | ||||
| @@ -235,6 +238,7 @@ func Handler(c *gin.Context, resp *http.Response) (*model.ErrorWithStatusCode, * | ||||
| 	if err != nil { | ||||
| 		return openai.ErrorWrapper(err, "close_response_body_failed", http.StatusInternalServerError), nil | ||||
| 	} | ||||
| 	logger.Debugf(ctx, "response body: %s\n", responseBody) | ||||
| 	err = json.Unmarshal(responseBody, &aliResponse) | ||||
| 	if err != nil { | ||||
| 		return openai.ErrorWrapper(err, "unmarshal_response_body_failed", http.StatusInternalServerError), nil | ||||
|   | ||||
| @@ -1,5 +1,10 @@ | ||||
| package ali | ||||
|  | ||||
| import ( | ||||
| 	"github.com/songquanpeng/one-api/relay/channel/openai" | ||||
| 	"github.com/songquanpeng/one-api/relay/model" | ||||
| ) | ||||
|  | ||||
| type Message struct { | ||||
| 	Content string `json:"content"` | ||||
| 	Role    string `json:"role"` | ||||
| @@ -11,13 +16,15 @@ type Input struct { | ||||
| } | ||||
|  | ||||
| type Parameters struct { | ||||
| 	TopP              float64 `json:"top_p,omitempty"` | ||||
| 	TopK              int     `json:"top_k,omitempty"` | ||||
| 	Seed              uint64  `json:"seed,omitempty"` | ||||
| 	EnableSearch      bool    `json:"enable_search,omitempty"` | ||||
| 	IncrementalOutput bool    `json:"incremental_output,omitempty"` | ||||
| 	MaxTokens         int     `json:"max_tokens,omitempty"` | ||||
| 	Temperature       float64 `json:"temperature,omitempty"` | ||||
| 	TopP              float64      `json:"top_p,omitempty"` | ||||
| 	TopK              int          `json:"top_k,omitempty"` | ||||
| 	Seed              uint64       `json:"seed,omitempty"` | ||||
| 	EnableSearch      bool         `json:"enable_search,omitempty"` | ||||
| 	IncrementalOutput bool         `json:"incremental_output,omitempty"` | ||||
| 	MaxTokens         int          `json:"max_tokens,omitempty"` | ||||
| 	Temperature       float64      `json:"temperature,omitempty"` | ||||
| 	ResultFormat      string       `json:"result_format,omitempty"` | ||||
| 	Tools             []model.Tool `json:"tools,omitempty"` | ||||
| } | ||||
|  | ||||
| type ChatRequest struct { | ||||
| @@ -62,8 +69,9 @@ type Usage struct { | ||||
| } | ||||
|  | ||||
| type Output struct { | ||||
| 	Text         string `json:"text"` | ||||
| 	FinishReason string `json:"finish_reason"` | ||||
| 	//Text         string                      `json:"text"` | ||||
| 	//FinishReason string                      `json:"finish_reason"` | ||||
| 	Choices []openai.TextResponseChoice `json:"choices"` | ||||
| } | ||||
|  | ||||
| type ChatResponse struct { | ||||
|   | ||||
| @@ -59,5 +59,5 @@ func (a *Adaptor) GetModelList() []string { | ||||
| } | ||||
|  | ||||
| func (a *Adaptor) GetChannelName() string { | ||||
| 	return "authropic" | ||||
| 	return "anthropic" | ||||
| } | ||||
|   | ||||
| @@ -2,7 +2,7 @@ package anthropic | ||||
|  | ||||
| var ModelList = []string{ | ||||
| 	"claude-instant-1.2", "claude-2.0", "claude-2.1", | ||||
| 	"claude-3-haiku-20240229", | ||||
| 	"claude-3-haiku-20240307", | ||||
| 	"claude-3-sonnet-20240229", | ||||
| 	"claude-3-opus-20240229", | ||||
| } | ||||
|   | ||||
| @@ -38,6 +38,7 @@ func ConvertRequest(textRequest model.GeneralOpenAIRequest) *Request { | ||||
| 		MaxTokens:   textRequest.MaxTokens, | ||||
| 		Temperature: textRequest.Temperature, | ||||
| 		TopP:        textRequest.TopP, | ||||
| 		TopK:        textRequest.TopK, | ||||
| 		Stream:      textRequest.Stream, | ||||
| 	} | ||||
| 	if claudeRequest.MaxTokens == 0 { | ||||
|   | ||||
| @@ -3,14 +3,15 @@ package baidu | ||||
| import ( | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"io" | ||||
| 	"net/http" | ||||
| 	"strings" | ||||
|  | ||||
| 	"github.com/gin-gonic/gin" | ||||
| 	"github.com/songquanpeng/one-api/relay/channel" | ||||
| 	"github.com/songquanpeng/one-api/relay/constant" | ||||
| 	"github.com/songquanpeng/one-api/relay/model" | ||||
| 	"github.com/songquanpeng/one-api/relay/util" | ||||
| 	"io" | ||||
| 	"net/http" | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| type Adaptor struct { | ||||
| @@ -23,7 +24,13 @@ func (a *Adaptor) Init(meta *util.RelayMeta) { | ||||
| func (a *Adaptor) GetRequestURL(meta *util.RelayMeta) (string, error) { | ||||
| 	// https://cloud.baidu.com/doc/WENXINWORKSHOP/s/clntwmv7t | ||||
| 	suffix := "chat/" | ||||
| 	if strings.HasPrefix("Embedding", meta.ActualModelName) { | ||||
| 	if strings.HasPrefix(meta.ActualModelName, "Embedding") { | ||||
| 		suffix = "embeddings/" | ||||
| 	} | ||||
| 	if strings.HasPrefix(meta.ActualModelName, "bge-large") { | ||||
| 		suffix = "embeddings/" | ||||
| 	} | ||||
| 	if strings.HasPrefix(meta.ActualModelName, "tao-8k") { | ||||
| 		suffix = "embeddings/" | ||||
| 	} | ||||
| 	switch meta.ActualModelName { | ||||
| @@ -45,6 +52,12 @@ func (a *Adaptor) GetRequestURL(meta *util.RelayMeta) (string, error) { | ||||
| 		suffix += "bloomz_7b1" | ||||
| 	case "Embedding-V1": | ||||
| 		suffix += "embedding-v1" | ||||
| 	case "bge-large-zh": | ||||
| 		suffix += "bge_large_zh" | ||||
| 	case "bge-large-en": | ||||
| 		suffix += "bge_large_en" | ||||
| 	case "tao-8k": | ||||
| 		suffix += "tao_8k" | ||||
| 	default: | ||||
| 		suffix += meta.ActualModelName | ||||
| 	} | ||||
|   | ||||
| @@ -7,4 +7,7 @@ var ModelList = []string{ | ||||
| 	"ERNIE-Speed", | ||||
| 	"ERNIE-Bot-turbo", | ||||
| 	"Embedding-V1", | ||||
| 	"bge-large-zh", | ||||
| 	"bge-large-en", | ||||
| 	"tao-8k", | ||||
| } | ||||
|   | ||||
| @@ -32,9 +32,16 @@ type Message struct { | ||||
| } | ||||
|  | ||||
| type ChatRequest struct { | ||||
| 	Messages []Message `json:"messages"` | ||||
| 	Stream   bool      `json:"stream"` | ||||
| 	UserId   string    `json:"user_id,omitempty"` | ||||
| 	Messages        []Message `json:"messages"` | ||||
| 	Temperature     float64   `json:"temperature,omitempty"` | ||||
| 	TopP            float64   `json:"top_p,omitempty"` | ||||
| 	PenaltyScore    float64   `json:"penalty_score,omitempty"` | ||||
| 	Stream          bool      `json:"stream,omitempty"` | ||||
| 	System          string    `json:"system,omitempty"` | ||||
| 	DisableSearch   bool      `json:"disable_search,omitempty"` | ||||
| 	EnableCitation  bool      `json:"enable_citation,omitempty"` | ||||
| 	MaxOutputTokens int       `json:"max_output_tokens,omitempty"` | ||||
| 	UserId          string    `json:"user_id,omitempty"` | ||||
| } | ||||
|  | ||||
| type Error struct { | ||||
| @@ -45,28 +52,28 @@ type Error struct { | ||||
| var baiduTokenStore sync.Map | ||||
|  | ||||
| func ConvertRequest(request model.GeneralOpenAIRequest) *ChatRequest { | ||||
| 	messages := make([]Message, 0, len(request.Messages)) | ||||
| 	baiduRequest := ChatRequest{ | ||||
| 		Messages:        make([]Message, 0, len(request.Messages)), | ||||
| 		Temperature:     request.Temperature, | ||||
| 		TopP:            request.TopP, | ||||
| 		PenaltyScore:    request.FrequencyPenalty, | ||||
| 		Stream:          request.Stream, | ||||
| 		DisableSearch:   false, | ||||
| 		EnableCitation:  false, | ||||
| 		MaxOutputTokens: request.MaxTokens, | ||||
| 		UserId:          request.User, | ||||
| 	} | ||||
| 	for _, message := range request.Messages { | ||||
| 		if message.Role == "system" { | ||||
| 			messages = append(messages, Message{ | ||||
| 				Role:    "user", | ||||
| 				Content: message.StringContent(), | ||||
| 			}) | ||||
| 			messages = append(messages, Message{ | ||||
| 				Role:    "assistant", | ||||
| 				Content: "Okay", | ||||
| 			}) | ||||
| 			baiduRequest.System = message.StringContent() | ||||
| 		} else { | ||||
| 			messages = append(messages, Message{ | ||||
| 			baiduRequest.Messages = append(baiduRequest.Messages, Message{ | ||||
| 				Role:    message.Role, | ||||
| 				Content: message.StringContent(), | ||||
| 			}) | ||||
| 		} | ||||
| 	} | ||||
| 	return &ChatRequest{ | ||||
| 		Messages: messages, | ||||
| 		Stream:   request.Stream, | ||||
| 	} | ||||
| 	return &baiduRequest | ||||
| } | ||||
|  | ||||
| func responseBaidu2OpenAI(response *ChatResponse) *openai.TextResponse { | ||||
|   | ||||
| @@ -1,6 +1,8 @@ | ||||
| package gemini | ||||
|  | ||||
| // https://ai.google.dev/models/gemini | ||||
|  | ||||
| var ModelList = []string{ | ||||
| 	"gemini-pro", "gemini-1.0-pro-001", | ||||
| 	"gemini-pro", "gemini-1.0-pro-001", "gemini-1.5-pro", | ||||
| 	"gemini-pro-vision", "gemini-1.0-pro-vision-001", | ||||
| } | ||||
|   | ||||
							
								
								
									
										9
									
								
								relay/channel/lingyiwanwu/constants.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								relay/channel/lingyiwanwu/constants.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | ||||
| package lingyiwanwu | ||||
|  | ||||
| // https://platform.lingyiwanwu.com/docs | ||||
|  | ||||
| var ModelList = []string{ | ||||
| 	"yi-34b-chat-0205", | ||||
| 	"yi-34b-chat-200k", | ||||
| 	"yi-vl-plus", | ||||
| } | ||||
							
								
								
									
										75
									
								
								relay/channel/ollama/adaptor.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										75
									
								
								relay/channel/ollama/adaptor.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,75 @@ | ||||
| package ollama | ||||
|  | ||||
| import ( | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"io" | ||||
| 	"net/http" | ||||
|  | ||||
| 	"github.com/gin-gonic/gin" | ||||
| 	"github.com/songquanpeng/one-api/relay/channel" | ||||
| 	"github.com/songquanpeng/one-api/relay/constant" | ||||
| 	"github.com/songquanpeng/one-api/relay/model" | ||||
| 	"github.com/songquanpeng/one-api/relay/util" | ||||
| ) | ||||
|  | ||||
| type Adaptor struct { | ||||
| } | ||||
|  | ||||
| func (a *Adaptor) Init(meta *util.RelayMeta) { | ||||
|  | ||||
| } | ||||
|  | ||||
| func (a *Adaptor) GetRequestURL(meta *util.RelayMeta) (string, error) { | ||||
| 	// https://github.com/ollama/ollama/blob/main/docs/api.md | ||||
| 	fullRequestURL := fmt.Sprintf("%s/api/chat", meta.BaseURL) | ||||
| 	if meta.Mode == constant.RelayModeEmbeddings { | ||||
| 		fullRequestURL = fmt.Sprintf("%s/api/embeddings", meta.BaseURL) | ||||
| 	} | ||||
| 	return fullRequestURL, nil | ||||
| } | ||||
|  | ||||
| func (a *Adaptor) SetupRequestHeader(c *gin.Context, req *http.Request, meta *util.RelayMeta) error { | ||||
| 	channel.SetupCommonRequestHeader(c, req, meta) | ||||
| 	req.Header.Set("Authorization", "Bearer "+meta.APIKey) | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (a *Adaptor) ConvertRequest(c *gin.Context, relayMode int, request *model.GeneralOpenAIRequest) (any, error) { | ||||
| 	if request == nil { | ||||
| 		return nil, errors.New("request is nil") | ||||
| 	} | ||||
| 	switch relayMode { | ||||
| 	case constant.RelayModeEmbeddings: | ||||
| 		ollamaEmbeddingRequest := ConvertEmbeddingRequest(*request) | ||||
| 		return ollamaEmbeddingRequest, nil | ||||
| 	default: | ||||
| 		return ConvertRequest(*request), nil | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (a *Adaptor) DoRequest(c *gin.Context, meta *util.RelayMeta, requestBody io.Reader) (*http.Response, error) { | ||||
| 	return channel.DoRequestHelper(a, c, meta, requestBody) | ||||
| } | ||||
|  | ||||
| func (a *Adaptor) DoResponse(c *gin.Context, resp *http.Response, meta *util.RelayMeta) (usage *model.Usage, err *model.ErrorWithStatusCode) { | ||||
| 	if meta.IsStream { | ||||
| 		err, usage = StreamHandler(c, resp) | ||||
| 	} else { | ||||
| 		switch meta.Mode { | ||||
| 		case constant.RelayModeEmbeddings: | ||||
| 			err, usage = EmbeddingHandler(c, resp) | ||||
| 		default: | ||||
| 			err, usage = Handler(c, resp) | ||||
| 		} | ||||
| 	} | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func (a *Adaptor) GetModelList() []string { | ||||
| 	return ModelList | ||||
| } | ||||
|  | ||||
| func (a *Adaptor) GetChannelName() string { | ||||
| 	return "ollama" | ||||
| } | ||||
							
								
								
									
										5
									
								
								relay/channel/ollama/constants.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								relay/channel/ollama/constants.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | ||||
| package ollama | ||||
|  | ||||
| var ModelList = []string{ | ||||
| 	"qwen:0.5b-chat", | ||||
| } | ||||
							
								
								
									
										237
									
								
								relay/channel/ollama/main.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										237
									
								
								relay/channel/ollama/main.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,237 @@ | ||||
| package ollama | ||||
|  | ||||
| import ( | ||||
| 	"bufio" | ||||
| 	"context" | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"io" | ||||
| 	"net/http" | ||||
| 	"strings" | ||||
|  | ||||
| 	"github.com/gin-gonic/gin" | ||||
| 	"github.com/songquanpeng/one-api/common" | ||||
| 	"github.com/songquanpeng/one-api/common/helper" | ||||
| 	"github.com/songquanpeng/one-api/common/logger" | ||||
| 	"github.com/songquanpeng/one-api/relay/channel/openai" | ||||
| 	"github.com/songquanpeng/one-api/relay/constant" | ||||
| 	"github.com/songquanpeng/one-api/relay/model" | ||||
| ) | ||||
|  | ||||
| func ConvertRequest(request model.GeneralOpenAIRequest) *ChatRequest { | ||||
| 	ollamaRequest := ChatRequest{ | ||||
| 		Model: request.Model, | ||||
| 		Options: &Options{ | ||||
| 			Seed:             int(request.Seed), | ||||
| 			Temperature:      request.Temperature, | ||||
| 			TopP:             request.TopP, | ||||
| 			FrequencyPenalty: request.FrequencyPenalty, | ||||
| 			PresencePenalty:  request.PresencePenalty, | ||||
| 		}, | ||||
| 		Stream: request.Stream, | ||||
| 	} | ||||
| 	for _, message := range request.Messages { | ||||
| 		ollamaRequest.Messages = append(ollamaRequest.Messages, Message{ | ||||
| 			Role:    message.Role, | ||||
| 			Content: message.StringContent(), | ||||
| 		}) | ||||
| 	} | ||||
| 	return &ollamaRequest | ||||
| } | ||||
|  | ||||
| func responseOllama2OpenAI(response *ChatResponse) *openai.TextResponse { | ||||
| 	choice := openai.TextResponseChoice{ | ||||
| 		Index: 0, | ||||
| 		Message: model.Message{ | ||||
| 			Role:    response.Message.Role, | ||||
| 			Content: response.Message.Content, | ||||
| 		}, | ||||
| 	} | ||||
| 	if response.Done { | ||||
| 		choice.FinishReason = "stop" | ||||
| 	} | ||||
| 	fullTextResponse := openai.TextResponse{ | ||||
| 		Id:      fmt.Sprintf("chatcmpl-%s", helper.GetUUID()), | ||||
| 		Object:  "chat.completion", | ||||
| 		Created: helper.GetTimestamp(), | ||||
| 		Choices: []openai.TextResponseChoice{choice}, | ||||
| 		Usage: model.Usage{ | ||||
| 			PromptTokens:     response.PromptEvalCount, | ||||
| 			CompletionTokens: response.EvalCount, | ||||
| 			TotalTokens:      response.PromptEvalCount + response.EvalCount, | ||||
| 		}, | ||||
| 	} | ||||
| 	return &fullTextResponse | ||||
| } | ||||
|  | ||||
| func streamResponseOllama2OpenAI(ollamaResponse *ChatResponse) *openai.ChatCompletionsStreamResponse { | ||||
| 	var choice openai.ChatCompletionsStreamResponseChoice | ||||
| 	choice.Delta.Role = ollamaResponse.Message.Role | ||||
| 	choice.Delta.Content = ollamaResponse.Message.Content | ||||
| 	if ollamaResponse.Done { | ||||
| 		choice.FinishReason = &constant.StopFinishReason | ||||
| 	} | ||||
| 	response := openai.ChatCompletionsStreamResponse{ | ||||
| 		Id:      fmt.Sprintf("chatcmpl-%s", helper.GetUUID()), | ||||
| 		Object:  "chat.completion.chunk", | ||||
| 		Created: helper.GetTimestamp(), | ||||
| 		Model:   ollamaResponse.Model, | ||||
| 		Choices: []openai.ChatCompletionsStreamResponseChoice{choice}, | ||||
| 	} | ||||
| 	return &response | ||||
| } | ||||
|  | ||||
| func StreamHandler(c *gin.Context, resp *http.Response) (*model.ErrorWithStatusCode, *model.Usage) { | ||||
| 	var usage model.Usage | ||||
| 	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"); 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 := strings.TrimPrefix(scanner.Text(), "}") | ||||
| 			dataChan <- data + "}" | ||||
| 		} | ||||
| 		stopChan <- true | ||||
| 	}() | ||||
| 	common.SetEventStreamHeaders(c) | ||||
| 	c.Stream(func(w io.Writer) bool { | ||||
| 		select { | ||||
| 		case data := <-dataChan: | ||||
| 			var ollamaResponse ChatResponse | ||||
| 			err := json.Unmarshal([]byte(data), &ollamaResponse) | ||||
| 			if err != nil { | ||||
| 				logger.SysError("error unmarshalling stream response: " + err.Error()) | ||||
| 				return true | ||||
| 			} | ||||
| 			if ollamaResponse.EvalCount != 0 { | ||||
| 				usage.PromptTokens = ollamaResponse.PromptEvalCount | ||||
| 				usage.CompletionTokens = ollamaResponse.EvalCount | ||||
| 				usage.TotalTokens = ollamaResponse.PromptEvalCount + ollamaResponse.EvalCount | ||||
| 			} | ||||
| 			response := streamResponseOllama2OpenAI(&ollamaResponse) | ||||
| 			jsonResponse, err := json.Marshal(response) | ||||
| 			if err != nil { | ||||
| 				logger.SysError("error marshalling stream response: " + err.Error()) | ||||
| 				return true | ||||
| 			} | ||||
| 			c.Render(-1, common.CustomEvent{Data: "data: " + string(jsonResponse)}) | ||||
| 			return true | ||||
| 		case <-stopChan: | ||||
| 			c.Render(-1, common.CustomEvent{Data: "data: [DONE]"}) | ||||
| 			return false | ||||
| 		} | ||||
| 	}) | ||||
| 	err := resp.Body.Close() | ||||
| 	if err != nil { | ||||
| 		return openai.ErrorWrapper(err, "close_response_body_failed", http.StatusInternalServerError), nil | ||||
| 	} | ||||
| 	return nil, &usage | ||||
| } | ||||
|  | ||||
| func ConvertEmbeddingRequest(request model.GeneralOpenAIRequest) *EmbeddingRequest { | ||||
| 	return &EmbeddingRequest{ | ||||
| 		Model:  request.Model, | ||||
| 		Prompt: strings.Join(request.ParseInput(), " "), | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func EmbeddingHandler(c *gin.Context, resp *http.Response) (*model.ErrorWithStatusCode, *model.Usage) { | ||||
| 	var ollamaResponse EmbeddingResponse | ||||
| 	err := json.NewDecoder(resp.Body).Decode(&ollamaResponse) | ||||
| 	if err != nil { | ||||
| 		return openai.ErrorWrapper(err, "unmarshal_response_body_failed", http.StatusInternalServerError), nil | ||||
| 	} | ||||
|  | ||||
| 	err = resp.Body.Close() | ||||
| 	if err != nil { | ||||
| 		return openai.ErrorWrapper(err, "close_response_body_failed", http.StatusInternalServerError), nil | ||||
| 	} | ||||
|  | ||||
| 	if ollamaResponse.Error != "" { | ||||
| 		return &model.ErrorWithStatusCode{ | ||||
| 			Error: model.Error{ | ||||
| 				Message: ollamaResponse.Error, | ||||
| 				Type:    "ollama_error", | ||||
| 				Param:   "", | ||||
| 				Code:    "ollama_error", | ||||
| 			}, | ||||
| 			StatusCode: resp.StatusCode, | ||||
| 		}, nil | ||||
| 	} | ||||
|  | ||||
| 	fullTextResponse := embeddingResponseOllama2OpenAI(&ollamaResponse) | ||||
| 	jsonResponse, err := json.Marshal(fullTextResponse) | ||||
| 	if err != nil { | ||||
| 		return openai.ErrorWrapper(err, "marshal_response_body_failed", http.StatusInternalServerError), nil | ||||
| 	} | ||||
| 	c.Writer.Header().Set("Content-Type", "application/json") | ||||
| 	c.Writer.WriteHeader(resp.StatusCode) | ||||
| 	_, err = c.Writer.Write(jsonResponse) | ||||
| 	return nil, &fullTextResponse.Usage | ||||
| } | ||||
|  | ||||
| func embeddingResponseOllama2OpenAI(response *EmbeddingResponse) *openai.EmbeddingResponse { | ||||
| 	openAIEmbeddingResponse := openai.EmbeddingResponse{ | ||||
| 		Object: "list", | ||||
| 		Data:   make([]openai.EmbeddingResponseItem, 0, 1), | ||||
| 		Model:  "text-embedding-v1", | ||||
| 		Usage:  model.Usage{TotalTokens: 0}, | ||||
| 	} | ||||
|  | ||||
| 	openAIEmbeddingResponse.Data = append(openAIEmbeddingResponse.Data, openai.EmbeddingResponseItem{ | ||||
| 		Object:    `embedding`, | ||||
| 		Index:     0, | ||||
| 		Embedding: response.Embedding, | ||||
| 	}) | ||||
| 	return &openAIEmbeddingResponse | ||||
| } | ||||
|  | ||||
| func Handler(c *gin.Context, resp *http.Response) (*model.ErrorWithStatusCode, *model.Usage) { | ||||
| 	ctx := context.TODO() | ||||
| 	var ollamaResponse ChatResponse | ||||
| 	responseBody, err := io.ReadAll(resp.Body) | ||||
| 	if err != nil { | ||||
| 		return openai.ErrorWrapper(err, "read_response_body_failed", http.StatusInternalServerError), nil | ||||
| 	} | ||||
| 	logger.Debugf(ctx, "ollama response: %s", string(responseBody)) | ||||
| 	err = resp.Body.Close() | ||||
| 	if err != nil { | ||||
| 		return openai.ErrorWrapper(err, "close_response_body_failed", http.StatusInternalServerError), nil | ||||
| 	} | ||||
| 	err = json.Unmarshal(responseBody, &ollamaResponse) | ||||
| 	if err != nil { | ||||
| 		return openai.ErrorWrapper(err, "unmarshal_response_body_failed", http.StatusInternalServerError), nil | ||||
| 	} | ||||
| 	if ollamaResponse.Error != "" { | ||||
| 		return &model.ErrorWithStatusCode{ | ||||
| 			Error: model.Error{ | ||||
| 				Message: ollamaResponse.Error, | ||||
| 				Type:    "ollama_error", | ||||
| 				Param:   "", | ||||
| 				Code:    "ollama_error", | ||||
| 			}, | ||||
| 			StatusCode: resp.StatusCode, | ||||
| 		}, nil | ||||
| 	} | ||||
| 	fullTextResponse := responseOllama2OpenAI(&ollamaResponse) | ||||
| 	jsonResponse, err := json.Marshal(fullTextResponse) | ||||
| 	if err != nil { | ||||
| 		return openai.ErrorWrapper(err, "marshal_response_body_failed", http.StatusInternalServerError), nil | ||||
| 	} | ||||
| 	c.Writer.Header().Set("Content-Type", "application/json") | ||||
| 	c.Writer.WriteHeader(resp.StatusCode) | ||||
| 	_, err = c.Writer.Write(jsonResponse) | ||||
| 	return nil, &fullTextResponse.Usage | ||||
| } | ||||
							
								
								
									
										47
									
								
								relay/channel/ollama/model.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										47
									
								
								relay/channel/ollama/model.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,47 @@ | ||||
| package ollama | ||||
|  | ||||
| type Options struct { | ||||
| 	Seed             int     `json:"seed,omitempty"` | ||||
| 	Temperature      float64 `json:"temperature,omitempty"` | ||||
| 	TopK             int     `json:"top_k,omitempty"` | ||||
| 	TopP             float64 `json:"top_p,omitempty"` | ||||
| 	FrequencyPenalty float64 `json:"frequency_penalty,omitempty"` | ||||
| 	PresencePenalty  float64 `json:"presence_penalty,omitempty"` | ||||
| } | ||||
|  | ||||
| type Message struct { | ||||
| 	Role    string   `json:"role,omitempty"` | ||||
| 	Content string   `json:"content,omitempty"` | ||||
| 	Images  []string `json:"images,omitempty"` | ||||
| } | ||||
|  | ||||
| type ChatRequest struct { | ||||
| 	Model    string    `json:"model,omitempty"` | ||||
| 	Messages []Message `json:"messages,omitempty"` | ||||
| 	Stream   bool      `json:"stream"` | ||||
| 	Options  *Options  `json:"options,omitempty"` | ||||
| } | ||||
|  | ||||
| type ChatResponse struct { | ||||
| 	Model           string  `json:"model,omitempty"` | ||||
| 	CreatedAt       string  `json:"created_at,omitempty"` | ||||
| 	Message         Message `json:"message,omitempty"` | ||||
| 	Response        string  `json:"response,omitempty"` // for stream response | ||||
| 	Done            bool    `json:"done,omitempty"` | ||||
| 	TotalDuration   int     `json:"total_duration,omitempty"` | ||||
| 	LoadDuration    int     `json:"load_duration,omitempty"` | ||||
| 	PromptEvalCount int     `json:"prompt_eval_count,omitempty"` | ||||
| 	EvalCount       int     `json:"eval_count,omitempty"` | ||||
| 	EvalDuration    int     `json:"eval_duration,omitempty"` | ||||
| 	Error           string  `json:"error,omitempty"` | ||||
| } | ||||
|  | ||||
| type EmbeddingRequest struct { | ||||
| 	Model  string `json:"model"` | ||||
| 	Prompt string `json:"prompt"` | ||||
| } | ||||
|  | ||||
| type EmbeddingResponse struct { | ||||
| 	Error     string    `json:"error,omitempty"` | ||||
| 	Embedding []float64 `json:"embedding,omitempty"` | ||||
| } | ||||
| @@ -31,11 +31,8 @@ func (a *Adaptor) GetRequestURL(meta *util.RelayMeta) (string, error) { | ||||
| 		task := strings.TrimPrefix(requestURL, "/v1/") | ||||
| 		model_ := meta.ActualModelName | ||||
| 		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") | ||||
|  | ||||
| 		//https://github.com/songquanpeng/one-api/issues/1191 | ||||
| 		// {your endpoint}/openai/deployments/{your azure_model}/chat/completions?api-version={api_version} | ||||
| 		requestURL = fmt.Sprintf("/openai/deployments/%s/%s", model_, task) | ||||
| 		return util.GetFullRequestURL(meta.BaseURL, requestURL, meta.ChannelType), nil | ||||
| 	case common.ChannelTypeMinimax: | ||||
| @@ -73,8 +70,10 @@ func (a *Adaptor) DoRequest(c *gin.Context, meta *util.RelayMeta, requestBody io | ||||
| func (a *Adaptor) DoResponse(c *gin.Context, resp *http.Response, meta *util.RelayMeta) (usage *model.Usage, err *model.ErrorWithStatusCode) { | ||||
| 	if meta.IsStream { | ||||
| 		var responseText string | ||||
| 		err, responseText, _ = StreamHandler(c, resp, meta.Mode) | ||||
| 		usage = ResponseText2Usage(responseText, meta.ActualModelName, meta.PromptTokens) | ||||
| 		err, responseText, usage = StreamHandler(c, resp, meta.Mode) | ||||
| 		if usage == nil { | ||||
| 			usage = ResponseText2Usage(responseText, meta.ActualModelName, meta.PromptTokens) | ||||
| 		} | ||||
| 	} else { | ||||
| 		err, usage = Handler(c, resp, meta.PromptTokens, meta.ActualModelName) | ||||
| 	} | ||||
|   | ||||
| @@ -5,6 +5,7 @@ import ( | ||||
| 	"github.com/songquanpeng/one-api/relay/channel/ai360" | ||||
| 	"github.com/songquanpeng/one-api/relay/channel/baichuan" | ||||
| 	"github.com/songquanpeng/one-api/relay/channel/groq" | ||||
| 	"github.com/songquanpeng/one-api/relay/channel/lingyiwanwu" | ||||
| 	"github.com/songquanpeng/one-api/relay/channel/minimax" | ||||
| 	"github.com/songquanpeng/one-api/relay/channel/mistral" | ||||
| 	"github.com/songquanpeng/one-api/relay/channel/moonshot" | ||||
| @@ -18,6 +19,7 @@ var CompatibleChannels = []int{ | ||||
| 	common.ChannelTypeMinimax, | ||||
| 	common.ChannelTypeMistral, | ||||
| 	common.ChannelTypeGroq, | ||||
| 	common.ChannelTypeLingYiWanWu, | ||||
| } | ||||
|  | ||||
| func GetCompatibleChannelMeta(channelType int) (string, []string) { | ||||
| @@ -36,6 +38,8 @@ func GetCompatibleChannelMeta(channelType int) (string, []string) { | ||||
| 		return "mistralai", mistral.ModelList | ||||
| 	case common.ChannelTypeGroq: | ||||
| 		return "groq", groq.ModelList | ||||
| 	case common.ChannelTypeLingYiWanWu: | ||||
| 		return "lingyiwanwu", lingyiwanwu.ModelList | ||||
| 	default: | ||||
| 		return "openai", ModelList | ||||
| 	} | ||||
|   | ||||
| @@ -6,6 +6,7 @@ import ( | ||||
| 	"encoding/json" | ||||
| 	"github.com/gin-gonic/gin" | ||||
| 	"github.com/songquanpeng/one-api/common" | ||||
| 	"github.com/songquanpeng/one-api/common/conv" | ||||
| 	"github.com/songquanpeng/one-api/common/logger" | ||||
| 	"github.com/songquanpeng/one-api/relay/constant" | ||||
| 	"github.com/songquanpeng/one-api/relay/model" | ||||
| @@ -53,7 +54,7 @@ func StreamHandler(c *gin.Context, resp *http.Response, relayMode int) (*model.E | ||||
| 						continue // just ignore the error | ||||
| 					} | ||||
| 					for _, choice := range streamResponse.Choices { | ||||
| 						responseText += choice.Delta.Content | ||||
| 						responseText += conv.AsString(choice.Delta.Content) | ||||
| 					} | ||||
| 					if streamResponse.Usage != nil { | ||||
| 						usage = streamResponse.Usage | ||||
|   | ||||
| @@ -118,12 +118,9 @@ type ImageResponse struct { | ||||
| } | ||||
|  | ||||
| type ChatCompletionsStreamResponseChoice struct { | ||||
| 	Index int `json:"index"` | ||||
| 	Delta struct { | ||||
| 		Content string `json:"content"` | ||||
| 		Role    string `json:"role,omitempty"` | ||||
| 	} `json:"delta"` | ||||
| 	FinishReason *string `json:"finish_reason,omitempty"` | ||||
| 	Index        int           `json:"index"` | ||||
| 	Delta        model.Message `json:"delta"` | ||||
| 	FinishReason *string       `json:"finish_reason,omitempty"` | ||||
| } | ||||
|  | ||||
| type ChatCompletionsStreamResponse struct { | ||||
|   | ||||
| @@ -10,6 +10,7 @@ import ( | ||||
| 	"fmt" | ||||
| 	"github.com/gin-gonic/gin" | ||||
| 	"github.com/songquanpeng/one-api/common" | ||||
| 	"github.com/songquanpeng/one-api/common/conv" | ||||
| 	"github.com/songquanpeng/one-api/common/helper" | ||||
| 	"github.com/songquanpeng/one-api/common/logger" | ||||
| 	"github.com/songquanpeng/one-api/relay/channel/openai" | ||||
| @@ -129,7 +130,7 @@ func StreamHandler(c *gin.Context, resp *http.Response) (*model.ErrorWithStatusC | ||||
| 			} | ||||
| 			response := streamResponseTencent2OpenAI(&TencentResponse) | ||||
| 			if len(response.Choices) != 0 { | ||||
| 				responseText += response.Choices[0].Delta.Content | ||||
| 				responseText += conv.AsString(response.Choices[0].Delta.Content) | ||||
| 			} | ||||
| 			jsonResponse, err := json.Marshal(response) | ||||
| 			if err != nil { | ||||
|   | ||||
| @@ -26,7 +26,11 @@ import ( | ||||
|  | ||||
| func requestOpenAI2Xunfei(request model.GeneralOpenAIRequest, xunfeiAppId string, domain string) *ChatRequest { | ||||
| 	messages := make([]Message, 0, len(request.Messages)) | ||||
| 	var lastToolCalls []model.Tool | ||||
| 	for _, message := range request.Messages { | ||||
| 		if message.ToolCalls != nil { | ||||
| 			lastToolCalls = message.ToolCalls | ||||
| 		} | ||||
| 		messages = append(messages, Message{ | ||||
| 			Role:    message.Role, | ||||
| 			Content: message.StringContent(), | ||||
| @@ -39,9 +43,33 @@ func requestOpenAI2Xunfei(request model.GeneralOpenAIRequest, xunfeiAppId string | ||||
| 	xunfeiRequest.Parameter.Chat.TopK = request.N | ||||
| 	xunfeiRequest.Parameter.Chat.MaxTokens = request.MaxTokens | ||||
| 	xunfeiRequest.Payload.Message.Text = messages | ||||
| 	if len(lastToolCalls) != 0 { | ||||
| 		for _, toolCall := range lastToolCalls { | ||||
| 			xunfeiRequest.Payload.Functions.Text = append(xunfeiRequest.Payload.Functions.Text, toolCall.Function) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return &xunfeiRequest | ||||
| } | ||||
|  | ||||
| func getToolCalls(response *ChatResponse) []model.Tool { | ||||
| 	var toolCalls []model.Tool | ||||
| 	if len(response.Payload.Choices.Text) == 0 { | ||||
| 		return toolCalls | ||||
| 	} | ||||
| 	item := response.Payload.Choices.Text[0] | ||||
| 	if item.FunctionCall == nil { | ||||
| 		return toolCalls | ||||
| 	} | ||||
| 	toolCall := model.Tool{ | ||||
| 		Id:       fmt.Sprintf("call_%s", helper.GetUUID()), | ||||
| 		Type:     "function", | ||||
| 		Function: *item.FunctionCall, | ||||
| 	} | ||||
| 	toolCalls = append(toolCalls, toolCall) | ||||
| 	return toolCalls | ||||
| } | ||||
|  | ||||
| func responseXunfei2OpenAI(response *ChatResponse) *openai.TextResponse { | ||||
| 	if len(response.Payload.Choices.Text) == 0 { | ||||
| 		response.Payload.Choices.Text = []ChatResponseTextItem{ | ||||
| @@ -53,8 +81,9 @@ func responseXunfei2OpenAI(response *ChatResponse) *openai.TextResponse { | ||||
| 	choice := openai.TextResponseChoice{ | ||||
| 		Index: 0, | ||||
| 		Message: model.Message{ | ||||
| 			Role:    "assistant", | ||||
| 			Content: response.Payload.Choices.Text[0].Content, | ||||
| 			Role:      "assistant", | ||||
| 			Content:   response.Payload.Choices.Text[0].Content, | ||||
| 			ToolCalls: getToolCalls(response), | ||||
| 		}, | ||||
| 		FinishReason: constant.StopFinishReason, | ||||
| 	} | ||||
| @@ -78,6 +107,7 @@ func streamResponseXunfei2OpenAI(xunfeiResponse *ChatResponse) *openai.ChatCompl | ||||
| 	} | ||||
| 	var choice openai.ChatCompletionsStreamResponseChoice | ||||
| 	choice.Delta.Content = xunfeiResponse.Payload.Choices.Text[0].Content | ||||
| 	choice.Delta.ToolCalls = getToolCalls(xunfeiResponse) | ||||
| 	if xunfeiResponse.Payload.Choices.Status == 2 { | ||||
| 		choice.FinishReason = &constant.StopFinishReason | ||||
| 	} | ||||
| @@ -121,7 +151,7 @@ func StreamHandler(c *gin.Context, textRequest model.GeneralOpenAIRequest, appId | ||||
| 	domain, authUrl := getXunfeiAuthUrl(c, apiKey, apiSecret, textRequest.Model) | ||||
| 	dataChan, stopChan, err := xunfeiMakeRequest(textRequest, domain, authUrl, appId) | ||||
| 	if err != nil { | ||||
| 		return openai.ErrorWrapper(err, "make xunfei request err", http.StatusInternalServerError), nil | ||||
| 		return openai.ErrorWrapper(err, "xunfei_request_failed", http.StatusInternalServerError), nil | ||||
| 	} | ||||
| 	common.SetEventStreamHeaders(c) | ||||
| 	var usage model.Usage | ||||
| @@ -151,7 +181,7 @@ func Handler(c *gin.Context, textRequest model.GeneralOpenAIRequest, appId strin | ||||
| 	domain, authUrl := getXunfeiAuthUrl(c, apiKey, apiSecret, textRequest.Model) | ||||
| 	dataChan, stopChan, err := xunfeiMakeRequest(textRequest, domain, authUrl, appId) | ||||
| 	if err != nil { | ||||
| 		return openai.ErrorWrapper(err, "make xunfei request err", http.StatusInternalServerError), nil | ||||
| 		return openai.ErrorWrapper(err, "xunfei_request_failed", http.StatusInternalServerError), nil | ||||
| 	} | ||||
| 	var usage model.Usage | ||||
| 	var content string | ||||
| @@ -171,11 +201,7 @@ func Handler(c *gin.Context, textRequest model.GeneralOpenAIRequest, appId strin | ||||
| 		} | ||||
| 	} | ||||
| 	if len(xunfeiResponse.Payload.Choices.Text) == 0 { | ||||
| 		xunfeiResponse.Payload.Choices.Text = []ChatResponseTextItem{ | ||||
| 			{ | ||||
| 				Content: "", | ||||
| 			}, | ||||
| 		} | ||||
| 		return openai.ErrorWrapper(err, "xunfei_empty_response_detected", http.StatusInternalServerError), nil | ||||
| 	} | ||||
| 	xunfeiResponse.Payload.Choices.Text[0].Content = content | ||||
|  | ||||
| @@ -202,15 +228,21 @@ func xunfeiMakeRequest(textRequest model.GeneralOpenAIRequest, domain, authUrl, | ||||
| 	if err != nil { | ||||
| 		return nil, nil, err | ||||
| 	} | ||||
| 	_, msg, err := conn.ReadMessage() | ||||
| 	if err != nil { | ||||
| 		return nil, nil, err | ||||
| 	} | ||||
|  | ||||
| 	dataChan := make(chan ChatResponse) | ||||
| 	stopChan := make(chan bool) | ||||
| 	go func() { | ||||
| 		for { | ||||
| 			_, msg, err := conn.ReadMessage() | ||||
| 			if err != nil { | ||||
| 				logger.SysError("error reading stream response: " + err.Error()) | ||||
| 				break | ||||
| 			if msg == nil { | ||||
| 				_, msg, err = conn.ReadMessage() | ||||
| 				if err != nil { | ||||
| 					logger.SysError("error reading stream response: " + err.Error()) | ||||
| 					break | ||||
| 				} | ||||
| 			} | ||||
| 			var response ChatResponse | ||||
| 			err = json.Unmarshal(msg, &response) | ||||
| @@ -218,6 +250,7 @@ func xunfeiMakeRequest(textRequest model.GeneralOpenAIRequest, domain, authUrl, | ||||
| 				logger.SysError("error unmarshalling stream response: " + err.Error()) | ||||
| 				break | ||||
| 			} | ||||
| 			msg = nil | ||||
| 			dataChan <- response | ||||
| 			if response.Payload.Choices.Status == 2 { | ||||
| 				err := conn.Close() | ||||
|   | ||||
| @@ -26,13 +26,18 @@ type ChatRequest struct { | ||||
| 		Message struct { | ||||
| 			Text []Message `json:"text"` | ||||
| 		} `json:"message"` | ||||
| 		Functions struct { | ||||
| 			Text []model.Function `json:"text,omitempty"` | ||||
| 		} `json:"functions"` | ||||
| 	} `json:"payload"` | ||||
| } | ||||
|  | ||||
| type ChatResponseTextItem struct { | ||||
| 	Content string `json:"content"` | ||||
| 	Role    string `json:"role"` | ||||
| 	Index   int    `json:"index"` | ||||
| 	Content      string          `json:"content"` | ||||
| 	Role         string          `json:"role"` | ||||
| 	Index        int             `json:"index"` | ||||
| 	ContentType  string          `json:"content_type"` | ||||
| 	FunctionCall *model.Function `json:"function_call"` | ||||
| } | ||||
|  | ||||
| type ChatResponse struct { | ||||
|   | ||||
| @@ -9,6 +9,7 @@ import ( | ||||
| 	"github.com/songquanpeng/one-api/relay/model" | ||||
| 	"github.com/songquanpeng/one-api/relay/util" | ||||
| 	"io" | ||||
| 	"math" | ||||
| 	"net/http" | ||||
| 	"strings" | ||||
| ) | ||||
| @@ -52,9 +53,13 @@ func (a *Adaptor) ConvertRequest(c *gin.Context, relayMode int, request *model.G | ||||
| 	if request == nil { | ||||
| 		return nil, errors.New("request is nil") | ||||
| 	} | ||||
| 	if request.TopP >= 1 { | ||||
| 		request.TopP = 0.99 | ||||
| 	} | ||||
| 	// TopP (0.0, 1.0) | ||||
| 	request.TopP = math.Min(0.99, request.TopP) | ||||
| 	request.TopP = math.Max(0.01, request.TopP) | ||||
|  | ||||
| 	// Temperature (0.0, 1.0) | ||||
| 	request.Temperature = math.Min(0.99, request.Temperature) | ||||
| 	request.Temperature = math.Max(0.01, request.Temperature) | ||||
| 	a.SetVersionByModeName(request.Model) | ||||
| 	if a.APIVersion == "v4" { | ||||
| 		return request, nil | ||||
|   | ||||
| @@ -15,6 +15,7 @@ const ( | ||||
| 	APITypeAIProxyLibrary | ||||
| 	APITypeTencent | ||||
| 	APITypeGemini | ||||
| 	APITypeOllama | ||||
|  | ||||
| 	APITypeDummy // this one is only for count, do not add any channel after this | ||||
| ) | ||||
| @@ -40,6 +41,8 @@ func ChannelType2APIType(channelType int) int { | ||||
| 		apiType = APITypeTencent | ||||
| 	case common.ChannelTypeGemini: | ||||
| 		apiType = APITypeGemini | ||||
| 	case common.ChannelTypeOllama: | ||||
| 		apiType = APITypeOllama | ||||
| 	} | ||||
| 	return apiType | ||||
| } | ||||
|   | ||||
| @@ -22,6 +22,7 @@ import ( | ||||
| ) | ||||
|  | ||||
| func RelayAudioHelper(c *gin.Context, relayMode int) *relaymodel.ErrorWithStatusCode { | ||||
| 	ctx := c.Request.Context() | ||||
| 	audioModel := "whisper-1" | ||||
|  | ||||
| 	tokenId := c.GetInt("token_id") | ||||
| @@ -49,16 +50,16 @@ func RelayAudioHelper(c *gin.Context, relayMode int) *relaymodel.ErrorWithStatus | ||||
| 	modelRatio := common.GetModelRatio(audioModel) | ||||
| 	groupRatio := common.GetGroupRatio(group) | ||||
| 	ratio := modelRatio * groupRatio | ||||
| 	var quota int | ||||
| 	var preConsumedQuota int | ||||
| 	var quota int64 | ||||
| 	var preConsumedQuota int64 | ||||
| 	switch relayMode { | ||||
| 	case constant.RelayModeAudioSpeech: | ||||
| 		preConsumedQuota = int(float64(len(ttsRequest.Input)) * ratio) | ||||
| 		preConsumedQuota = int64(float64(len(ttsRequest.Input)) * ratio) | ||||
| 		quota = preConsumedQuota | ||||
| 	default: | ||||
| 		preConsumedQuota = int(float64(config.PreConsumedQuota) * ratio) | ||||
| 		preConsumedQuota = int64(float64(config.PreConsumedQuota) * ratio) | ||||
| 	} | ||||
| 	userQuota, err := model.CacheGetUserQuota(userId) | ||||
| 	userQuota, err := model.CacheGetUserQuota(ctx, userId) | ||||
| 	if err != nil { | ||||
| 		return openai.ErrorWrapper(err, "get_user_quota_failed", http.StatusInternalServerError) | ||||
| 	} | ||||
| @@ -82,6 +83,24 @@ func RelayAudioHelper(c *gin.Context, relayMode int) *relaymodel.ErrorWithStatus | ||||
| 			return openai.ErrorWrapper(err, "pre_consume_token_quota_failed", http.StatusForbidden) | ||||
| 		} | ||||
| 	} | ||||
| 	succeed := false | ||||
| 	defer func() { | ||||
| 		if succeed { | ||||
| 			return | ||||
| 		} | ||||
| 		if preConsumedQuota > 0 { | ||||
| 			// we need to roll back the pre-consumed quota | ||||
| 			defer func(ctx context.Context) { | ||||
| 				go func() { | ||||
| 					// negative means add quota back for token & user | ||||
| 					err := model.PostConsumeTokenQuota(tokenId, -preConsumedQuota) | ||||
| 					if err != nil { | ||||
| 						logger.Error(ctx, fmt.Sprintf("error rollback pre-consumed quota: %s", err.Error())) | ||||
| 					} | ||||
| 				}() | ||||
| 			}(c.Request.Context()) | ||||
| 		} | ||||
| 	}() | ||||
|  | ||||
| 	// map model name | ||||
| 	modelMapping := c.GetString("model_mapping") | ||||
| @@ -103,10 +122,15 @@ func RelayAudioHelper(c *gin.Context, relayMode int) *relaymodel.ErrorWithStatus | ||||
| 	} | ||||
|  | ||||
| 	fullRequestURL := util.GetFullRequestURL(baseURL, requestURL, channelType) | ||||
| 	if relayMode == constant.RelayModeAudioTranscription && channelType == common.ChannelTypeAzure { | ||||
| 		// https://learn.microsoft.com/en-us/azure/ai-services/openai/whisper-quickstart?tabs=command-line#rest-api | ||||
| 	if channelType == common.ChannelTypeAzure { | ||||
| 		apiVersion := util.GetAzureAPIVersion(c) | ||||
| 		fullRequestURL = fmt.Sprintf("%s/openai/deployments/%s/audio/transcriptions?api-version=%s", baseURL, audioModel, apiVersion) | ||||
| 		if relayMode == constant.RelayModeAudioTranscription { | ||||
| 			// https://learn.microsoft.com/en-us/azure/ai-services/openai/whisper-quickstart?tabs=command-line#rest-api | ||||
| 			fullRequestURL = fmt.Sprintf("%s/openai/deployments/%s/audio/transcriptions?api-version=%s", baseURL, audioModel, apiVersion) | ||||
| 		} else if relayMode == constant.RelayModeAudioSpeech { | ||||
| 			// https://learn.microsoft.com/en-us/azure/ai-services/openai/text-to-speech-quickstart?tabs=command-line#rest-api | ||||
| 			fullRequestURL = fmt.Sprintf("%s/openai/deployments/%s/audio/speech?api-version=%s", baseURL, audioModel, apiVersion) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	requestBody := &bytes.Buffer{} | ||||
| @@ -122,7 +146,7 @@ func RelayAudioHelper(c *gin.Context, relayMode int) *relaymodel.ErrorWithStatus | ||||
| 		return openai.ErrorWrapper(err, "new_request_failed", http.StatusInternalServerError) | ||||
| 	} | ||||
|  | ||||
| 	if relayMode == constant.RelayModeAudioTranscription && channelType == common.ChannelTypeAzure { | ||||
| 	if (relayMode == constant.RelayModeAudioTranscription || relayMode == constant.RelayModeAudioSpeech) && channelType == common.ChannelTypeAzure { | ||||
| 		// https://learn.microsoft.com/en-us/azure/ai-services/openai/whisper-quickstart?tabs=command-line#rest-api | ||||
| 		apiKey := c.Request.Header.Get("Authorization") | ||||
| 		apiKey = strings.TrimPrefix(apiKey, "Bearer ") | ||||
| @@ -183,24 +207,13 @@ func RelayAudioHelper(c *gin.Context, relayMode int) *relaymodel.ErrorWithStatus | ||||
| 		if err != nil { | ||||
| 			return openai.ErrorWrapper(err, "get_text_from_body_err", http.StatusInternalServerError) | ||||
| 		} | ||||
| 		quota = openai.CountTokenText(text, audioModel) | ||||
| 		quota = int64(openai.CountTokenText(text, audioModel)) | ||||
| 		resp.Body = io.NopCloser(bytes.NewBuffer(responseBody)) | ||||
| 	} | ||||
| 	if resp.StatusCode != http.StatusOK { | ||||
| 		if preConsumedQuota > 0 { | ||||
| 			// we need to roll back the pre-consumed quota | ||||
| 			defer func(ctx context.Context) { | ||||
| 				go func() { | ||||
| 					// negative means add quota back for token & user | ||||
| 					err := model.PostConsumeTokenQuota(tokenId, -preConsumedQuota) | ||||
| 					if err != nil { | ||||
| 						logger.Error(ctx, fmt.Sprintf("error rollback pre-consumed quota: %s", err.Error())) | ||||
| 					} | ||||
| 				}() | ||||
| 			}(c.Request.Context()) | ||||
| 		} | ||||
| 		return util.RelayErrorHandler(resp) | ||||
| 	} | ||||
| 	succeed = true | ||||
| 	quotaDelta := quota - preConsumedQuota | ||||
| 	defer func(ctx context.Context) { | ||||
| 		go util.PostConsumeQuota(ctx, tokenId, quotaDelta, quota, userId, channelId, modelRatio, groupRatio, audioModel, tokenName) | ||||
|   | ||||
| @@ -107,18 +107,18 @@ func getPromptTokens(textRequest *relaymodel.GeneralOpenAIRequest, relayMode int | ||||
| 	return 0 | ||||
| } | ||||
|  | ||||
| func getPreConsumedQuota(textRequest *relaymodel.GeneralOpenAIRequest, promptTokens int, ratio float64) int { | ||||
| func getPreConsumedQuota(textRequest *relaymodel.GeneralOpenAIRequest, promptTokens int, ratio float64) int64 { | ||||
| 	preConsumedTokens := config.PreConsumedQuota | ||||
| 	if textRequest.MaxTokens != 0 { | ||||
| 		preConsumedTokens = promptTokens + textRequest.MaxTokens | ||||
| 		preConsumedTokens = int64(promptTokens) + int64(textRequest.MaxTokens) | ||||
| 	} | ||||
| 	return int(float64(preConsumedTokens) * ratio) | ||||
| 	return int64(float64(preConsumedTokens) * ratio) | ||||
| } | ||||
|  | ||||
| func preConsumeQuota(ctx context.Context, textRequest *relaymodel.GeneralOpenAIRequest, promptTokens int, ratio float64, meta *util.RelayMeta) (int, *relaymodel.ErrorWithStatusCode) { | ||||
| func preConsumeQuota(ctx context.Context, textRequest *relaymodel.GeneralOpenAIRequest, promptTokens int, ratio float64, meta *util.RelayMeta) (int64, *relaymodel.ErrorWithStatusCode) { | ||||
| 	preConsumedQuota := getPreConsumedQuota(textRequest, promptTokens, ratio) | ||||
|  | ||||
| 	userQuota, err := model.CacheGetUserQuota(meta.UserId) | ||||
| 	userQuota, err := model.CacheGetUserQuota(ctx, meta.UserId) | ||||
| 	if err != nil { | ||||
| 		return preConsumedQuota, openai.ErrorWrapper(err, "get_user_quota_failed", http.StatusInternalServerError) | ||||
| 	} | ||||
| @@ -144,16 +144,16 @@ func preConsumeQuota(ctx context.Context, textRequest *relaymodel.GeneralOpenAIR | ||||
| 	return preConsumedQuota, nil | ||||
| } | ||||
|  | ||||
| func postConsumeQuota(ctx context.Context, usage *relaymodel.Usage, meta *util.RelayMeta, textRequest *relaymodel.GeneralOpenAIRequest, ratio float64, preConsumedQuota int, modelRatio float64, groupRatio float64) { | ||||
| func postConsumeQuota(ctx context.Context, usage *relaymodel.Usage, meta *util.RelayMeta, textRequest *relaymodel.GeneralOpenAIRequest, ratio float64, preConsumedQuota int64, modelRatio float64, groupRatio float64) { | ||||
| 	if usage == nil { | ||||
| 		logger.Error(ctx, "usage is nil, which is unexpected") | ||||
| 		return | ||||
| 	} | ||||
| 	quota := 0 | ||||
| 	var quota int64 | ||||
| 	completionRatio := common.GetCompletionRatio(textRequest.Model) | ||||
| 	promptTokens := usage.PromptTokens | ||||
| 	completionTokens := usage.CompletionTokens | ||||
| 	quota = int(math.Ceil((float64(promptTokens) + float64(completionTokens)*completionRatio) * ratio)) | ||||
| 	quota = int64(math.Ceil((float64(promptTokens) + float64(completionTokens)*completionRatio) * ratio)) | ||||
| 	if ratio != 0 && quota <= 0 { | ||||
| 		quota = 1 | ||||
| 	} | ||||
| @@ -168,7 +168,7 @@ func postConsumeQuota(ctx context.Context, usage *relaymodel.Usage, meta *util.R | ||||
| 	if err != nil { | ||||
| 		logger.Error(ctx, "error consuming token remain quota: "+err.Error()) | ||||
| 	} | ||||
| 	err = model.CacheUpdateUserQuota(meta.UserId) | ||||
| 	err = model.CacheUpdateUserQuota(ctx, meta.UserId) | ||||
| 	if err != nil { | ||||
| 		logger.Error(ctx, "error update user quota cache: "+err.Error()) | ||||
| 	} | ||||
|   | ||||
| @@ -61,7 +61,7 @@ func RelayImageHelper(c *gin.Context, relayMode int) *relaymodel.ErrorWithStatus | ||||
| 	if meta.ChannelType == common.ChannelTypeAzure { | ||||
| 		// https://learn.microsoft.com/en-us/azure/ai-services/openai/dall-e-quickstart?tabs=dalle3%2Ccommand-line&pivots=rest-api | ||||
| 		apiVersion := util.GetAzureAPIVersion(c) | ||||
| 		// https://{resource_name}.openai.azure.com/openai/deployments/dall-e-3/images/generations?api-version=2023-06-01-preview | ||||
| 		// https://{resource_name}.openai.azure.com/openai/deployments/dall-e-3/images/generations?api-version=2024-03-01-preview | ||||
| 		fullRequestURL = fmt.Sprintf("%s/openai/deployments/%s/images/generations?api-version=%s", meta.BaseURL, imageRequest.Model, apiVersion) | ||||
| 	} | ||||
|  | ||||
| @@ -79,9 +79,9 @@ func RelayImageHelper(c *gin.Context, relayMode int) *relaymodel.ErrorWithStatus | ||||
| 	modelRatio := common.GetModelRatio(imageRequest.Model) | ||||
| 	groupRatio := common.GetGroupRatio(meta.Group) | ||||
| 	ratio := modelRatio * groupRatio | ||||
| 	userQuota, err := model.CacheGetUserQuota(meta.UserId) | ||||
| 	userQuota, err := model.CacheGetUserQuota(ctx, meta.UserId) | ||||
|  | ||||
| 	quota := int(ratio*imageCostRatio*1000) * imageRequest.N | ||||
| 	quota := int64(ratio*imageCostRatio*1000) * int64(imageRequest.N) | ||||
|  | ||||
| 	if userQuota-quota < 0 { | ||||
| 		return openai.ErrorWrapper(errors.New("user quota is not enough"), "insufficient_user_quota", http.StatusForbidden) | ||||
| @@ -125,7 +125,7 @@ func RelayImageHelper(c *gin.Context, relayMode int) *relaymodel.ErrorWithStatus | ||||
| 		if err != nil { | ||||
| 			logger.SysError("error consuming token remain quota: " + err.Error()) | ||||
| 		} | ||||
| 		err = model.CacheUpdateUserQuota(meta.UserId) | ||||
| 		err = model.CacheUpdateUserQuota(ctx, meta.UserId) | ||||
| 		if err != nil { | ||||
| 			logger.SysError("error update user quota cache: " + err.Error()) | ||||
| 		} | ||||
|   | ||||
| @@ -74,6 +74,7 @@ func RelayTextHelper(c *gin.Context) *model.ErrorWithStatusCode { | ||||
| 		if err != nil { | ||||
| 			return openai.ErrorWrapper(err, "json_marshal_failed", http.StatusInternalServerError) | ||||
| 		} | ||||
| 		logger.Debugf(ctx, "converted request: \n%s", string(jsonData)) | ||||
| 		requestBody = bytes.NewBuffer(jsonData) | ||||
| 	} | ||||
|  | ||||
|   | ||||
| @@ -7,6 +7,7 @@ import ( | ||||
| 	"github.com/songquanpeng/one-api/relay/channel/anthropic" | ||||
| 	"github.com/songquanpeng/one-api/relay/channel/baidu" | ||||
| 	"github.com/songquanpeng/one-api/relay/channel/gemini" | ||||
| 	"github.com/songquanpeng/one-api/relay/channel/ollama" | ||||
| 	"github.com/songquanpeng/one-api/relay/channel/openai" | ||||
| 	"github.com/songquanpeng/one-api/relay/channel/palm" | ||||
| 	"github.com/songquanpeng/one-api/relay/channel/tencent" | ||||
| @@ -37,6 +38,8 @@ func GetAdaptor(apiType int) channel.Adaptor { | ||||
| 		return &xunfei.Adaptor{} | ||||
| 	case constant.APITypeZhipu: | ||||
| 		return &zhipu.Adaptor{} | ||||
| 	case constant.APITypeOllama: | ||||
| 		return &ollama.Adaptor{} | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|   | ||||
| @@ -5,25 +5,29 @@ type ResponseFormat struct { | ||||
| } | ||||
|  | ||||
| type GeneralOpenAIRequest struct { | ||||
| 	Model            string          `json:"model,omitempty"` | ||||
| 	Messages         []Message       `json:"messages,omitempty"` | ||||
| 	Prompt           any             `json:"prompt,omitempty"` | ||||
| 	Stream           bool            `json:"stream,omitempty"` | ||||
| 	MaxTokens        int             `json:"max_tokens,omitempty"` | ||||
| 	Temperature      float64         `json:"temperature,omitempty"` | ||||
| 	TopP             float64         `json:"top_p,omitempty"` | ||||
| 	N                int             `json:"n,omitempty"` | ||||
| 	Input            any             `json:"input,omitempty"` | ||||
| 	Instruction      string          `json:"instruction,omitempty"` | ||||
| 	Size             string          `json:"size,omitempty"` | ||||
| 	Functions        any             `json:"functions,omitempty"` | ||||
| 	Model            string          `json:"model,omitempty"` | ||||
| 	FrequencyPenalty float64         `json:"frequency_penalty,omitempty"` | ||||
| 	MaxTokens        int             `json:"max_tokens,omitempty"` | ||||
| 	N                int             `json:"n,omitempty"` | ||||
| 	PresencePenalty  float64         `json:"presence_penalty,omitempty"` | ||||
| 	ResponseFormat   *ResponseFormat `json:"response_format,omitempty"` | ||||
| 	Seed             float64         `json:"seed,omitempty"` | ||||
| 	Tools            any             `json:"tools,omitempty"` | ||||
| 	Stream           bool            `json:"stream,omitempty"` | ||||
| 	Temperature      float64         `json:"temperature,omitempty"` | ||||
| 	TopP             float64         `json:"top_p,omitempty"` | ||||
| 	TopK             int             `json:"top_k,omitempty"` | ||||
| 	Tools            []Tool          `json:"tools,omitempty"` | ||||
| 	ToolChoice       any             `json:"tool_choice,omitempty"` | ||||
| 	FunctionCall     any             `json:"function_call,omitempty"` | ||||
| 	Functions        any             `json:"functions,omitempty"` | ||||
| 	User             string          `json:"user,omitempty"` | ||||
| 	Prompt           any             `json:"prompt,omitempty"` | ||||
| 	Input            any             `json:"input,omitempty"` | ||||
| 	EncodingFormat   string          `json:"encoding_format,omitempty"` | ||||
| 	Dimensions       int             `json:"dimensions,omitempty"` | ||||
| 	Instruction      string          `json:"instruction,omitempty"` | ||||
| 	Size             string          `json:"size,omitempty"` | ||||
| } | ||||
|  | ||||
| func (r GeneralOpenAIRequest) ParseInput() []string { | ||||
|   | ||||
| @@ -1,9 +1,10 @@ | ||||
| package model | ||||
|  | ||||
| type Message struct { | ||||
| 	Role    string  `json:"role"` | ||||
| 	Content any     `json:"content"` | ||||
| 	Name    *string `json:"name,omitempty"` | ||||
| 	Role      string  `json:"role,omitempty"` | ||||
| 	Content   any     `json:"content,omitempty"` | ||||
| 	Name      *string `json:"name,omitempty"` | ||||
| 	ToolCalls []Tool  `json:"tool_calls,omitempty"` | ||||
| } | ||||
|  | ||||
| func (m Message) IsStringContent() bool { | ||||
|   | ||||
							
								
								
									
										14
									
								
								relay/model/tool.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								relay/model/tool.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | ||||
| package model | ||||
|  | ||||
| type Tool struct { | ||||
| 	Id       string   `json:"id,omitempty"` | ||||
| 	Type     string   `json:"type"` | ||||
| 	Function Function `json:"function"` | ||||
| } | ||||
|  | ||||
| type Function struct { | ||||
| 	Description string `json:"description,omitempty"` | ||||
| 	Name        string `json:"name"` | ||||
| 	Parameters  any    `json:"parameters,omitempty"` // request | ||||
| 	Arguments   any    `json:"arguments,omitempty"`  // response | ||||
| } | ||||
| @@ -6,7 +6,7 @@ import ( | ||||
| 	"github.com/songquanpeng/one-api/model" | ||||
| ) | ||||
|  | ||||
| func ReturnPreConsumedQuota(ctx context.Context, preConsumedQuota int, tokenId int) { | ||||
| func ReturnPreConsumedQuota(ctx context.Context, preConsumedQuota int64, tokenId int) { | ||||
| 	if preConsumedQuota != 0 { | ||||
| 		go func(ctx context.Context) { | ||||
| 			// return pre-consumed quota | ||||
|   | ||||
| @@ -27,7 +27,23 @@ func ShouldDisableChannel(err *relaymodel.Error, statusCode int) bool { | ||||
| 	if statusCode == http.StatusUnauthorized { | ||||
| 		return true | ||||
| 	} | ||||
| 	if err.Type == "insufficient_quota" || err.Code == "invalid_api_key" || err.Code == "account_deactivated" { | ||||
| 	switch err.Type { | ||||
| 	case "insufficient_quota": | ||||
| 		return true | ||||
| 	// https://docs.anthropic.com/claude/reference/errors | ||||
| 	case "authentication_error": | ||||
| 		return true | ||||
| 	case "permission_error": | ||||
| 		return true | ||||
| 	case "forbidden": | ||||
| 		return true | ||||
| 	} | ||||
| 	if err.Code == "invalid_api_key" || err.Code == "account_deactivated" { | ||||
| 		return true | ||||
| 	} | ||||
| 	if strings.HasPrefix(err.Message, "Your credit balance is too low") { // anthropic | ||||
| 		return true | ||||
| 	} else if strings.HasPrefix(err.Message, "This organization has been disabled.") { | ||||
| 		return true | ||||
| 	} | ||||
| 	return false | ||||
| @@ -101,6 +117,9 @@ func RelayErrorHandler(resp *http.Response) (ErrorWithStatusCode *relaymodel.Err | ||||
| 	if err != nil { | ||||
| 		return | ||||
| 	} | ||||
| 	if config.DebugEnabled { | ||||
| 		logger.SysLog(fmt.Sprintf("error happened, status code: %d, response: \n%s", resp.StatusCode, string(responseBody))) | ||||
| 	} | ||||
| 	err = resp.Body.Close() | ||||
| 	if err != nil { | ||||
| 		return | ||||
| @@ -136,20 +155,20 @@ func GetFullRequestURL(baseURL string, requestURL string, channelType int) strin | ||||
| 	return fullRequestURL | ||||
| } | ||||
|  | ||||
| func PostConsumeQuota(ctx context.Context, tokenId int, quotaDelta int, totalQuota int, userId int, channelId int, modelRatio float64, groupRatio float64, modelName string, tokenName string) { | ||||
| func PostConsumeQuota(ctx context.Context, tokenId int, quotaDelta int64, totalQuota int64, userId int, channelId int, modelRatio float64, groupRatio float64, modelName string, tokenName string) { | ||||
| 	// quotaDelta is remaining quota to be consumed | ||||
| 	err := model.PostConsumeTokenQuota(tokenId, quotaDelta) | ||||
| 	if err != nil { | ||||
| 		logger.SysError("error consuming token remain quota: " + err.Error()) | ||||
| 	} | ||||
| 	err = model.CacheUpdateUserQuota(userId) | ||||
| 	err = model.CacheUpdateUserQuota(ctx, userId) | ||||
| 	if err != nil { | ||||
| 		logger.SysError("error update user quota cache: " + err.Error()) | ||||
| 	} | ||||
| 	// totalQuota is total quota consumed | ||||
| 	if totalQuota != 0 { | ||||
| 		logContent := fmt.Sprintf("模型倍率 %.2f,分组倍率 %.2f", modelRatio, groupRatio) | ||||
| 		model.RecordConsumeLog(ctx, userId, channelId, totalQuota, 0, modelName, tokenName, totalQuota, logContent) | ||||
| 		model.RecordConsumeLog(ctx, userId, channelId, int(totalQuota), 0, modelName, tokenName, totalQuota, logContent) | ||||
| 		model.UpdateUserUsedQuotaAndRequestCount(userId, totalQuota) | ||||
| 		model.UpdateChannelUsedQuota(channelId, totalQuota) | ||||
| 	} | ||||
|   | ||||
| @@ -43,6 +43,7 @@ func SetApiRouter(router *gin.Engine) { | ||||
| 				selfRoute.GET("/token", controller.GenerateAccessToken) | ||||
| 				selfRoute.GET("/aff", controller.GetAffCode) | ||||
| 				selfRoute.POST("/topup", controller.TopUp) | ||||
| 				selfRoute.GET("/available_models", controller.GetUserAvailableModels) | ||||
| 			} | ||||
|  | ||||
| 			adminRoute := userRoute.Group("/") | ||||
|   | ||||
| @@ -9,6 +9,7 @@ import ( | ||||
|  | ||||
| func SetDashboardRouter(router *gin.Engine) { | ||||
| 	apiRouter := router.Group("/") | ||||
| 	apiRouter.Use(middleware.CORS()) | ||||
| 	apiRouter.Use(gzip.Gzip(gzip.DefaultCompression)) | ||||
| 	apiRouter.Use(middleware.GlobalAPIRateLimit()) | ||||
| 	apiRouter.Use(middleware.TokenAuth()) | ||||
|   | ||||
| @@ -9,7 +9,7 @@ | ||||
| 1. 在 `web` 文件夹下新建一个文件夹,文件夹名为主题名。 | ||||
| 2. 把你的主题文件放到这个文件夹下。 | ||||
| 3. 修改你的 `package.json` 文件,把 `build` 命令改为:`"build": "react-scripts build && mv -f build ../build/default"`,其中 `default` 为你的主题名。 | ||||
| 4. 修改 `common/constants.go` 中的 `ValidThemes`,把你的主题名称注册进去。 | ||||
| 4. 修改 `common/config/config.go` 中的 `ValidThemes`,把你的主题名称注册进去。 | ||||
| 5. 修改 `web/THEMES` 文件,这里也需要同步修改。 | ||||
|  | ||||
| ## 主题列表 | ||||
| @@ -33,6 +33,12 @@ | ||||
| ||| | ||||
| ||| | ||||
|  | ||||
| ### 主题:air | ||||
| 由 [Calon](https://github.com/Calcium-Ion) 开发。 | ||||
| ||| | ||||
| |:---:|:---:| | ||||
|  | ||||
|  | ||||
| #### 开发说明 | ||||
|  | ||||
| 请查看 [web/berry/README.md](https://github.com/songquanpeng/one-api/tree/main/web/berry/README.md) | ||||
|   | ||||
| @@ -1,2 +1,3 @@ | ||||
| default | ||||
| berry | ||||
| air | ||||
|   | ||||
							
								
								
									
										26
									
								
								web/air/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								web/air/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,26 @@ | ||||
| # See https://help.github.com/articles/ignoring-files/ for more about ignoring files. | ||||
|  | ||||
| # dependencies | ||||
| /node_modules | ||||
| /.pnp | ||||
| .pnp.js | ||||
|  | ||||
| # testing | ||||
| /coverage | ||||
|  | ||||
| # production | ||||
| /build | ||||
|  | ||||
| # misc | ||||
| .DS_Store | ||||
| .env.local | ||||
| .env.development.local | ||||
| .env.test.local | ||||
| .env.production.local | ||||
|  | ||||
| npm-debug.log* | ||||
| yarn-debug.log* | ||||
| yarn-error.log* | ||||
| .idea | ||||
| package-lock.json | ||||
| yarn.lock | ||||
							
								
								
									
										21
									
								
								web/air/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								web/air/README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,21 @@ | ||||
| # React Template | ||||
|  | ||||
| ## Basic Usages | ||||
|  | ||||
| ```shell | ||||
| # Runs the app in the development mode | ||||
| npm start | ||||
|  | ||||
| # Builds the app for production to the `build` folder | ||||
| npm run build | ||||
| ``` | ||||
|  | ||||
| If you want to change the default server, please set `REACT_APP_SERVER` environment variables before build, | ||||
| for example: `REACT_APP_SERVER=http://your.domain.com`. | ||||
|  | ||||
| Before you start editing, make sure your `Actions on Save` options have `Optimize imports` & `Run Prettier` enabled. | ||||
|  | ||||
| ## Reference | ||||
|  | ||||
| 1. https://github.com/OIerDb-ng/OIerDb | ||||
| 2. https://github.com/cornflourblue/react-hooks-redux-registration-login-example | ||||
							
								
								
									
										60
									
								
								web/air/package.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										60
									
								
								web/air/package.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,60 @@ | ||||
| { | ||||
|   "name": "react-template", | ||||
|   "version": "0.1.0", | ||||
|   "private": true, | ||||
|   "dependencies": { | ||||
|     "@douyinfe/semi-icons": "^2.46.1", | ||||
|     "@douyinfe/semi-ui": "^2.46.1", | ||||
|     "@visactor/react-vchart": "~1.8.8", | ||||
|     "@visactor/vchart": "~1.8.8", | ||||
|     "@visactor/vchart-semi-theme": "~1.8.8", | ||||
|     "axios": "^0.27.2", | ||||
|     "history": "^5.3.0", | ||||
|     "marked": "^4.1.1", | ||||
|     "react": "^18.2.0", | ||||
|     "react-dom": "^18.2.0", | ||||
|     "react-dropzone": "^14.2.3", | ||||
|     "react-fireworks": "^1.0.4", | ||||
|     "react-router-dom": "^6.3.0", | ||||
|     "react-scripts": "5.0.1", | ||||
|     "react-telegram-login": "^1.1.2", | ||||
|     "react-toastify": "^9.0.8", | ||||
|     "react-turnstile": "^1.0.5", | ||||
|     "semantic-ui-css": "^2.5.0", | ||||
|     "semantic-ui-react": "^2.1.3", | ||||
|     "usehooks-ts": "^2.9.1" | ||||
|   }, | ||||
|   "scripts": { | ||||
|     "start": "react-scripts start", | ||||
|     "build": "react-scripts build && mv -f build ../build/air", | ||||
|     "test": "react-scripts test", | ||||
|     "eject": "react-scripts eject" | ||||
|   }, | ||||
|   "eslintConfig": { | ||||
|     "extends": [ | ||||
|       "react-app", | ||||
|       "react-app/jest" | ||||
|     ] | ||||
|   }, | ||||
|   "browserslist": { | ||||
|     "production": [ | ||||
|       ">0.2%", | ||||
|       "not dead", | ||||
|       "not op_mini all" | ||||
|     ], | ||||
|     "development": [ | ||||
|       "last 1 chrome version", | ||||
|       "last 1 firefox version", | ||||
|       "last 1 safari version" | ||||
|     ] | ||||
|   }, | ||||
|   "devDependencies": { | ||||
|     "prettier": "2.8.8", | ||||
|     "typescript": "4.4.2" | ||||
|   }, | ||||
|   "prettier": { | ||||
|     "singleQuote": true, | ||||
|     "jsxSingleQuote": true | ||||
|   }, | ||||
|   "proxy": "http://localhost:3000" | ||||
| } | ||||
							
								
								
									
										
											BIN
										
									
								
								web/air/public/favicon.ico
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								web/air/public/favicon.ico
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 4.2 KiB | 
							
								
								
									
										18
									
								
								web/air/public/index.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								web/air/public/index.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| <!DOCTYPE html> | ||||
| <html lang="en"> | ||||
| <head> | ||||
|   <meta charset="utf-8" /> | ||||
|   <link rel="icon" href="logo.png" /> | ||||
|   <meta name="viewport" content="width=device-width, initial-scale=1" /> | ||||
|   <meta name="theme-color" content="#ffffff" /> | ||||
|   <meta | ||||
|           name="description" | ||||
|           content="OpenAI 接口聚合管理,支持多种渠道包括 Azure,可用于二次分发管理 key,仅单可执行文件,已打包好 Docker 镜像,一键部署,开箱即用" | ||||
|   /> | ||||
|   <title>One API</title> | ||||
| </head> | ||||
| <body> | ||||
| <noscript>You need to enable JavaScript to run this app.</noscript> | ||||
| <div id="root"></div> | ||||
| </body> | ||||
| </html> | ||||
							
								
								
									
										
											BIN
										
									
								
								web/air/public/logo.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								web/air/public/logo.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 7.9 KiB | 
							
								
								
									
										3
									
								
								web/air/public/robots.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								web/air/public/robots.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| # https://www.robotstxt.org/robotstxt.html | ||||
| User-agent: * | ||||
| Disallow: | ||||
							
								
								
									
										242
									
								
								web/air/src/App.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										242
									
								
								web/air/src/App.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,242 @@ | ||||
| import React, { lazy, Suspense, useContext, useEffect } from 'react'; | ||||
| import { Route, Routes } from 'react-router-dom'; | ||||
| import Loading from './components/Loading'; | ||||
| import User from './pages/User'; | ||||
| import { PrivateRoute } from './components/PrivateRoute'; | ||||
| import RegisterForm from './components/RegisterForm'; | ||||
| import LoginForm from './components/LoginForm'; | ||||
| import NotFound from './pages/NotFound'; | ||||
| import Setting from './pages/Setting'; | ||||
| import EditUser from './pages/User/EditUser'; | ||||
| import { getLogo, getSystemName } from './helpers'; | ||||
| import PasswordResetForm from './components/PasswordResetForm'; | ||||
| import GitHubOAuth from './components/GitHubOAuth'; | ||||
| import PasswordResetConfirm from './components/PasswordResetConfirm'; | ||||
| import { UserContext } from './context/User'; | ||||
| import Channel from './pages/Channel'; | ||||
| import Token from './pages/Token'; | ||||
| import EditChannel from './pages/Channel/EditChannel'; | ||||
| import Redemption from './pages/Redemption'; | ||||
| import TopUp from './pages/TopUp'; | ||||
| import Log from './pages/Log'; | ||||
| import Chat from './pages/Chat'; | ||||
| import { Layout } from '@douyinfe/semi-ui'; | ||||
| import Midjourney from './pages/Midjourney'; | ||||
| import Detail from './pages/Detail'; | ||||
|  | ||||
| const Home = lazy(() => import('./pages/Home')); | ||||
| const About = lazy(() => import('./pages/About')); | ||||
|  | ||||
| function App() { | ||||
|   const [userState, userDispatch] = useContext(UserContext); | ||||
|   // const [statusState, statusDispatch] = useContext(StatusContext); | ||||
|  | ||||
|   const loadUser = () => { | ||||
|     let user = localStorage.getItem('user'); | ||||
|     if (user) { | ||||
|       let data = JSON.parse(user); | ||||
|       userDispatch({ type: 'login', payload: data }); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   useEffect(() => { | ||||
|     loadUser(); | ||||
|     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 ( | ||||
|     <Layout> | ||||
|       <Layout.Content> | ||||
|         <Routes> | ||||
|           <Route | ||||
|             path="/" | ||||
|             element={ | ||||
|               <Suspense fallback={<Loading></Loading>}> | ||||
|                 <Home /> | ||||
|               </Suspense> | ||||
|             } | ||||
|           /> | ||||
|           <Route | ||||
|             path="/channel" | ||||
|             element={ | ||||
|               <PrivateRoute> | ||||
|                 <Channel /> | ||||
|               </PrivateRoute> | ||||
|             } | ||||
|           /> | ||||
|           <Route | ||||
|             path="/channel/edit/:id" | ||||
|             element={ | ||||
|               <Suspense fallback={<Loading></Loading>}> | ||||
|                 <EditChannel /> | ||||
|               </Suspense> | ||||
|             } | ||||
|           /> | ||||
|           <Route | ||||
|             path="/channel/add" | ||||
|             element={ | ||||
|               <Suspense fallback={<Loading></Loading>}> | ||||
|                 <EditChannel /> | ||||
|               </Suspense> | ||||
|             } | ||||
|           /> | ||||
|           <Route | ||||
|             path="/token" | ||||
|             element={ | ||||
|               <PrivateRoute> | ||||
|                 <Token /> | ||||
|               </PrivateRoute> | ||||
|             } | ||||
|           /> | ||||
|           <Route | ||||
|             path="/redemption" | ||||
|             element={ | ||||
|               <PrivateRoute> | ||||
|                 <Redemption /> | ||||
|               </PrivateRoute> | ||||
|             } | ||||
|           /> | ||||
|           <Route | ||||
|             path="/user" | ||||
|             element={ | ||||
|               <PrivateRoute> | ||||
|                 <User /> | ||||
|               </PrivateRoute> | ||||
|             } | ||||
|           /> | ||||
|           <Route | ||||
|             path="/user/edit/:id" | ||||
|             element={ | ||||
|               <Suspense fallback={<Loading></Loading>}> | ||||
|                 <EditUser /> | ||||
|               </Suspense> | ||||
|             } | ||||
|           /> | ||||
|           <Route | ||||
|             path="/user/edit" | ||||
|             element={ | ||||
|               <Suspense fallback={<Loading></Loading>}> | ||||
|                 <EditUser /> | ||||
|               </Suspense> | ||||
|             } | ||||
|           /> | ||||
|           <Route | ||||
|             path="/user/reset" | ||||
|             element={ | ||||
|               <Suspense fallback={<Loading></Loading>}> | ||||
|                 <PasswordResetConfirm /> | ||||
|               </Suspense> | ||||
|             } | ||||
|           /> | ||||
|           <Route | ||||
|             path="/login" | ||||
|             element={ | ||||
|               <Suspense fallback={<Loading></Loading>}> | ||||
|                 <LoginForm /> | ||||
|               </Suspense> | ||||
|             } | ||||
|           /> | ||||
|           <Route | ||||
|             path="/register" | ||||
|             element={ | ||||
|               <Suspense fallback={<Loading></Loading>}> | ||||
|                 <RegisterForm /> | ||||
|               </Suspense> | ||||
|             } | ||||
|           /> | ||||
|           <Route | ||||
|             path="/reset" | ||||
|             element={ | ||||
|               <Suspense fallback={<Loading></Loading>}> | ||||
|                 <PasswordResetForm /> | ||||
|               </Suspense> | ||||
|             } | ||||
|           /> | ||||
|           <Route | ||||
|             path="/oauth/github" | ||||
|             element={ | ||||
|               <Suspense fallback={<Loading></Loading>}> | ||||
|                 <GitHubOAuth /> | ||||
|               </Suspense> | ||||
|             } | ||||
|           /> | ||||
|           <Route | ||||
|             path="/setting" | ||||
|             element={ | ||||
|               <PrivateRoute> | ||||
|                 <Suspense fallback={<Loading></Loading>}> | ||||
|                   <Setting /> | ||||
|                 </Suspense> | ||||
|               </PrivateRoute> | ||||
|             } | ||||
|           /> | ||||
|           <Route | ||||
|             path="/topup" | ||||
|             element={ | ||||
|               <PrivateRoute> | ||||
|                 <Suspense fallback={<Loading></Loading>}> | ||||
|                   <TopUp /> | ||||
|                 </Suspense> | ||||
|               </PrivateRoute> | ||||
|             } | ||||
|           /> | ||||
|           <Route | ||||
|             path="/log" | ||||
|             element={ | ||||
|               <PrivateRoute> | ||||
|                 <Log /> | ||||
|               </PrivateRoute> | ||||
|             } | ||||
|           /> | ||||
|           <Route | ||||
|             path="/detail" | ||||
|             element={ | ||||
|               <PrivateRoute> | ||||
|                 <Detail /> | ||||
|               </PrivateRoute> | ||||
|             } | ||||
|           /> | ||||
|           <Route | ||||
|             path="/midjourney" | ||||
|             element={ | ||||
|               <PrivateRoute> | ||||
|                 <Midjourney /> | ||||
|               </PrivateRoute> | ||||
|             } | ||||
|           /> | ||||
|           <Route | ||||
|             path="/about" | ||||
|             element={ | ||||
|               <Suspense fallback={<Loading></Loading>}> | ||||
|                 <About /> | ||||
|               </Suspense> | ||||
|             } | ||||
|           /> | ||||
|           <Route | ||||
|             path="/chat" | ||||
|             element={ | ||||
|               <Suspense fallback={<Loading></Loading>}> | ||||
|                 <Chat /> | ||||
|               </Suspense> | ||||
|             } | ||||
|           /> | ||||
|           <Route path="*" element={ | ||||
|             <NotFound /> | ||||
|           } /> | ||||
|         </Routes> | ||||
|       </Layout.Content> | ||||
|     </Layout> | ||||
|   ); | ||||
| } | ||||
|  | ||||
| export default App; | ||||
							
								
								
									
										738
									
								
								web/air/src/components/ChannelsTable.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										738
									
								
								web/air/src/components/ChannelsTable.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,738 @@ | ||||
| import React, { useEffect, useState } from 'react'; | ||||
| import { API, isMobile, shouldShowPrompt, showError, showInfo, showSuccess, timestamp2string } from '../helpers'; | ||||
|  | ||||
| import { CHANNEL_OPTIONS, ITEMS_PER_PAGE } from '../constants'; | ||||
| import { renderGroup, renderNumberWithPoint, renderQuota } from '../helpers/render'; | ||||
| import { | ||||
|   Button, | ||||
|   Dropdown, | ||||
|   Form, | ||||
|   InputNumber, | ||||
|   Popconfirm, | ||||
|   Space, | ||||
|   SplitButtonGroup, | ||||
|   Switch, | ||||
|   Table, | ||||
|   Tag, | ||||
|   Tooltip, | ||||
|   Typography | ||||
| } from '@douyinfe/semi-ui'; | ||||
| import EditChannel from '../pages/Channel/EditChannel'; | ||||
| import { IconTreeTriangleDown } from '@douyinfe/semi-icons'; | ||||
|  | ||||
| function renderTimestamp(timestamp) { | ||||
|   return ( | ||||
|     <> | ||||
|       {timestamp2string(timestamp)} | ||||
|     </> | ||||
|   ); | ||||
| } | ||||
|  | ||||
| let type2label = undefined; | ||||
|  | ||||
| function renderType(type) { | ||||
|   if (!type2label) { | ||||
|     type2label = new Map(); | ||||
|     for (let i = 0; i < CHANNEL_OPTIONS.length; i++) { | ||||
|       type2label[CHANNEL_OPTIONS[i].value] = CHANNEL_OPTIONS[i]; | ||||
|     } | ||||
|     type2label[0] = { value: 0, text: '未知类型', color: 'grey' }; | ||||
|   } | ||||
|   return <Tag size="large" color={type2label[type]?.color}>{type2label[type]?.text}</Tag>; | ||||
| } | ||||
|  | ||||
| const ChannelsTable = () => { | ||||
|   const columns = [ | ||||
|     // { | ||||
|     //     title: '', | ||||
|     //     dataIndex: 'checkbox', | ||||
|     //     className: 'checkbox', | ||||
|     // }, | ||||
|     { | ||||
|       title: 'ID', | ||||
|       dataIndex: 'id' | ||||
|     }, | ||||
|     { | ||||
|       title: '名称', | ||||
|       dataIndex: 'name' | ||||
|     }, | ||||
|     // { | ||||
|     //   title: '分组', | ||||
|     //   dataIndex: 'group', | ||||
|     //   render: (text, record, index) => { | ||||
|     //     return ( | ||||
|     //       <div> | ||||
|     //         <Space spacing={2}> | ||||
|     //           { | ||||
|     //             text.split(',').map((item, index) => { | ||||
|     //               return (renderGroup(item)); | ||||
|     //             }) | ||||
|     //           } | ||||
|     //         </Space> | ||||
|     //       </div> | ||||
|     //     ); | ||||
|     //   } | ||||
|     // }, | ||||
|     { | ||||
|       title: '类型', | ||||
|       dataIndex: 'type', | ||||
|       render: (text, record, index) => { | ||||
|         return ( | ||||
|           <div> | ||||
|             {renderType(text)} | ||||
|           </div> | ||||
|         ); | ||||
|       } | ||||
|     }, | ||||
|     { | ||||
|       title: '状态', | ||||
|       dataIndex: 'status', | ||||
|       render: (text, record, index) => { | ||||
|         return ( | ||||
|           <div> | ||||
|             {renderStatus(text)} | ||||
|           </div> | ||||
|         ); | ||||
|       } | ||||
|     }, | ||||
|     { | ||||
|       title: '响应时间', | ||||
|       dataIndex: 'response_time', | ||||
|       render: (text, record, index) => { | ||||
|         return ( | ||||
|           <div> | ||||
|             {renderResponseTime(text)} | ||||
|           </div> | ||||
|         ); | ||||
|       } | ||||
|     }, | ||||
|     { | ||||
|       title: '已用/剩余', | ||||
|       dataIndex: 'expired_time', | ||||
|       render: (text, record, index) => { | ||||
|         return ( | ||||
|           <div> | ||||
|             <Space spacing={1}> | ||||
|               <Tooltip content={'已用额度'}> | ||||
|                 <Tag color="white" type="ghost" size="large">{renderQuota(record.used_quota)}</Tag> | ||||
|               </Tooltip> | ||||
|               <Tooltip content={'剩余额度' + record.balance + ',点击更新'}> | ||||
|                 <Tag color="white" type="ghost" size="large" onClick={() => { | ||||
|                   updateChannelBalance(record); | ||||
|                 }}>${renderNumberWithPoint(record.balance)}</Tag> | ||||
|               </Tooltip> | ||||
|             </Space> | ||||
|           </div> | ||||
|         ); | ||||
|       } | ||||
|     }, | ||||
|     { | ||||
|       title: '优先级', | ||||
|       dataIndex: 'priority', | ||||
|       render: (text, record, index) => { | ||||
|         return ( | ||||
|           <div> | ||||
|             <InputNumber | ||||
|               style={{ width: 70 }} | ||||
|               name="priority" | ||||
|               onBlur={e => { | ||||
|                 manageChannel(record.id, 'priority', record, e.target.value); | ||||
|               }} | ||||
|               keepFocus={true} | ||||
|               innerButtons | ||||
|               defaultValue={record.priority} | ||||
|               min={-999} | ||||
|             /> | ||||
|           </div> | ||||
|         ); | ||||
|       } | ||||
|     }, | ||||
|     // { | ||||
|     //   title: '权重', | ||||
|     //   dataIndex: 'weight', | ||||
|     //   render: (text, record, index) => { | ||||
|     //     return ( | ||||
|     //       <div> | ||||
|     //         <InputNumber | ||||
|     //           style={{ width: 70 }} | ||||
|     //           name="weight" | ||||
|     //           onBlur={e => { | ||||
|     //             manageChannel(record.id, 'weight', record, e.target.value); | ||||
|     //           }} | ||||
|     //           keepFocus={true} | ||||
|     //           innerButtons | ||||
|     //           defaultValue={record.weight} | ||||
|     //           min={0} | ||||
|     //         /> | ||||
|     //       </div> | ||||
|     //     ); | ||||
|     //   } | ||||
|     // }, | ||||
|     { | ||||
|       title: '', | ||||
|       dataIndex: 'operate', | ||||
|       render: (text, record, index) => ( | ||||
|         <div> | ||||
|           {/* <SplitButtonGroup style={{ marginRight: 1 }} aria-label="测试操作项目组"> | ||||
|             <Button theme="light" onClick={() => { | ||||
|               testChannel(record, ''); | ||||
|             }}>测试</Button> | ||||
|             <Dropdown trigger="click" position="bottomRight" menu={record.test_models} | ||||
|             > | ||||
|               <Button style={{ padding: '8px 4px' }} type="primary" icon={<IconTreeTriangleDown />}></Button> | ||||
|             </Dropdown> | ||||
|           </SplitButtonGroup> */} | ||||
|           <Button theme='light' type='primary' style={{ marginRight: 1 }} onClick={() => testChannel(record)}>测试</Button> | ||||
|           <Popconfirm | ||||
|             title="确定是否要删除此渠道?" | ||||
|             content="此修改将不可逆" | ||||
|             okType={'danger'} | ||||
|             position={'left'} | ||||
|             onConfirm={() => { | ||||
|               manageChannel(record.id, 'delete', record).then( | ||||
|                 () => { | ||||
|                   removeRecord(record.id); | ||||
|                 } | ||||
|               ); | ||||
|             }} | ||||
|           > | ||||
|             <Button theme="light" type="danger" style={{ marginRight: 1 }}>删除</Button> | ||||
|           </Popconfirm> | ||||
|           { | ||||
|             record.status === 1 ? | ||||
|               <Button theme="light" type="warning" style={{ marginRight: 1 }} onClick={ | ||||
|                 async () => { | ||||
|                   manageChannel( | ||||
|                     record.id, | ||||
|                     'disable', | ||||
|                     record | ||||
|                   ); | ||||
|                 } | ||||
|               }>禁用</Button> : | ||||
|               <Button theme="light" type="secondary" style={{ marginRight: 1 }} onClick={ | ||||
|                 async () => { | ||||
|                   manageChannel( | ||||
|                     record.id, | ||||
|                     'enable', | ||||
|                     record | ||||
|                   ); | ||||
|                 } | ||||
|               }>启用</Button> | ||||
|           } | ||||
|           <Button theme="light" type="tertiary" style={{ marginRight: 1 }} onClick={ | ||||
|             () => { | ||||
|               setEditingChannel(record); | ||||
|               setShowEdit(true); | ||||
|             } | ||||
|           }>编辑</Button> | ||||
|         </div> | ||||
|       ) | ||||
|     } | ||||
|   ]; | ||||
|  | ||||
|   const [channels, setChannels] = useState([]); | ||||
|   const [loading, setLoading] = useState(true); | ||||
|   const [activePage, setActivePage] = useState(1); | ||||
|   const [idSort, setIdSort] = useState(false); | ||||
|   const [searchKeyword, setSearchKeyword] = useState(''); | ||||
|   const [searchGroup, setSearchGroup] = useState(''); | ||||
|   const [searchModel, setSearchModel] = useState(''); | ||||
|   const [searching, setSearching] = useState(false); | ||||
|   const [updatingBalance, setUpdatingBalance] = useState(false); | ||||
|   const [pageSize, setPageSize] = useState(ITEMS_PER_PAGE); | ||||
|   const [showPrompt, setShowPrompt] = useState(shouldShowPrompt('channel-test')); | ||||
|   const [channelCount, setChannelCount] = useState(pageSize); | ||||
|   const [groupOptions, setGroupOptions] = useState([]); | ||||
|   const [showEdit, setShowEdit] = useState(false); | ||||
|   const [enableBatchDelete, setEnableBatchDelete] = useState(false); | ||||
|   const [editingChannel, setEditingChannel] = useState({ | ||||
|     id: undefined | ||||
|   }); | ||||
|   const [selectedChannels, setSelectedChannels] = useState([]); | ||||
|  | ||||
|   const removeRecord = id => { | ||||
|     let newDataSource = [...channels]; | ||||
|     if (id != null) { | ||||
|       let idx = newDataSource.findIndex(data => data.id === id); | ||||
|  | ||||
|       if (idx > -1) { | ||||
|         newDataSource.splice(idx, 1); | ||||
|         setChannels(newDataSource); | ||||
|       } | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   const setChannelFormat = (channels) => { | ||||
|     for (let i = 0; i < channels.length; i++) { | ||||
|       channels[i].key = '' + channels[i].id; | ||||
|       let test_models = []; | ||||
|       channels[i].models.split(',').forEach((item, index) => { | ||||
|         test_models.push({ | ||||
|           node: 'item', | ||||
|           name: item, | ||||
|           onClick: () => { | ||||
|             testChannel(channels[i], item); | ||||
|           } | ||||
|         }); | ||||
|       }); | ||||
|       channels[i].test_models = test_models; | ||||
|     } | ||||
|     // data.key = '' + data.id | ||||
|     setChannels(channels); | ||||
|     if (channels.length >= pageSize) { | ||||
|       setChannelCount(channels.length + pageSize); | ||||
|     } else { | ||||
|       setChannelCount(channels.length); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   const loadChannels = async (startIdx, pageSize, idSort) => { | ||||
|     setLoading(true); | ||||
|     const res = await API.get(`/api/channel/?p=${startIdx}&page_size=${pageSize}&id_sort=${idSort}`); | ||||
|     const { success, message, data } = res.data; | ||||
|     if (success) { | ||||
|       if (startIdx === 0) { | ||||
|         setChannelFormat(data); | ||||
|       } else { | ||||
|         let newChannels = [...channels]; | ||||
|         newChannels.splice(startIdx * pageSize, data.length, ...data); | ||||
|         setChannelFormat(newChannels); | ||||
|       } | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
|     setLoading(false); | ||||
|   }; | ||||
|  | ||||
|   const refresh = async () => { | ||||
|     await loadChannels(activePage - 1, pageSize, idSort); | ||||
|   }; | ||||
|  | ||||
|   useEffect(() => { | ||||
|     // console.log('default effect') | ||||
|     const localIdSort = localStorage.getItem('id-sort') === 'true'; | ||||
|     const localPageSize = parseInt(localStorage.getItem('page-size')) || ITEMS_PER_PAGE; | ||||
|     setIdSort(localIdSort); | ||||
|     setPageSize(localPageSize); | ||||
|     loadChannels(0, localPageSize, localIdSort) | ||||
|       .then() | ||||
|       .catch((reason) => { | ||||
|         showError(reason); | ||||
|       }); | ||||
|     fetchGroups().then(); | ||||
|   }, []); | ||||
|  | ||||
|   const manageChannel = async (id, action, record, value) => { | ||||
|     let data = { id }; | ||||
|     let res; | ||||
|     switch (action) { | ||||
|       case 'delete': | ||||
|         res = await API.delete(`/api/channel/${id}/`); | ||||
|         break; | ||||
|       case 'enable': | ||||
|         data.status = 1; | ||||
|         res = await API.put('/api/channel/', data); | ||||
|         break; | ||||
|       case 'disable': | ||||
|         data.status = 2; | ||||
|         res = await API.put('/api/channel/', data); | ||||
|         break; | ||||
|       case 'priority': | ||||
|         if (value === '') { | ||||
|           return; | ||||
|         } | ||||
|         data.priority = parseInt(value); | ||||
|         res = await API.put('/api/channel/', data); | ||||
|         break; | ||||
|       case 'weight': | ||||
|         if (value === '') { | ||||
|           return; | ||||
|         } | ||||
|         data.weight = parseInt(value); | ||||
|         if (data.weight < 0) { | ||||
|           data.weight = 0; | ||||
|         } | ||||
|         res = await API.put('/api/channel/', data); | ||||
|         break; | ||||
|     } | ||||
|     const { success, message } = res.data; | ||||
|     if (success) { | ||||
|       showSuccess('操作成功完成!'); | ||||
|       let channel = res.data.data; | ||||
|       let newChannels = [...channels]; | ||||
|       if (action === 'delete') { | ||||
|  | ||||
|       } else { | ||||
|         record.status = channel.status; | ||||
|       } | ||||
|       setChannels(newChannels); | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   const renderStatus = (status) => { | ||||
|     switch (status) { | ||||
|       case 1: | ||||
|         return <Tag size="large" color="green">已启用</Tag>; | ||||
|       case 2: | ||||
|         return ( | ||||
|           <Tag size="large" color="yellow"> | ||||
|             已禁用 | ||||
|           </Tag> | ||||
|         ); | ||||
|       case 3: | ||||
|         return ( | ||||
|           <Tag size="large" color="yellow"> | ||||
|             自动禁用 | ||||
|           </Tag> | ||||
|         ); | ||||
|       default: | ||||
|         return ( | ||||
|           <Tag size="large" color="grey"> | ||||
|             未知状态 | ||||
|           </Tag> | ||||
|         ); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   const renderResponseTime = (responseTime) => { | ||||
|     let time = responseTime / 1000; | ||||
|     time = time.toFixed(2) + ' 秒'; | ||||
|     if (responseTime === 0) { | ||||
|       return <Tag size="large" color="grey">未测试</Tag>; | ||||
|     } else if (responseTime <= 1000) { | ||||
|       return <Tag size="large" color="green">{time}</Tag>; | ||||
|     } else if (responseTime <= 3000) { | ||||
|       return <Tag size="large" color="lime">{time}</Tag>; | ||||
|     } else if (responseTime <= 5000) { | ||||
|       return <Tag size="large" color="yellow">{time}</Tag>; | ||||
|     } else { | ||||
|       return <Tag size="large" color="red">{time}</Tag>; | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   const searchChannels = async (searchKeyword, searchGroup, searchModel) => { | ||||
|     if (searchKeyword === '' && searchGroup === '' && searchModel === '') { | ||||
|       // if keyword is blank, load files instead. | ||||
|       await loadChannels(0, pageSize, idSort); | ||||
|       setActivePage(1); | ||||
|       return; | ||||
|     } | ||||
|     setSearching(true); | ||||
|     const res = await API.get(`/api/channel/search?keyword=${searchKeyword}&group=${searchGroup}&model=${searchModel}`); | ||||
|     const { success, message, data } = res.data; | ||||
|     if (success) { | ||||
|       setChannels(data); | ||||
|       setActivePage(1); | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
|     setSearching(false); | ||||
|   }; | ||||
|  | ||||
|   const testChannel = async (record, model) => { | ||||
|     const res = await API.get(`/api/channel/test/${record.id}?model=${model}`); | ||||
|     const { success, message, time } = res.data; | ||||
|     if (success) { | ||||
|       record.response_time = time * 1000; | ||||
|       record.test_time = Date.now() / 1000; | ||||
|       showInfo(`渠道 ${record.name} 测试成功,耗时 ${time.toFixed(2)} 秒。`); | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   const testChannels = async (scope) => { | ||||
|     const res = await API.get(`/api/channel/test?scope=${scope}`); | ||||
|     const { success, message } = res.data; | ||||
|     if (success) { | ||||
|       showInfo('已成功开始测试渠道,请刷新页面查看结果。'); | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   const deleteAllDisabledChannels = async () => { | ||||
|     const res = await API.delete(`/api/channel/disabled`); | ||||
|     const { success, message, data } = res.data; | ||||
|     if (success) { | ||||
|       showSuccess(`已删除所有禁用渠道,共计 ${data} 个`); | ||||
|       await refresh(); | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   const updateChannelBalance = async (record) => { | ||||
|     const res = await API.get(`/api/channel/update_balance/${record.id}/`); | ||||
|     const { success, message, balance } = res.data; | ||||
|     if (success) { | ||||
|       record.balance = balance; | ||||
|       record.balance_updated_time = Date.now() / 1000; | ||||
|       showInfo(`渠道 ${record.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 batchDeleteChannels = async () => { | ||||
|     if (selectedChannels.length === 0) { | ||||
|       showError('请先选择要删除的渠道!'); | ||||
|       return; | ||||
|     } | ||||
|     setLoading(true); | ||||
|     let ids = []; | ||||
|     selectedChannels.forEach((channel) => { | ||||
|       ids.push(channel.id); | ||||
|     }); | ||||
|     const res = await API.post(`/api/channel/batch`, { ids: ids }); | ||||
|     const { success, message, data } = res.data; | ||||
|     if (success) { | ||||
|       showSuccess(`已删除 ${data} 个渠道!`); | ||||
|       await refresh(); | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
|     setLoading(false); | ||||
|   }; | ||||
|  | ||||
|   const fixChannelsAbilities = async () => { | ||||
|     const res = await API.post(`/api/channel/fix`); | ||||
|     const { success, message, data } = res.data; | ||||
|     if (success) { | ||||
|       showSuccess(`已修复 ${data} 个渠道!`); | ||||
|       await refresh(); | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   let pageData = channels.slice((activePage - 1) * pageSize, activePage * pageSize); | ||||
|  | ||||
|   const handlePageChange = page => { | ||||
|     setActivePage(page); | ||||
|     if (page === Math.ceil(channels.length / pageSize) + 1) { | ||||
|       // In this case we have to load more data and then append them. | ||||
|       loadChannels(page - 1, pageSize, idSort).then(r => { | ||||
|       }); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   const handlePageSizeChange = async (size) => { | ||||
|     localStorage.setItem('page-size', size + ''); | ||||
|     setPageSize(size); | ||||
|     setActivePage(1); | ||||
|     loadChannels(0, size, idSort) | ||||
|       .then() | ||||
|       .catch((reason) => { | ||||
|         showError(reason); | ||||
|       }); | ||||
|   }; | ||||
|  | ||||
|   const fetchGroups = async () => { | ||||
|     try { | ||||
|       let res = await API.get(`/api/group/`); | ||||
|       // add 'all' option | ||||
|       // res.data.data.unshift('all'); | ||||
|       setGroupOptions(res.data.data.map((group) => ({ | ||||
|         label: group, | ||||
|         value: group | ||||
|       }))); | ||||
|     } catch (error) { | ||||
|       showError(error.message); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   const closeEdit = () => { | ||||
|     setShowEdit(false); | ||||
|   }; | ||||
|  | ||||
|   const handleRow = (record, index) => { | ||||
|     if (record.status !== 1) { | ||||
|       return { | ||||
|         style: { | ||||
|           background: 'var(--semi-color-disabled-border)' | ||||
|         } | ||||
|       }; | ||||
|     } else { | ||||
|       return {}; | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|  | ||||
|   return ( | ||||
|     <> | ||||
|       <EditChannel refresh={refresh} visible={showEdit} handleClose={closeEdit} editingChannel={editingChannel} /> | ||||
|       <div style={{ display: "flex", placeItems: "center", justifyContent: "space-between" }}> | ||||
|         <Form onSubmit={() => { | ||||
|           searchChannels(searchKeyword, searchGroup, searchModel); | ||||
|         }} labelPosition="left"> | ||||
|           <div style={{ display: 'flex' }}> | ||||
|             <Space> | ||||
|               <Form.Input | ||||
|                 field="search_keyword" | ||||
|                 label="搜索" | ||||
|                 placeholder="ID,名称和密钥 ..." | ||||
|                 value={searchKeyword} | ||||
|                 loading={searching} | ||||
|                 onChange={(v) => { | ||||
|                   setSearchKeyword(v.trim()); | ||||
|                 }} | ||||
|               /> | ||||
|               {/* <Form.Input | ||||
|               field="search_model" | ||||
|               label="模型" | ||||
|               placeholder="模型关键字" | ||||
|               value={searchModel} | ||||
|               loading={searching} | ||||
|               onChange={(v) => { | ||||
|                 setSearchModel(v.trim()); | ||||
|               }} | ||||
|             /> | ||||
|             <Form.Select field="group" label="分组" optionList={groupOptions} onChange={(v) => { | ||||
|               setSearchGroup(v); | ||||
|               searchChannels(searchKeyword, v, searchModel); | ||||
|             }} /> */} | ||||
|               <Button label="查询" type="primary" htmlType="submit" className="btn-margin-right" | ||||
|                 style={{ marginRight: 8 }}>查询</Button> | ||||
|             </Space> | ||||
|           </div> | ||||
|         </Form> | ||||
|         <div style={{ | ||||
|           display: isMobile() ? '' : 'flex', | ||||
|           marginTop: isMobile() ? 0 : -45, | ||||
|           zIndex: 999, | ||||
|           position: 'relative', | ||||
|           pointerEvents: 'none' | ||||
|         }}> | ||||
|           <Space style={{ pointerEvents: 'auto', marginTop: isMobile() ? 0 : 45 }}> | ||||
|             <Button theme="light" type="primary" style={{ marginRight: 8 }} onClick={ | ||||
|               () => { | ||||
|                 setEditingChannel({ | ||||
|                   id: undefined | ||||
|                 }); | ||||
|                 setShowEdit(true); | ||||
|               } | ||||
|             }>添加新的渠道</Button> | ||||
|             <Popconfirm | ||||
|               title="确定?" | ||||
|               okType={'warning'} | ||||
|               onConfirm={() => { testChannels("all") }} | ||||
|               position={isMobile() ? 'top' : 'left'} | ||||
|             > | ||||
|               <Button theme="light" type="warning" style={{ marginRight: 8 }}>测试所有渠道</Button> | ||||
|             </Popconfirm> | ||||
|             <Popconfirm | ||||
|               title="确定?" | ||||
|               okType={'warning'} | ||||
|               onConfirm={() => { testChannels("disabled") }} | ||||
|               position={isMobile() ? 'top' : 'left'} | ||||
|             > | ||||
|               <Button theme="light" type="warning" style={{ marginRight: 8 }}>测试禁用渠道</Button> | ||||
|             </Popconfirm> | ||||
|             {/* <Popconfirm | ||||
|             title="确定?" | ||||
|             okType={'secondary'} | ||||
|             onConfirm={updateAllChannelsBalance} | ||||
|           > | ||||
|             <Button theme="light" type="secondary" style={{ marginRight: 8 }}>更新所有已启用渠道余额</Button> | ||||
|           </Popconfirm> */} | ||||
|             <Popconfirm | ||||
|               title="确定是否要删除禁用渠道?" | ||||
|               content="此修改将不可逆" | ||||
|               okType={'danger'} | ||||
|               onConfirm={deleteAllDisabledChannels} | ||||
|               position={isMobile() ? 'top' : 'left'} | ||||
|             > | ||||
|               <Button theme="light" type="danger" style={{ marginRight: 8 }}>删除禁用渠道</Button> | ||||
|             </Popconfirm> | ||||
|  | ||||
|             <Button theme="light" type="primary" style={{ marginRight: 8 }} onClick={refresh}>刷新</Button> | ||||
|           </Space> | ||||
|           {/*<div style={{width: '100%', pointerEvents: 'none', position: 'absolute'}}>*/} | ||||
|  | ||||
|           {/*</div>*/} | ||||
|         </div> | ||||
|         {/* <div style={{ marginTop: 20 }}> | ||||
|           <Space> | ||||
|             <Typography.Text strong>开启批量删除</Typography.Text> | ||||
|             <Switch label="开启批量删除" uncheckedText="关" aria-label="是否开启批量删除" onChange={(v) => { | ||||
|               setEnableBatchDelete(v); | ||||
|             }}></Switch> | ||||
|             <Popconfirm | ||||
|               title="确定是否要删除所选渠道?" | ||||
|               content="此修改将不可逆" | ||||
|               okType={'danger'} | ||||
|               onConfirm={batchDeleteChannels} | ||||
|               disabled={!enableBatchDelete} | ||||
|               position={'top'} | ||||
|             > | ||||
|               <Button disabled={!enableBatchDelete} theme="light" type="danger" | ||||
|                 style={{ marginRight: 8 }}>删除所选渠道</Button> | ||||
|             </Popconfirm> | ||||
|             <Popconfirm | ||||
|               title="确定是否要修复数据库一致性?" | ||||
|               content="进行该操作时,可能导致渠道访问错误,请仅在数据库出现问题时使用" | ||||
|               okType={'warning'} | ||||
|               onConfirm={fixChannelsAbilities} | ||||
|               position={'top'} | ||||
|             > | ||||
|               <Button theme="light" type="secondary" style={{ marginRight: 8 }}>修复数据库一致性</Button> | ||||
|             </Popconfirm> | ||||
|           </Space> | ||||
|         </div> | ||||
|         <div style={{ marginTop: 10, display: 'flex' }}> | ||||
|           <Space> | ||||
|             <Space> | ||||
|               <Typography.Text strong>使用ID排序</Typography.Text> | ||||
|               <Switch checked={idSort} label="使用ID排序" uncheckedText="关" aria-label="是否用ID排序" onChange={(v) => { | ||||
|                 localStorage.setItem('id-sort', v + ''); | ||||
|                 setIdSort(v); | ||||
|                 loadChannels(0, pageSize, v) | ||||
|                   .then() | ||||
|                   .catch((reason) => { | ||||
|                     showError(reason); | ||||
|                   }); | ||||
|               }}></Switch> | ||||
|             </Space> | ||||
|           </Space> | ||||
|         </div> */} | ||||
|       </div> | ||||
|       <Table className={'channel-table'} style={{ marginTop: 15 }} columns={columns} dataSource={pageData} pagination={{ | ||||
|         currentPage: activePage, | ||||
|         pageSize: pageSize, | ||||
|         total: channelCount, | ||||
|         pageSizeOpts: [10, 20, 50, 100], | ||||
|         showSizeChanger: true, | ||||
|         formatPageText: (page) => '', | ||||
|         onPageSizeChange: (size) => { | ||||
|           handlePageSizeChange(size).then(); | ||||
|         }, | ||||
|         onPageChange: handlePageChange | ||||
|       }} loading={loading} onRow={handleRow} rowSelection={ | ||||
|         enableBatchDelete ? | ||||
|           { | ||||
|             onChange: (selectedRowKeys, selectedRows) => { | ||||
|               // console.log(`selectedRowKeys: ${selectedRowKeys}`, 'selectedRows: ', selectedRows); | ||||
|               setSelectedChannels(selectedRows); | ||||
|             } | ||||
|           } : null | ||||
|       } /> | ||||
|     </> | ||||
|   ); | ||||
| }; | ||||
|  | ||||
| export default ChannelsTable; | ||||
							
								
								
									
										64
									
								
								web/air/src/components/Footer.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										64
									
								
								web/air/src/components/Footer.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,64 @@ | ||||
| import React, { useEffect, useState } from 'react'; | ||||
|  | ||||
| import { Container, Segment } from 'semantic-ui-react'; | ||||
| import { getFooterHTML, getSystemName } from '../helpers'; | ||||
|  | ||||
| const Footer = () => { | ||||
|   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(() => { | ||||
|     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' | ||||
|             dangerouslySetInnerHTML={{ __html: footer }} | ||||
|           ></div> | ||||
|         ) : ( | ||||
|           <div className='custom-footer'> | ||||
|             <a | ||||
|               href='https://github.com/songquanpeng/one-api' | ||||
|               target='_blank' | ||||
|             > | ||||
|               {systemName} {process.env.REACT_APP_VERSION}{' '} | ||||
|             </a> | ||||
|             由{' '} | ||||
|             <a href='https://github.com/songquanpeng' target='_blank'> | ||||
|               JustSong | ||||
|             </a>{' '} | ||||
|             构建,主题 air 来自{' '} | ||||
|             <a href='https://github.com/Calcium-Ion' target='_blank'> | ||||
|               Calon | ||||
|             </a>{' '},源代码遵循{' '} | ||||
|             <a href='https://opensource.org/licenses/mit-license.php'> | ||||
|               MIT 协议 | ||||
|             </a> | ||||
|           </div> | ||||
|         )} | ||||
|       </Container> | ||||
|     </Segment> | ||||
|   ); | ||||
| }; | ||||
|  | ||||
| export default Footer; | ||||
							
								
								
									
										58
									
								
								web/air/src/components/GitHubOAuth.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										58
									
								
								web/air/src/components/GitHubOAuth.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,58 @@ | ||||
| import React, { useContext, useEffect, useState } from 'react'; | ||||
| import { Dimmer, Loader, Segment } from 'semantic-ui-react'; | ||||
| import { useNavigate, useSearchParams } from 'react-router-dom'; | ||||
| import { API, showError, showSuccess } from '../helpers'; | ||||
| import { UserContext } from '../context/User'; | ||||
|  | ||||
| const GitHubOAuth = () => { | ||||
|   const [searchParams, setSearchParams] = useSearchParams(); | ||||
|  | ||||
|   const [userState, userDispatch] = useContext(UserContext); | ||||
|   const [prompt, setPrompt] = useState('处理中...'); | ||||
|   const [processing, setProcessing] = useState(true); | ||||
|  | ||||
|   let navigate = useNavigate(); | ||||
|  | ||||
|   const sendCode = async (code, state, count) => { | ||||
|     const res = await API.get(`/api/oauth/github?code=${code}&state=${state}`); | ||||
|     const { success, message, data } = res.data; | ||||
|     if (success) { | ||||
|       if (message === 'bind') { | ||||
|         showSuccess('绑定成功!'); | ||||
|         navigate('/setting'); | ||||
|       } else { | ||||
|         userDispatch({ type: 'login', payload: data }); | ||||
|         localStorage.setItem('user', JSON.stringify(data)); | ||||
|         showSuccess('登录成功!'); | ||||
|         navigate('/'); | ||||
|       } | ||||
|     } else { | ||||
|       showError(message); | ||||
|       if (count === 0) { | ||||
|         setPrompt(`操作失败,重定向至登录界面中...`); | ||||
|         navigate('/setting'); // in case this is failed to bind GitHub | ||||
|         return; | ||||
|       } | ||||
|       count++; | ||||
|       setPrompt(`出现错误,第 ${count} 次重试中...`); | ||||
|       await new Promise((resolve) => setTimeout(resolve, count * 2000)); | ||||
|       await sendCode(code, state, count); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   useEffect(() => { | ||||
|     let code = searchParams.get('code'); | ||||
|     let state = searchParams.get('state'); | ||||
|     sendCode(code, state, 0).then(); | ||||
|   }, []); | ||||
|  | ||||
|   return ( | ||||
|     <Segment style={{ minHeight: '300px' }}> | ||||
|       <Dimmer active inverted> | ||||
|         <Loader size="large">{prompt}</Loader> | ||||
|       </Dimmer> | ||||
|     </Segment> | ||||
|   ); | ||||
| }; | ||||
|  | ||||
| export default GitHubOAuth; | ||||
							
								
								
									
										161
									
								
								web/air/src/components/HeaderBar.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										161
									
								
								web/air/src/components/HeaderBar.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,161 @@ | ||||
| import React, { useContext, useEffect, useState } from 'react'; | ||||
| import { Link, useNavigate } from 'react-router-dom'; | ||||
| import { UserContext } from '../context/User'; | ||||
|  | ||||
| import { API, getLogo, getSystemName, showSuccess } from '../helpers'; | ||||
| import '../index.css'; | ||||
|  | ||||
| import fireworks from 'react-fireworks'; | ||||
|  | ||||
| import { IconHelpCircle, IconKey, IconUser } from '@douyinfe/semi-icons'; | ||||
| import { Avatar, Dropdown, Layout, Nav, Switch } from '@douyinfe/semi-ui'; | ||||
| import { stringToColor } from '../helpers/render'; | ||||
|  | ||||
| // HeaderBar Buttons | ||||
| let headerButtons = [ | ||||
|   { | ||||
|     text: '关于', | ||||
|     itemKey: 'about', | ||||
|     to: '/about', | ||||
|     icon: <IconHelpCircle /> | ||||
|   } | ||||
| ]; | ||||
|  | ||||
| if (localStorage.getItem('chat_link')) { | ||||
|   headerButtons.splice(1, 0, { | ||||
|     name: '聊天', | ||||
|     to: '/chat', | ||||
|     icon: 'comments' | ||||
|   }); | ||||
| } | ||||
|  | ||||
| const HeaderBar = () => { | ||||
|   const [userState, userDispatch] = useContext(UserContext); | ||||
|   let navigate = useNavigate(); | ||||
|  | ||||
|   const [showSidebar, setShowSidebar] = useState(false); | ||||
|   const [dark, setDark] = useState(false); | ||||
|   const systemName = getSystemName(); | ||||
|   const logo = getLogo(); | ||||
|   var themeMode = localStorage.getItem('theme-mode'); | ||||
|   const currentDate = new Date(); | ||||
|   // enable fireworks on new year(1.1 and 2.9-2.24) | ||||
|   const isNewYear = (currentDate.getMonth() === 0 && currentDate.getDate() === 1) || (currentDate.getMonth() === 1 && currentDate.getDate() >= 9 && currentDate.getDate() <= 24); | ||||
|  | ||||
|   async function logout() { | ||||
|     setShowSidebar(false); | ||||
|     await API.get('/api/user/logout'); | ||||
|     showSuccess('注销成功!'); | ||||
|     userDispatch({ type: 'logout' }); | ||||
|     localStorage.removeItem('user'); | ||||
|     navigate('/login'); | ||||
|   } | ||||
|  | ||||
|   const handleNewYearClick = () => { | ||||
|     fireworks.init('root', {}); | ||||
|     fireworks.start(); | ||||
|     setTimeout(() => { | ||||
|       fireworks.stop(); | ||||
|       setTimeout(() => { | ||||
|         window.location.reload(); | ||||
|       }, 10000); | ||||
|     }, 3000); | ||||
|   }; | ||||
|  | ||||
|   useEffect(() => { | ||||
|     if (themeMode === 'dark') { | ||||
|       switchMode(true); | ||||
|     } | ||||
|     if (isNewYear) { | ||||
|       console.log('Happy New Year!'); | ||||
|     } | ||||
|   }, []); | ||||
|  | ||||
|   const switchMode = (model) => { | ||||
|     const body = document.body; | ||||
|     if (!model) { | ||||
|       body.removeAttribute('theme-mode'); | ||||
|       localStorage.setItem('theme-mode', 'light'); | ||||
|     } else { | ||||
|       body.setAttribute('theme-mode', 'dark'); | ||||
|       localStorage.setItem('theme-mode', 'dark'); | ||||
|     } | ||||
|     setDark(model); | ||||
|   }; | ||||
|   return ( | ||||
|     <> | ||||
|       <Layout> | ||||
|         <div style={{ width: '100%' }}> | ||||
|           <Nav | ||||
|             mode={'horizontal'} | ||||
|             // bodyStyle={{ height: 100 }} | ||||
|             renderWrapper={({ itemElement, isSubNav, isInSubNav, props }) => { | ||||
|               const routerMap = { | ||||
|                 about: '/about', | ||||
|                 login: '/login', | ||||
|                 register: '/register' | ||||
|               }; | ||||
|               return ( | ||||
|                 <Link | ||||
|                   style={{ textDecoration: 'none' }} | ||||
|                   to={routerMap[props.itemKey]} | ||||
|                 > | ||||
|                   {itemElement} | ||||
|                 </Link> | ||||
|               ); | ||||
|             }} | ||||
|             selectedKeys={[]} | ||||
|             // items={headerButtons} | ||||
|             onSelect={key => { | ||||
|  | ||||
|             }} | ||||
|             footer={ | ||||
|               <> | ||||
|                 {isNewYear && | ||||
|                   // happy new year | ||||
|                   <Dropdown | ||||
|                     position="bottomRight" | ||||
|                     render={ | ||||
|                       <Dropdown.Menu> | ||||
|                         <Dropdown.Item onClick={handleNewYearClick}>Happy New Year!!!</Dropdown.Item> | ||||
|                       </Dropdown.Menu> | ||||
|                     } | ||||
|                   > | ||||
|                     <Nav.Item itemKey={'new-year'} text={'🏮'} /> | ||||
|                   </Dropdown> | ||||
|                 } | ||||
|                 <Nav.Item itemKey={'about'} icon={<IconHelpCircle />} /> | ||||
|                 <Switch checkedText="🌞" size={'large'} checked={dark} uncheckedText="🌙" onChange={switchMode} /> | ||||
|                 {userState.user ? | ||||
|                   <> | ||||
|                     <Dropdown | ||||
|                       position="bottomRight" | ||||
|                       render={ | ||||
|                         <Dropdown.Menu> | ||||
|                           <Dropdown.Item onClick={logout}>退出</Dropdown.Item> | ||||
|                         </Dropdown.Menu> | ||||
|                       } | ||||
|                     > | ||||
|                       <Avatar size="small" color={stringToColor(userState.user.username)} style={{ margin: 4 }}> | ||||
|                         {userState.user.username[0]} | ||||
|                       </Avatar> | ||||
|                       <span>{userState.user.username}</span> | ||||
|                     </Dropdown> | ||||
|                   </> | ||||
|                   : | ||||
|                   <> | ||||
|                     <Nav.Item itemKey={'login'} text={'登录'} icon={<IconKey />} /> | ||||
|                     <Nav.Item itemKey={'register'} text={'注册'} icon={<IconUser />} /> | ||||
|                   </> | ||||
|                 } | ||||
|               </> | ||||
|             } | ||||
|           > | ||||
|           </Nav> | ||||
|         </div> | ||||
|       </Layout> | ||||
|     </> | ||||
|   ); | ||||
| }; | ||||
|  | ||||
| export default HeaderBar; | ||||
							
								
								
									
										14
									
								
								web/air/src/components/Loading.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								web/air/src/components/Loading.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | ||||
| import React from 'react'; | ||||
| import { Dimmer, Loader, Segment } from 'semantic-ui-react'; | ||||
|  | ||||
| const Loading = ({ prompt: name = 'page' }) => { | ||||
|   return ( | ||||
|     <Segment style={{ height: 100 }}> | ||||
|       <Dimmer active inverted> | ||||
|         <Loader indeterminate>加载{name}中...</Loader> | ||||
|       </Dimmer> | ||||
|     </Segment> | ||||
|   ); | ||||
| }; | ||||
|  | ||||
| export default Loading; | ||||
							
								
								
									
										254
									
								
								web/air/src/components/LoginForm.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										254
									
								
								web/air/src/components/LoginForm.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,254 @@ | ||||
| import React, { useContext, useEffect, useState } from 'react'; | ||||
| import { Link, useNavigate, useSearchParams } from 'react-router-dom'; | ||||
| import { UserContext } from '../context/User'; | ||||
| import { API, getLogo, showError, showInfo, showSuccess } from '../helpers'; | ||||
| import { onGitHubOAuthClicked } from './utils'; | ||||
| import Turnstile from 'react-turnstile'; | ||||
| import { Button, Card, Divider, Form, Icon, Layout, Modal } from '@douyinfe/semi-ui'; | ||||
| import Title from '@douyinfe/semi-ui/lib/es/typography/title'; | ||||
| import Text from '@douyinfe/semi-ui/lib/es/typography/text'; | ||||
| import TelegramLoginButton from 'react-telegram-login'; | ||||
|  | ||||
| import { IconGithubLogo } from '@douyinfe/semi-icons'; | ||||
| import WeChatIcon from './WeChatIcon'; | ||||
|  | ||||
| const LoginForm = () => { | ||||
|   const [inputs, setInputs] = useState({ | ||||
|     username: '', | ||||
|     password: '', | ||||
|     wechat_verification_code: '' | ||||
|   }); | ||||
|   const [searchParams, setSearchParams] = useSearchParams(); | ||||
|   const [submitted, setSubmitted] = useState(false); | ||||
|   const { username, password } = inputs; | ||||
|   const [userState, userDispatch] = useContext(UserContext); | ||||
|   const [turnstileEnabled, setTurnstileEnabled] = useState(false); | ||||
|   const [turnstileSiteKey, setTurnstileSiteKey] = useState(''); | ||||
|   const [turnstileToken, setTurnstileToken] = useState(''); | ||||
|   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); | ||||
|       setStatus(status); | ||||
|       if (status.turnstile_check) { | ||||
|         setTurnstileEnabled(true); | ||||
|         setTurnstileSiteKey(status.turnstile_site_key); | ||||
|       } | ||||
|     } | ||||
|   }, []); | ||||
|  | ||||
|   const [showWeChatLoginModal, setShowWeChatLoginModal] = useState(false); | ||||
|  | ||||
|   const onWeChatLoginClicked = () => { | ||||
|     setShowWeChatLoginModal(true); | ||||
|   }; | ||||
|  | ||||
|   const onSubmitWeChatVerificationCode = async () => { | ||||
|     if (turnstileEnabled && turnstileToken === '') { | ||||
|       showInfo('请稍后几秒重试,Turnstile 正在检查用户环境!'); | ||||
|       return; | ||||
|     } | ||||
|     const res = await API.get( | ||||
|       `/api/oauth/wechat?code=${inputs.wechat_verification_code}` | ||||
|     ); | ||||
|     const { success, message, data } = res.data; | ||||
|     if (success) { | ||||
|       userDispatch({ type: 'login', payload: data }); | ||||
|       localStorage.setItem('user', JSON.stringify(data)); | ||||
|       navigate('/'); | ||||
|       showSuccess('登录成功!'); | ||||
|       setShowWeChatLoginModal(false); | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   function handleChange(name, value) { | ||||
|     setInputs((inputs) => ({ ...inputs, [name]: value })); | ||||
|   } | ||||
|  | ||||
|   async function handleSubmit(e) { | ||||
|     if (turnstileEnabled && turnstileToken === '') { | ||||
|       showInfo('请稍后几秒重试,Turnstile 正在检查用户环境!'); | ||||
|       return; | ||||
|     } | ||||
|     setSubmitted(true); | ||||
|     if (username && password) { | ||||
|       const res = await API.post(`/api/user/login?turnstile=${turnstileToken}`, { | ||||
|         username, | ||||
|         password | ||||
|       }); | ||||
|       const { success, message, data } = res.data; | ||||
|       if (success) { | ||||
|         userDispatch({ type: 'login', payload: data }); | ||||
|         localStorage.setItem('user', JSON.stringify(data)); | ||||
|         showSuccess('登录成功!'); | ||||
|         if (username === 'root' && password === '123456') { | ||||
|           Modal.error({ title: '您正在使用默认密码!', content: '请立刻修改默认密码!', centered: true }); | ||||
|         } | ||||
|         navigate('/token'); | ||||
|       } else { | ||||
|         showError(message); | ||||
|       } | ||||
|     } else { | ||||
|       showError('请输入用户名和密码!'); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   // 添加Telegram登录处理函数 | ||||
|   const onTelegramLoginClicked = async (response) => { | ||||
|     const fields = ['id', 'first_name', 'last_name', 'username', 'photo_url', 'auth_date', 'hash', 'lang']; | ||||
|     const params = {}; | ||||
|     fields.forEach((field) => { | ||||
|       if (response[field]) { | ||||
|         params[field] = response[field]; | ||||
|       } | ||||
|     }); | ||||
|     const res = await API.get(`/api/oauth/telegram/login`, { params }); | ||||
|     const { success, message, data } = res.data; | ||||
|     if (success) { | ||||
|       userDispatch({ type: 'login', payload: data }); | ||||
|       localStorage.setItem('user', JSON.stringify(data)); | ||||
|       showSuccess('登录成功!'); | ||||
|       navigate('/'); | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   return ( | ||||
|     <div> | ||||
|       <Layout> | ||||
|         <Layout.Header> | ||||
|         </Layout.Header> | ||||
|         <Layout.Content> | ||||
|           <div style={{ justifyContent: 'center', display: 'flex', marginTop: 120 }}> | ||||
|             <div style={{ width: 500 }}> | ||||
|               <Card> | ||||
|                 <Title heading={2} style={{ textAlign: 'center' }}> | ||||
|                   用户登录 | ||||
|                 </Title> | ||||
|                 <Form> | ||||
|                   <Form.Input | ||||
|                     field={'username'} | ||||
|                     label={'用户名'} | ||||
|                     placeholder="用户名" | ||||
|                     name="username" | ||||
|                     onChange={(value) => handleChange('username', value)} | ||||
|                   /> | ||||
|                   <Form.Input | ||||
|                     field={'password'} | ||||
|                     label={'密码'} | ||||
|                     placeholder="密码" | ||||
|                     name="password" | ||||
|                     type="password" | ||||
|                     onChange={(value) => handleChange('password', value)} | ||||
|                   /> | ||||
|  | ||||
|                   <Button theme="solid" style={{ width: '100%' }} type={'primary'} size="large" | ||||
|                           htmlType={'submit'} onClick={handleSubmit}> | ||||
|                     登录 | ||||
|                   </Button> | ||||
|                 </Form> | ||||
|                 <div style={{ display: 'flex', justifyContent: 'space-between', marginTop: 20 }}> | ||||
|                   <Text> | ||||
|                     没有账号请先 <Link to="/register">注册账号</Link> | ||||
|                   </Text> | ||||
|                   <Text> | ||||
|                     忘记密码 <Link to="/reset">点击重置</Link> | ||||
|                   </Text> | ||||
|                 </div> | ||||
|                 {status.github_oauth || status.wechat_login || status.telegram_oauth ? ( | ||||
|                   <> | ||||
|                     <Divider margin="12px" align="center"> | ||||
|                       第三方登录 | ||||
|                     </Divider> | ||||
|                     <div style={{ display: 'flex', justifyContent: 'center', marginTop: 20 }}> | ||||
|                       {status.github_oauth ? ( | ||||
|                         <Button | ||||
|                           type="primary" | ||||
|                           icon={<IconGithubLogo />} | ||||
|                           onClick={() => onGitHubOAuthClicked(status.github_client_id)} | ||||
|                         /> | ||||
|                       ) : ( | ||||
|                         <></> | ||||
|                       )} | ||||
|                       {status.wechat_login ? ( | ||||
|                         <Button | ||||
|                           type="primary" | ||||
|                           style={{ color: 'rgba(var(--semi-green-5), 1)' }} | ||||
|                           icon={<Icon svg={<WeChatIcon />} />} | ||||
|                           onClick={onWeChatLoginClicked} | ||||
|                         /> | ||||
|                       ) : ( | ||||
|                         <></> | ||||
|                       )} | ||||
|  | ||||
|                       {status.telegram_oauth ? ( | ||||
|                         <TelegramLoginButton dataOnauth={onTelegramLoginClicked} botName={status.telegram_bot_name} /> | ||||
|                       ) : ( | ||||
|                         <></> | ||||
|                       )} | ||||
|                     </div> | ||||
|                   </> | ||||
|                 ) : ( | ||||
|                   <></> | ||||
|                 )} | ||||
|                 <Modal | ||||
|                   title="微信扫码登录" | ||||
|                   visible={showWeChatLoginModal} | ||||
|                   maskClosable={true} | ||||
|                   onOk={onSubmitWeChatVerificationCode} | ||||
|                   onCancel={() => setShowWeChatLoginModal(false)} | ||||
|                   okText={'登录'} | ||||
|                   size={'small'} | ||||
|                   centered={true} | ||||
|                 > | ||||
|                   <div style={{ display: 'flex', alignItem: 'center', flexDirection: 'column' }}> | ||||
|                     <img src={status.wechat_qrcode} /> | ||||
|                   </div> | ||||
|                   <div style={{ textAlign: 'center' }}> | ||||
|                     <p> | ||||
|                       微信扫码关注公众号,输入「验证码」获取验证码(三分钟内有效) | ||||
|                     </p> | ||||
|                   </div> | ||||
|                   <Form size="large"> | ||||
|                     <Form.Input | ||||
|                       field={'wechat_verification_code'} | ||||
|                       placeholder="验证码" | ||||
|                       label={'验证码'} | ||||
|                       value={inputs.wechat_verification_code} | ||||
|                       onChange={(value) => handleChange('wechat_verification_code', value)} | ||||
|                     /> | ||||
|                   </Form> | ||||
|                 </Modal> | ||||
|               </Card> | ||||
|               {turnstileEnabled ? ( | ||||
|                 <div style={{ display: 'flex', justifyContent: 'center', marginTop: 20 }}> | ||||
|                   <Turnstile | ||||
|                     sitekey={turnstileSiteKey} | ||||
|                     onVerify={(token) => { | ||||
|                       setTurnstileToken(token); | ||||
|                     }} | ||||
|                   /> | ||||
|                 </div> | ||||
|               ) : ( | ||||
|                 <></> | ||||
|               )} | ||||
|             </div> | ||||
|           </div> | ||||
|  | ||||
|         </Layout.Content> | ||||
|       </Layout> | ||||
|     </div> | ||||
|   ); | ||||
| }; | ||||
|  | ||||
| export default LoginForm; | ||||
							
								
								
									
										401
									
								
								web/air/src/components/LogsTable.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										401
									
								
								web/air/src/components/LogsTable.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,401 @@ | ||||
| import React, { useEffect, useState } from 'react'; | ||||
| import { API, copy, isAdmin, showError, showSuccess, timestamp2string } from '../helpers'; | ||||
|  | ||||
| import { Avatar, Button, Form, Layout, Modal, Select, Space, Spin, Table, Tag } from '@douyinfe/semi-ui'; | ||||
| import { ITEMS_PER_PAGE } from '../constants'; | ||||
| import { renderNumber, renderQuota, stringToColor } from '../helpers/render'; | ||||
| import Paragraph from '@douyinfe/semi-ui/lib/es/typography/paragraph'; | ||||
|  | ||||
| const { Header } = Layout; | ||||
|  | ||||
| function renderTimestamp(timestamp) { | ||||
|   return (<> | ||||
|     {timestamp2string(timestamp)} | ||||
|   </>); | ||||
| } | ||||
|  | ||||
| const MODE_OPTIONS = [{ key: 'all', text: '全部用户', value: 'all' }, { key: 'self', text: '当前用户', value: 'self' }]; | ||||
|  | ||||
| const colors = ['amber', 'blue', 'cyan', 'green', 'grey', 'indigo', 'light-blue', 'lime', 'orange', 'pink', 'purple', 'red', 'teal', 'violet', 'yellow']; | ||||
|  | ||||
| function renderType(type) { | ||||
|   switch (type) { | ||||
|     case 1: | ||||
|       return <Tag color="cyan" size="large"> 充值 </Tag>; | ||||
|     case 2: | ||||
|       return <Tag color="lime" size="large"> 消费 </Tag>; | ||||
|     case 3: | ||||
|       return <Tag color="orange" size="large"> 管理 </Tag>; | ||||
|     case 4: | ||||
|       return <Tag color="purple" size="large"> 系统 </Tag>; | ||||
|     default: | ||||
|       return <Tag color="black" size="large"> 未知 </Tag>; | ||||
|   } | ||||
| } | ||||
|  | ||||
| function renderIsStream(bool) { | ||||
|   if (bool) { | ||||
|     return <Tag color="blue" size="large">流</Tag>; | ||||
|   } else { | ||||
|     return <Tag color="purple" size="large">非流</Tag>; | ||||
|   } | ||||
| } | ||||
|  | ||||
| function renderUseTime(type) { | ||||
|   const time = parseInt(type); | ||||
|   if (time < 101) { | ||||
|     return <Tag color="green" size="large"> {time} s </Tag>; | ||||
|   } else if (time < 300) { | ||||
|     return <Tag color="orange" size="large"> {time} s </Tag>; | ||||
|   } else { | ||||
|     return <Tag color="red" size="large"> {time} s </Tag>; | ||||
|   } | ||||
| } | ||||
|  | ||||
| const LogsTable = () => { | ||||
|   const columns = [{ | ||||
|     title: '时间', dataIndex: 'timestamp2string' | ||||
|   }, { | ||||
|     title: '渠道', | ||||
|     dataIndex: 'channel', | ||||
|     className: isAdmin() ? 'tableShow' : 'tableHiddle', | ||||
|     render: (text, record, index) => { | ||||
|       return (isAdminUser ? record.type === 0 || record.type === 2 ? <div> | ||||
|         {<Tag color={colors[parseInt(text) % colors.length]} size="large"> {text} </Tag>} | ||||
|       </div> : <></> : <></>); | ||||
|     } | ||||
|   }, { | ||||
|     title: '用户', | ||||
|     dataIndex: 'username', | ||||
|     className: isAdmin() ? 'tableShow' : 'tableHiddle', | ||||
|     render: (text, record, index) => { | ||||
|       return (isAdminUser ? <div> | ||||
|         <Avatar size="small" color={stringToColor(text)} style={{ marginRight: 4 }} | ||||
|           onClick={() => showUserInfo(record.user_id)}> | ||||
|           {typeof text === 'string' && text.slice(0, 1)} | ||||
|         </Avatar> | ||||
|         {text} | ||||
|       </div> : <></>); | ||||
|     } | ||||
|   }, { | ||||
|     title: '令牌', dataIndex: 'token_name', render: (text, record, index) => { | ||||
|       return (record.type === 0 || record.type === 2 ? <div> | ||||
|         <Tag color="grey" size="large" onClick={() => { | ||||
|           copyText(text); | ||||
|         }}> {text} </Tag> | ||||
|       </div> : <></>); | ||||
|     } | ||||
|   }, { | ||||
|     title: '类型', dataIndex: 'type', render: (text, record, index) => { | ||||
|       return (<div> | ||||
|         {renderType(text)} | ||||
|       </div>); | ||||
|     } | ||||
|   }, { | ||||
|     title: '模型', dataIndex: 'model_name', render: (text, record, index) => { | ||||
|       return (record.type === 0 || record.type === 2 ? <div> | ||||
|         <Tag color={stringToColor(text)} size="large" onClick={() => { | ||||
|           copyText(text); | ||||
|         }}> {text} </Tag> | ||||
|       </div> : <></>); | ||||
|     } | ||||
|   }, | ||||
|   // { | ||||
|   //   title: '用时', dataIndex: 'use_time', render: (text, record, index) => { | ||||
|   //     return (<div> | ||||
|   //       <Space> | ||||
|   //         {renderUseTime(text)} | ||||
|   //         {renderIsStream(record.is_stream)} | ||||
|   //       </Space> | ||||
|   //     </div>); | ||||
|   //   } | ||||
|   // }, | ||||
|   { | ||||
|     title: '提示', dataIndex: 'prompt_tokens', render: (text, record, index) => { | ||||
|       return (record.type === 0 || record.type === 2 ? <div> | ||||
|         {<span> {text} </span>} | ||||
|       </div> : <></>); | ||||
|     } | ||||
|   }, { | ||||
|     title: '补全', dataIndex: 'completion_tokens', render: (text, record, index) => { | ||||
|       return (parseInt(text) > 0 && (record.type === 0 || record.type === 2) ? <div> | ||||
|         {<span> {text} </span>} | ||||
|       </div> : <></>); | ||||
|     } | ||||
|   }, { | ||||
|     title: '花费', dataIndex: 'quota', render: (text, record, index) => { | ||||
|       return (record.type === 0 || record.type === 2 ? <div> | ||||
|         {renderQuota(text, 6)} | ||||
|       </div> : <></>); | ||||
|     } | ||||
|   }, { | ||||
|     title: '详情', dataIndex: 'content', render: (text, record, index) => { | ||||
|       return <Paragraph ellipsis={{ rows: 2, showTooltip: { type: 'popover', opts: { style: { width: 240 } } } }} | ||||
|         style={{ maxWidth: 240 }}> | ||||
|         {text} | ||||
|       </Paragraph>; | ||||
|     } | ||||
|   }]; | ||||
|  | ||||
|   const [logs, setLogs] = useState([]); | ||||
|   const [showStat, setShowStat] = useState(false); | ||||
|   const [loading, setLoading] = useState(false); | ||||
|   const [loadingStat, setLoadingStat] = useState(false); | ||||
|   const [activePage, setActivePage] = useState(1); | ||||
|   const [logCount, setLogCount] = useState(ITEMS_PER_PAGE); | ||||
|   const [pageSize, setPageSize] = useState(ITEMS_PER_PAGE); | ||||
|   const [searchKeyword, setSearchKeyword] = useState(''); | ||||
|   const [searching, setSearching] = useState(false); | ||||
|   const [logType, setLogType] = useState(0); | ||||
|   const isAdminUser = isAdmin(); | ||||
|   let now = new Date(); | ||||
|   // 初始化start_timestamp为前一天 | ||||
|   const [inputs, setInputs] = useState({ | ||||
|     username: '', | ||||
|     token_name: '', | ||||
|     model_name: '', | ||||
|     start_timestamp: timestamp2string(now.getTime() / 1000 - 86400), | ||||
|     end_timestamp: timestamp2string(now.getTime() / 1000 + 3600), | ||||
|     channel: '' | ||||
|   }); | ||||
|   const { username, token_name, model_name, start_timestamp, end_timestamp, channel } = inputs; | ||||
|  | ||||
|   const [stat, setStat] = useState({ | ||||
|     quota: 0, token: 0 | ||||
|   }); | ||||
|  | ||||
|   const handleInputChange = (value, name) => { | ||||
|     setInputs((inputs) => ({ ...inputs, [name]: value })); | ||||
|   }; | ||||
|  | ||||
|   const getLogSelfStat = async () => { | ||||
|     let localStartTimestamp = Date.parse(start_timestamp) / 1000; | ||||
|     let localEndTimestamp = Date.parse(end_timestamp) / 1000; | ||||
|     let res = await API.get(`/api/log/self/stat?type=${logType}&token_name=${token_name}&model_name=${model_name}&start_timestamp=${localStartTimestamp}&end_timestamp=${localEndTimestamp}`); | ||||
|     const { success, message, data } = res.data; | ||||
|     if (success) { | ||||
|       setStat(data); | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   const getLogStat = async () => { | ||||
|     let localStartTimestamp = Date.parse(start_timestamp) / 1000; | ||||
|     let localEndTimestamp = Date.parse(end_timestamp) / 1000; | ||||
|     let res = await API.get(`/api/log/stat?type=${logType}&username=${username}&token_name=${token_name}&model_name=${model_name}&start_timestamp=${localStartTimestamp}&end_timestamp=${localEndTimestamp}&channel=${channel}`); | ||||
|     const { success, message, data } = res.data; | ||||
|     if (success) { | ||||
|       setStat(data); | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   const handleEyeClick = async () => { | ||||
|     setLoadingStat(true); | ||||
|     if (isAdminUser) { | ||||
|       await getLogStat(); | ||||
|     } else { | ||||
|       await getLogSelfStat(); | ||||
|     } | ||||
|     setShowStat(true); | ||||
|     setLoadingStat(false); | ||||
|   }; | ||||
|  | ||||
|   const showUserInfo = async (userId) => { | ||||
|     if (!isAdminUser) { | ||||
|       return; | ||||
|     } | ||||
|     const res = await API.get(`/api/user/${userId}`); | ||||
|     const { success, message, data } = res.data; | ||||
|     if (success) { | ||||
|       Modal.info({ | ||||
|         title: '用户信息', content: <div style={{ padding: 12 }}> | ||||
|           <p>用户名: {data.username}</p> | ||||
|           <p>余额: {renderQuota(data.quota)}</p> | ||||
|           <p>已用额度:{renderQuota(data.used_quota)}</p> | ||||
|           <p>请求次数:{renderNumber(data.request_count)}</p> | ||||
|         </div>, centered: true | ||||
|       }); | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   const setLogsFormat = (logs) => { | ||||
|     for (let i = 0; i < logs.length; i++) { | ||||
|       logs[i].timestamp2string = timestamp2string(logs[i].created_at); | ||||
|       logs[i].key = '' + logs[i].id; | ||||
|     } | ||||
|     // data.key = '' + data.id | ||||
|     setLogs(logs); | ||||
|     setLogCount(logs.length + ITEMS_PER_PAGE); | ||||
|     // console.log(logCount); | ||||
|   }; | ||||
|  | ||||
|   const loadLogs = async (startIdx, pageSize, logType = 0) => { | ||||
|     setLoading(true); | ||||
|  | ||||
|     let url = ''; | ||||
|     let localStartTimestamp = Date.parse(start_timestamp) / 1000; | ||||
|     let localEndTimestamp = Date.parse(end_timestamp) / 1000; | ||||
|     if (isAdminUser) { | ||||
|       url = `/api/log/?p=${startIdx}&page_size=${pageSize}&type=${logType}&username=${username}&token_name=${token_name}&model_name=${model_name}&start_timestamp=${localStartTimestamp}&end_timestamp=${localEndTimestamp}&channel=${channel}`; | ||||
|     } else { | ||||
|       url = `/api/log/self/?p=${startIdx}&page_size=${pageSize}&type=${logType}&token_name=${token_name}&model_name=${model_name}&start_timestamp=${localStartTimestamp}&end_timestamp=${localEndTimestamp}`; | ||||
|     } | ||||
|     const res = await API.get(url); | ||||
|     const { success, message, data } = res.data; | ||||
|     if (success) { | ||||
|       if (startIdx === 0) { | ||||
|         setLogsFormat(data); | ||||
|       } else { | ||||
|         let newLogs = [...logs]; | ||||
|         newLogs.splice(startIdx * pageSize, data.length, ...data); | ||||
|         setLogsFormat(newLogs); | ||||
|       } | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
|     setLoading(false); | ||||
|   }; | ||||
|  | ||||
|   const pageData = logs.slice((activePage - 1) * pageSize, activePage * pageSize); | ||||
|  | ||||
|   const handlePageChange = page => { | ||||
|     setActivePage(page); | ||||
|     if (page === Math.ceil(logs.length / pageSize) + 1) { | ||||
|       // In this case we have to load more data and then append them. | ||||
|       loadLogs(page - 1, pageSize).then(r => { | ||||
|       }); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   const handlePageSizeChange = async (size) => { | ||||
|     localStorage.setItem('page-size', size + ''); | ||||
|     setPageSize(size); | ||||
|     setActivePage(1); | ||||
|     loadLogs(0, size) | ||||
|       .then() | ||||
|       .catch((reason) => { | ||||
|         showError(reason); | ||||
|       }); | ||||
|   }; | ||||
|  | ||||
|   const refresh = async (localLogType) => { | ||||
|     // setLoading(true); | ||||
|     setActivePage(1); | ||||
|     await loadLogs(0, pageSize, localLogType); | ||||
|   }; | ||||
|  | ||||
|   const copyText = async (text) => { | ||||
|     if (await copy(text)) { | ||||
|       showSuccess('已复制:' + text); | ||||
|     } else { | ||||
|       // setSearchKeyword(text); | ||||
|       Modal.error({ title: '无法复制到剪贴板,请手动复制', content: text }); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   useEffect(() => { | ||||
|     // console.log('default effect') | ||||
|     const localPageSize = parseInt(localStorage.getItem('page-size')) || ITEMS_PER_PAGE; | ||||
|     setPageSize(localPageSize); | ||||
|     loadLogs(0, localPageSize) | ||||
|       .then() | ||||
|       .catch((reason) => { | ||||
|         showError(reason); | ||||
|       }); | ||||
|   }, []); | ||||
|  | ||||
|   const searchLogs = async () => { | ||||
|     if (searchKeyword === '') { | ||||
|       // if keyword is blank, load files instead. | ||||
|       await loadLogs(0, pageSize); | ||||
|       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); | ||||
|   }; | ||||
|  | ||||
|   return (<> | ||||
|     <Layout> | ||||
|       <Header> | ||||
|         <Spin spinning={loadingStat}> | ||||
|           <h3>使用明细(总消耗额度: | ||||
|             <span onClick={handleEyeClick} style={{ | ||||
|               cursor: 'pointer', color: 'gray' | ||||
|             }}>{showStat ? renderQuota(stat.quota) : '点击查看'}</span> | ||||
|             ) | ||||
|           </h3> | ||||
|         </Spin> | ||||
|       </Header> | ||||
|       <Form layout="horizontal" style={{ marginTop: 10 }}> | ||||
|         <> | ||||
|           <Form.Input field="token_name" label="令牌名称" style={{ width: 176 }} value={token_name} | ||||
|             placeholder={'可选值'} name="token_name" | ||||
|             onChange={value => handleInputChange(value, 'token_name')} /> | ||||
|           <Form.Input field="model_name" label="模型名称" style={{ width: 176 }} value={model_name} | ||||
|             placeholder="可选值" | ||||
|             name="model_name" | ||||
|             onChange={value => handleInputChange(value, 'model_name')} /> | ||||
|           <Form.DatePicker field="start_timestamp" label="起始时间" style={{ width: 272 }} | ||||
|             initValue={start_timestamp} | ||||
|             value={start_timestamp} type="dateTime" | ||||
|             name="start_timestamp" | ||||
|             onChange={value => handleInputChange(value, 'start_timestamp')} /> | ||||
|           <Form.DatePicker field="end_timestamp" fluid label="结束时间" style={{ width: 272 }} | ||||
|             initValue={end_timestamp} | ||||
|             value={end_timestamp} type="dateTime" | ||||
|             name="end_timestamp" | ||||
|             onChange={value => handleInputChange(value, 'end_timestamp')} /> | ||||
|           {isAdminUser && <> | ||||
|             <Form.Input field="channel" label="渠道 ID" style={{ width: 176 }} value={channel} | ||||
|               placeholder="可选值" name="channel" | ||||
|               onChange={value => handleInputChange(value, 'channel')} /> | ||||
|             <Form.Input field="username" label="用户名称" style={{ width: 176 }} value={username} | ||||
|               placeholder={'可选值'} name="username" | ||||
|               onChange={value => handleInputChange(value, 'username')} /> | ||||
|           </>} | ||||
|           <Form.Section> | ||||
|             <Button label="查询" type="primary" htmlType="submit" className="btn-margin-right" | ||||
|               onClick={refresh} loading={loading}>查询</Button> | ||||
|           </Form.Section> | ||||
|         </> | ||||
|       </Form> | ||||
|       <Table style={{ marginTop: 5 }} columns={columns} dataSource={pageData} pagination={{ | ||||
|         currentPage: activePage, | ||||
|         pageSize: pageSize, | ||||
|         total: logCount, | ||||
|         pageSizeOpts: [10, 20, 50, 100], | ||||
|         showSizeChanger: true, | ||||
|         onPageSizeChange: (size) => { | ||||
|           handlePageSizeChange(size).then(); | ||||
|         }, | ||||
|         onPageChange: handlePageChange | ||||
|       }} /> | ||||
|       <Select defaultValue="0" style={{ width: 120 }} onChange={(value) => { | ||||
|         setLogType(parseInt(value)); | ||||
|         refresh(parseInt(value)).then(); | ||||
|       }}> | ||||
|         <Select.Option value="0">全部</Select.Option> | ||||
|         <Select.Option value="1">充值</Select.Option> | ||||
|         <Select.Option value="2">消费</Select.Option> | ||||
|         <Select.Option value="3">管理</Select.Option> | ||||
|         <Select.Option value="4">系统</Select.Option> | ||||
|       </Select> | ||||
|     </Layout> | ||||
|   </>); | ||||
| }; | ||||
|  | ||||
| export default LogsTable; | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user