mirror of
				https://github.com/songquanpeng/one-api.git
				synced 2025-10-27 03:43:43 +08:00 
			
		
		
		
	Compare commits
	
		
			7 Commits
		
	
	
		
			v0.4.1-alp
			...
			v0.4.2-alp
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 9d0bec83df | ||
|  | f97a9ce597 | ||
|  | 4339f45f74 | ||
|  | e398e0756b | ||
|  | 64db39320a | ||
|  | 0b4bf30908 | ||
|  | d29c273073 | 
							
								
								
									
										13
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										13
									
								
								README.md
									
									
									
									
									
								
							| @@ -44,7 +44,7 @@ _✨ All in one 的 OpenAI 接口,整合各种 API 访问方式,开箱即用 | ||||
|   <a href="https://iamazing.cn/page/reward">赞赏支持</a> | ||||
| </p> | ||||
|  | ||||
| > **Warning**:使用 Docker 拉取的最新镜像可能是 `alpha` 版本,如果追求稳定性请手动指定版本。 | ||||
| > **Note**:使用 Docker 拉取的最新镜像可能是 `alpha` 版本,如果追求稳定性请手动指定版本。 | ||||
|  | ||||
| > **Warning**:从 `v0.3` 版本升级到 `v0.4` 版本需要手动迁移数据库,请手动执行[数据库迁移脚本](./bin/migration_v0.3-v0.4.sql)。 | ||||
|  | ||||
| @@ -68,16 +68,17 @@ _✨ All in one 的 OpenAI 接口,整合各种 API 访问方式,开箱即用 | ||||
| 7. 支持**通道管理**,批量创建通道。 | ||||
| 8. 支持**用户分组**以及**渠道分组**。 | ||||
| 9. 支持渠道**设置模型列表**。 | ||||
| 10. 支持发布公告,设置充值链接,设置新用户初始额度。 | ||||
| 11. 支持丰富的**自定义**设置, | ||||
| 10. 支持**查看额度明细**。 | ||||
| 11. 支持发布公告,设置充值链接,设置新用户初始额度。 | ||||
| 12. 支持丰富的**自定义**设置, | ||||
|     1. 支持自定义系统名称,logo 以及页脚。 | ||||
|     2. 支持自定义首页和关于页面,可以选择使用 HTML & Markdown 代码进行自定义,或者使用一个单独的网页通过 iframe 嵌入。 | ||||
| 12. 支持通过系统访问令牌访问管理 API。 | ||||
| 13. 支持用户管理,支持**多种用户登录注册方式**: | ||||
| 13. 支持通过系统访问令牌访问管理 API。 | ||||
| 14. 支持用户管理,支持**多种用户登录注册方式**: | ||||
|     + 邮箱登录注册以及通过邮箱进行密码重置。 | ||||
|     + [GitHub 开放授权](https://github.com/settings/applications/new)。 | ||||
|     + 微信公众号授权(需要额外部署 [WeChat Server](https://github.com/songquanpeng/wechat-server))。 | ||||
| 14. 未来其他大模型开放 API 后,将第一时间支持,并将其封装成同样的 API 访问方式。 | ||||
| 15. 未来其他大模型开放 API 后,将第一时间支持,并将其封装成同样的 API 访问方式。 | ||||
|  | ||||
| ## 部署 | ||||
| ### 基于 Docker 进行部署 | ||||
|   | ||||
| @@ -26,8 +26,8 @@ var ModelRatio = map[string]float64{ | ||||
| 	"ada":                     10, | ||||
| 	"text-embedding-ada-002":  0.2, | ||||
| 	"text-search-ada-doc-001": 10, | ||||
| 	"text-moderation-stable":  10, | ||||
| 	"text-moderation-latest":  10, | ||||
| 	"text-moderation-stable":  0.1, | ||||
| 	"text-moderation-latest":  0.1, | ||||
| } | ||||
|  | ||||
| func ModelRatio2JSONString() string { | ||||
|   | ||||
| @@ -59,7 +59,7 @@ func testChannel(channel *model.Channel, request *ChatRequest) error { | ||||
| 		return err | ||||
| 	} | ||||
| 	if response.Usage.CompletionTokens == 0 { | ||||
| 		return errors.New(fmt.Sprintf("type %s, code %s, message %s", response.Error.Type, response.Error.Code, response.Error.Message)) | ||||
| 		return errors.New(fmt.Sprintf("type %s, code %v, message %s", response.Error.Type, response.Error.Code, response.Error.Message)) | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|   | ||||
| @@ -161,6 +161,24 @@ func init() { | ||||
| 			Root:       "text-ada-001", | ||||
| 			Parent:     nil, | ||||
| 		}, | ||||
| 		{ | ||||
| 			Id:         "text-moderation-latest", | ||||
| 			Object:     "model", | ||||
| 			Created:    1677649963, | ||||
| 			OwnedBy:    "openai", | ||||
| 			Permission: permission, | ||||
| 			Root:       "text-moderation-latest", | ||||
| 			Parent:     nil, | ||||
| 		}, | ||||
| 		{ | ||||
| 			Id:         "text-moderation-stable", | ||||
| 			Object:     "model", | ||||
| 			Created:    1677649963, | ||||
| 			OwnedBy:    "openai", | ||||
| 			Permission: permission, | ||||
| 			Root:       "text-moderation-stable", | ||||
| 			Parent:     nil, | ||||
| 		}, | ||||
| 	} | ||||
| 	openAIModelsMap = make(map[string]OpenAIModels) | ||||
| 	for _, model := range openAIModels { | ||||
|   | ||||
| @@ -24,6 +24,7 @@ const ( | ||||
| 	RelayModeChatCompletions | ||||
| 	RelayModeCompletions | ||||
| 	RelayModeEmbeddings | ||||
| 	RelayModeModeration | ||||
| ) | ||||
|  | ||||
| // https://platform.openai.com/docs/api-reference/chat | ||||
| @@ -37,6 +38,7 @@ type GeneralOpenAIRequest struct { | ||||
| 	Temperature float64   `json:"temperature"` | ||||
| 	TopP        float64   `json:"top_p"` | ||||
| 	N           int       `json:"n"` | ||||
| 	Input       string    `json:"input"` | ||||
| } | ||||
|  | ||||
| type ChatRequest struct { | ||||
| @@ -63,7 +65,7 @@ type OpenAIError struct { | ||||
| 	Message string `json:"message"` | ||||
| 	Type    string `json:"type"` | ||||
| 	Param   string `json:"param"` | ||||
| 	Code    string `json:"code"` | ||||
| 	Code    any    `json:"code"` | ||||
| } | ||||
|  | ||||
| type OpenAIErrorWithStatusCode struct { | ||||
| @@ -100,11 +102,13 @@ func Relay(c *gin.Context) { | ||||
| 		relayMode = RelayModeCompletions | ||||
| 	} else if strings.HasPrefix(c.Request.URL.Path, "/v1/embeddings") { | ||||
| 		relayMode = RelayModeEmbeddings | ||||
| 	} else if strings.HasPrefix(c.Request.URL.Path, "/v1/moderations") { | ||||
| 		relayMode = RelayModeModeration | ||||
| 	} | ||||
| 	err := relayHelper(c, relayMode) | ||||
| 	if err != nil { | ||||
| 		if err.StatusCode == http.StatusTooManyRequests { | ||||
| 			err.OpenAIError.Message = "负载已满,请稍后再试,或升级账户以提升服务质量。" | ||||
| 			err.OpenAIError.Message = "当前分组负载已饱和,请稍后再试,或升级账户以提升服务质量。" | ||||
| 		} | ||||
| 		c.JSON(err.StatusCode, gin.H{ | ||||
| 			"error": err.OpenAIError, | ||||
| @@ -143,6 +147,9 @@ func relayHelper(c *gin.Context, relayMode int) *OpenAIErrorWithStatusCode { | ||||
| 			return errorWrapper(err, "bind_request_body_failed", http.StatusBadRequest) | ||||
| 		} | ||||
| 	} | ||||
| 	if relayMode == RelayModeModeration && textRequest.Model == "" { | ||||
| 		textRequest.Model = "text-moderation-latest" | ||||
| 	} | ||||
| 	baseURL := common.ChannelBaseURLs[channelType] | ||||
| 	requestURL := c.Request.URL.String() | ||||
| 	if channelType == common.ChannelTypeCustom { | ||||
| @@ -180,6 +187,8 @@ func relayHelper(c *gin.Context, relayMode int) *OpenAIErrorWithStatusCode { | ||||
| 		promptTokens = countTokenMessages(textRequest.Messages, textRequest.Model) | ||||
| 	case RelayModeCompletions: | ||||
| 		promptTokens = countTokenText(textRequest.Prompt, textRequest.Model) | ||||
| 	case RelayModeModeration: | ||||
| 		promptTokens = countTokenText(textRequest.Input, textRequest.Model) | ||||
| 	} | ||||
| 	preConsumedTokens := common.PreConsumedQuota | ||||
| 	if textRequest.MaxTokens != 0 { | ||||
| @@ -239,6 +248,9 @@ func relayHelper(c *gin.Context, relayMode int) *OpenAIErrorWithStatusCode { | ||||
| 				quota = textResponse.Usage.PromptTokens + textResponse.Usage.CompletionTokens*completionRatio | ||||
| 			} | ||||
| 			quota = int(float64(quota) * ratio) | ||||
| 			if ratio != 0 && quota <= 0 { | ||||
| 				quota = 1 | ||||
| 			} | ||||
| 			quotaDelta := quota - preConsumedQuota | ||||
| 			err := model.PostConsumeTokenQuota(tokenId, quotaDelta) | ||||
| 			if err != nil { | ||||
|   | ||||
| @@ -2,6 +2,7 @@ package controller | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"github.com/gin-contrib/sessions" | ||||
| 	"github.com/gin-gonic/gin" | ||||
| 	"net/http" | ||||
| @@ -351,6 +352,9 @@ func UpdateUser(c *gin.Context) { | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	if originUser.Quota != updatedUser.Quota { | ||||
| 		model.RecordLog(originUser.Id, model.LogTypeManage, fmt.Sprintf("管理员将用户额度从 %d 点修改为 %d 点", originUser.Quota, updatedUser.Quota)) | ||||
| 	} | ||||
| 	c.JSON(http.StatusOK, gin.H{ | ||||
| 		"success": true, | ||||
| 		"message": "", | ||||
|   | ||||
| @@ -7,6 +7,7 @@ import ( | ||||
| 	"one-api/common" | ||||
| 	"one-api/model" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| type ModelRequest struct { | ||||
| @@ -64,6 +65,11 @@ func Distribute() func(c *gin.Context) { | ||||
| 				c.Abort() | ||||
| 				return | ||||
| 			} | ||||
| 			if strings.HasPrefix(c.Request.URL.Path, "/v1/moderations") { | ||||
| 				if modelRequest.Model == "" { | ||||
| 					modelRequest.Model = "text-moderation-stable" | ||||
| 				} | ||||
| 			} | ||||
| 			userId := c.GetInt("id") | ||||
| 			userGroup, _ := model.GetUserGroup(userId) | ||||
| 			channel, err = model.GetRandomSatisfiedChannel(userGroup, modelRequest.Model) | ||||
|   | ||||
							
								
								
									
										10
									
								
								model/log.go
									
									
									
									
									
								
							
							
						
						
									
										10
									
								
								model/log.go
									
									
									
									
									
								
							| @@ -17,6 +17,8 @@ const ( | ||||
| 	LogTypeUnknown = iota | ||||
| 	LogTypeTopup | ||||
| 	LogTypeConsume | ||||
| 	LogTypeManage | ||||
| 	LogTypeSystem | ||||
| ) | ||||
|  | ||||
| func RecordLog(userId int, logType int, content string) { | ||||
| @@ -33,7 +35,13 @@ func RecordLog(userId int, logType int, content string) { | ||||
| } | ||||
|  | ||||
| func GetAllLogs(logType int, startIdx int, num int) (logs []*Log, err error) { | ||||
| 	err = DB.Where("type = ?", logType).Order("id desc").Limit(num).Offset(startIdx).Find(&logs).Error | ||||
| 	var tx *gorm.DB | ||||
| 	if logType == LogTypeUnknown { | ||||
| 		tx = DB | ||||
| 	} else { | ||||
| 		tx = DB.Where("type = ?", logType) | ||||
| 	} | ||||
| 	err = tx.Order("id desc").Limit(num).Offset(startIdx).Find(&logs).Error | ||||
| 	return logs, err | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -2,6 +2,7 @@ package model | ||||
|  | ||||
| import ( | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"gorm.io/gorm" | ||||
| 	"one-api/common" | ||||
| 	"strings" | ||||
| @@ -73,8 +74,14 @@ func (user *User) Insert() error { | ||||
| 	} | ||||
| 	user.Quota = common.QuotaForNewUser | ||||
| 	user.AccessToken = common.GetUUID() | ||||
| 	err = DB.Create(user).Error | ||||
| 	return err | ||||
| 	result := DB.Create(user) | ||||
| 	if result.Error != nil { | ||||
| 		return result.Error | ||||
| 	} | ||||
| 	if common.QuotaForNewUser > 0 { | ||||
| 		RecordLog(user.Id, LogTypeSystem, fmt.Sprintf("新用户注册赠送 %d 点额度", common.QuotaForNewUser)) | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (user *User) Update(updatePassword bool) error { | ||||
|   | ||||
| @@ -37,6 +37,6 @@ func SetRelayRouter(router *gin.Engine) { | ||||
| 		relayV1Router.POST("/fine-tunes/:id/cancel", controller.RelayNotImplemented) | ||||
| 		relayV1Router.GET("/fine-tunes/:id/events", controller.RelayNotImplemented) | ||||
| 		relayV1Router.DELETE("/models/:model", controller.RelayNotImplemented) | ||||
| 		relayV1Router.POST("/moderations", controller.RelayNotImplemented) | ||||
| 		relayV1Router.POST("/moderations", controller.Relay) | ||||
| 	} | ||||
| } | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| import React, { useEffect, useState } from 'react'; | ||||
| import { Button, Label, Pagination, Table } from 'semantic-ui-react'; | ||||
| import { API, showError, timestamp2string } from '../helpers'; | ||||
| import { Button, Label, Pagination, Select, Table } from 'semantic-ui-react'; | ||||
| import { API, isAdmin, showError, timestamp2string } from '../helpers'; | ||||
|  | ||||
| import { ITEMS_PER_PAGE } from '../constants'; | ||||
|  | ||||
| @@ -12,12 +12,29 @@ function renderTimestamp(timestamp) { | ||||
|   ); | ||||
| } | ||||
|  | ||||
| const MODE_OPTIONS = [ | ||||
|   { key: 'all', text: '全部用户', value: 'all' }, | ||||
|   { key: 'self', text: '当前用户', value: 'self' }, | ||||
| ]; | ||||
|  | ||||
| const LOG_OPTIONS = [ | ||||
|   { key: '0', text: '全部', value: 0 }, | ||||
|   { key: '1', text: '充值', value: 1 }, | ||||
|   { key: '2', text: '消费', value: 2 }, | ||||
|   { key: '3', text: '管理', value: 3 }, | ||||
|   { key: '4', text: '系统', value: 4 } | ||||
| ]; | ||||
|  | ||||
| function renderType(type) { | ||||
|   switch (type) { | ||||
|     case 1: | ||||
|       return <Label basic color='green'> 充值 </Label>; | ||||
|     case 2: | ||||
|       return <Label basic color='olive'> 消费 </Label>; | ||||
|     case 3: | ||||
|       return <Label basic color='orange'> 管理 </Label>; | ||||
|     case 4: | ||||
|       return <Label basic color='purple'> 系统 </Label>; | ||||
|     default: | ||||
|       return <Label basic color='black'> 未知 </Label>; | ||||
|   } | ||||
| @@ -29,9 +46,16 @@ const LogsTable = () => { | ||||
|   const [activePage, setActivePage] = useState(1); | ||||
|   const [searchKeyword, setSearchKeyword] = useState(''); | ||||
|   const [searching, setSearching] = useState(false); | ||||
|   const [logType, setLogType] = useState(0); | ||||
|   const [mode, setMode] = useState('self'); // all, self | ||||
|   const showModePanel = isAdmin(); | ||||
|  | ||||
|   const loadLogs = async (startIdx) => { | ||||
|     const res = await API.get(`/api/log/self/?p=${startIdx}`); | ||||
|     let url = `/api/log/self/?p=${startIdx}&type=${logType}`; | ||||
|     if (mode === 'all') { | ||||
|       url = `/api/log/?p=${startIdx}&type=${logType}`; | ||||
|     } | ||||
|     const res = await API.get(url); | ||||
|     const { success, message, data } = res.data; | ||||
|     if (success) { | ||||
|       if (startIdx === 0) { | ||||
| @@ -70,6 +94,10 @@ const LogsTable = () => { | ||||
|       }); | ||||
|   }, []); | ||||
|  | ||||
|   useEffect(() => { | ||||
|     refresh().then(); | ||||
|   }, [mode, logType]); | ||||
|  | ||||
|   const searchLogs = async () => { | ||||
|     if (searchKeyword === '') { | ||||
|       // if keyword is blank, load files instead. | ||||
| @@ -121,6 +149,19 @@ const LogsTable = () => { | ||||
|             > | ||||
|               时间 | ||||
|             </Table.HeaderCell> | ||||
|             { | ||||
|               showModePanel && ( | ||||
|                 <Table.HeaderCell | ||||
|                   style={{ cursor: 'pointer' }} | ||||
|                   onClick={() => { | ||||
|                     sortLog('user_id'); | ||||
|                   }} | ||||
|                   width={1} | ||||
|                 > | ||||
|                   用户 | ||||
|                 </Table.HeaderCell> | ||||
|               ) | ||||
|             } | ||||
|             <Table.HeaderCell | ||||
|               style={{ cursor: 'pointer' }} | ||||
|               onClick={() => { | ||||
| @@ -135,7 +176,7 @@ const LogsTable = () => { | ||||
|               onClick={() => { | ||||
|                 sortLog('content'); | ||||
|               }} | ||||
|               width={11} | ||||
|               width={showModePanel ? 10 : 11} | ||||
|             > | ||||
|               详情 | ||||
|             </Table.HeaderCell> | ||||
| @@ -153,6 +194,11 @@ const LogsTable = () => { | ||||
|               return ( | ||||
|                 <Table.Row key={log.created_at}> | ||||
|                   <Table.Cell>{renderTimestamp(log.created_at)}</Table.Cell> | ||||
|                   { | ||||
|                     showModePanel && ( | ||||
|                       <Table.Cell><Label>{log.user_id}</Label></Table.Cell> | ||||
|                     ) | ||||
|                   } | ||||
|                   <Table.Cell>{renderType(log.type)}</Table.Cell> | ||||
|                   <Table.Cell>{log.content}</Table.Cell> | ||||
|                 </Table.Row> | ||||
| @@ -162,7 +208,31 @@ const LogsTable = () => { | ||||
|  | ||||
|         <Table.Footer> | ||||
|           <Table.Row> | ||||
|             <Table.HeaderCell colSpan='4'> | ||||
|             <Table.HeaderCell colSpan={showModePanel ? '5' : '4'}> | ||||
|               { | ||||
|                 showModePanel && ( | ||||
|                   <Select | ||||
|                     placeholder='选择模式' | ||||
|                     options={MODE_OPTIONS} | ||||
|                     style={{ marginRight: '8px' }} | ||||
|                     name='mode' | ||||
|                     value={mode} | ||||
|                     onChange={(e, { name, value }) => { | ||||
|                       setMode(value); | ||||
|                     }} | ||||
|                   /> | ||||
|                 ) | ||||
|               } | ||||
|               <Select | ||||
|                 placeholder='选择明细分类' | ||||
|                 options={LOG_OPTIONS} | ||||
|                 style={{ marginRight: '8px' }} | ||||
|                 name='logType' | ||||
|                 value={logType} | ||||
|                 onChange={(e, { name, value }) => { | ||||
|                   setLogType(value); | ||||
|                 }} | ||||
|               /> | ||||
|               <Button size='small' onClick={refresh} loading={loading}>刷新</Button> | ||||
|               <Pagination | ||||
|                 floated='right' | ||||
|   | ||||
		Reference in New Issue
	
	Block a user