mirror of
				https://github.com/songquanpeng/one-api.git
				synced 2025-10-23 09:53:42 +08:00 
			
		
		
		
	Compare commits
	
		
			65 Commits
		
	
	
		
			v0.1.2
			...
			v0.2.5-alp
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 246b981e23 | ||
|  | 2edd52e851 | ||
|  | e123c66bc7 | ||
|  | 9edc82bde0 | ||
|  | d84c2f5c70 | ||
|  | 46e77389a4 | ||
|  | f5f4e6fbc6 | ||
|  | dc4a6cb711 | ||
|  | 5798fdac50 | ||
|  | 3710688efd | ||
|  | 83e86b9f8a | ||
|  | 74c1ba7cbc | ||
|  | 73aa53f536 | ||
|  | da9ccb528d | ||
|  | 44729da277 | ||
|  | 7a3378b4b7 | ||
|  | fd19d7d246 | ||
|  | 5c694a1503 | ||
|  | 9edc54ca69 | ||
|  | e6af636fa0 | ||
|  | 6e1ef75009 | ||
|  | d9db16e999 | ||
|  | 241ade2fae | ||
|  | 80065de8a3 | ||
|  | 16f53b5afb | ||
|  | 3071300c0c | ||
|  | 8b056bf408 | ||
|  | e5640857b1 | ||
|  | 331177d97e | ||
|  | 4fed003f1a | ||
|  | a1ea1bf696 | ||
|  | 7c66fc6c21 | ||
|  | d93cb8f645 | ||
|  | b08cd7e104 | ||
|  | aea6c859e7 | ||
|  | 480e789cd8 | ||
|  | 23ec541ba6 | ||
|  | 053bb85a1c | ||
|  | 601fa5cea8 | ||
|  | 7a5057f02d | ||
|  | c76027a210 | ||
|  | f97c2b4c22 | ||
|  | 54b1e4adef | ||
|  | 9272884381 | ||
|  | 195e94a75d | ||
|  | 5bfc224669 | ||
|  | fd149c242f | ||
|  | b9cc5dfa3f | ||
|  | 8c305dc1bc | ||
|  | f62a671fbe | ||
|  | 9e2f2383b9 | ||
|  | e7a809b082 | ||
|  | 4f8cbd643d | ||
|  | 1dd92a3f92 | ||
|  | 34a3329f5f | ||
|  | 4fb07b6d6d | ||
|  | 8be7c9ae80 | ||
|  | 4e8dc8d0cf | ||
|  | 1e46b9d135 | ||
|  | f16a2a5645 | ||
|  | 03491029f2 | ||
|  | faf84d833d | ||
|  | 109736cc05 | ||
|  | eb8f43acb5 | ||
|  | 05dd7dfd2a | 
							
								
								
									
										1
									
								
								.github/FUNDING.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								.github/FUNDING.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | ||||
| custom: ['https://iamazing.cn/page/reward'] | ||||
							
								
								
									
										29
									
								
								.github/workflows/github-pages.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										29
									
								
								.github/workflows/github-pages.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,29 +0,0 @@ | ||||
| name: Build GitHub Pages | ||||
| on: | ||||
|   workflow_dispatch: | ||||
|     inputs: | ||||
|       name: | ||||
|         description: 'Reason' | ||||
|         required: false | ||||
| jobs: | ||||
|   build-and-deploy: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Checkout 🛎️ | ||||
|         uses: actions/checkout@v2 # If you're using actions/checkout@v2 you must set persist-credentials to false in most cases for the deployment to work correctly. | ||||
|         with: | ||||
|           persist-credentials: false | ||||
|       - name: Install and Build 🔧 # This example project is built using npm and outputs the result to the 'build' folder. Replace with the commands required to build your project, or remove this step entirely if your site is pre-built. | ||||
|         env: | ||||
|           CI: "" | ||||
|         run: | | ||||
|           cd web | ||||
|           npm install | ||||
|           npm run build | ||||
|  | ||||
|       - name: Deploy 🚀 | ||||
|         uses: JamesIves/github-pages-deploy-action@releases/v3 | ||||
|         with: | ||||
|           ACCESS_TOKEN: ${{ secrets.ACCESS_TOKEN }} | ||||
|           BRANCH: gh-pages # The branch the action should deploy to. | ||||
|           FOLDER: web/build # The folder the action should deploy. | ||||
							
								
								
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -3,4 +3,5 @@ | ||||
| upload | ||||
| *.exe | ||||
| *.db | ||||
| build | ||||
| build | ||||
| *.db-journal | ||||
| @@ -24,7 +24,7 @@ RUN apk update \ | ||||
|     && apk upgrade \ | ||||
|     && apk add --no-cache ca-certificates tzdata \ | ||||
|     && update-ca-certificates 2>/dev/null || true | ||||
| ENV PORT=3000 | ||||
|  | ||||
| COPY --from=builder2 /build/one-api / | ||||
| EXPOSE 3000 | ||||
| WORKDIR /data | ||||
|   | ||||
							
								
								
									
										2
									
								
								LICENSE
									
									
									
									
									
								
							
							
						
						
									
										2
									
								
								LICENSE
									
									
									
									
									
								
							| @@ -1,6 +1,6 @@ | ||||
| MIT License | ||||
|  | ||||
| Copyright (c) 2022 JustSong | ||||
| Copyright (c) 2023 JustSong | ||||
|  | ||||
| Permission is hereby granted, free of charge, to any person obtaining a copy | ||||
| of this software and associated documentation files (the "Software"), to deal | ||||
|   | ||||
							
								
								
									
										30
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										30
									
								
								README.md
									
									
									
									
									
								
							| @@ -42,24 +42,28 @@ _✨ All in one 的 OpenAI 接口,整合各种 API 访问方式,开箱即用 | ||||
|  | ||||
| ## 功能 | ||||
| 1. 支持多种 API 访问渠道,欢迎 PR 或提 issue 添加更多渠道: | ||||
|    + [x] One API 服务端中继 | ||||
|    + [x] OpenAI 官方通道 | ||||
|    + [x] [API2D](https://api2d.com/r/197971) | ||||
|    + [ ] Azure OpenAI API | ||||
|    + [x] Azure OpenAI API | ||||
|    + [x] [CloseAI](https://console.openai-asia.com) | ||||
|    + [x] [OpenAI-SB](https://openai-sb.com) | ||||
|    + [x] [OpenAI Max](https://openaimax.com) | ||||
|    + [x] [OhMyGPT](https://www.ohmygpt.com) | ||||
|    + [x] 自定义渠道 | ||||
|    + [x] 自定义渠道:例如使用自行搭建的 OpenAI 代理 | ||||
| 2. 支持通过负载均衡的方式访问多个渠道。 | ||||
| 3. 支持单个访问渠道设置多个 API Key,利用起来你的多个 API Key。 | ||||
| 4. 支持设置令牌的过期时间和使用次数。 | ||||
| 5. 支持 HTTP SSE。 | ||||
| 6. 多种用户登录注册方式: | ||||
|    + 邮箱登录注册以及通过邮箱进行密码重置。 | ||||
|    + [GitHub 开放授权](https://github.com/settings/applications/new)。 | ||||
|    + 微信公众号授权(需要额外部署 [WeChat Server](https://github.com/songquanpeng/wechat-server))。 | ||||
| 7. 支持用户管理。 | ||||
| 8. 未来其他大模型开放 API 后,将第一时间支持,并将其封装成同样的 API 访问方式。 | ||||
| 4. 支持 HTTP SSE,可以通过流式传输实现打字机效果。 | ||||
| 5. 支持设置令牌的过期时间和使用次数。 | ||||
| 6. 支持批量生成和导出兑换码,可使用兑换码为令牌进行充值。 | ||||
| 7. 支持为新用户设置初始配额。 | ||||
| 8. 支持自定义首页,发布公告,自定义关于页面,设置充值链接,自定义页脚。 | ||||
| 9. 支持通过系统访问令牌访问管理 API。 | ||||
| 10. 多种用户登录注册方式: | ||||
|     + 邮箱登录注册以及通过邮箱进行密码重置。 | ||||
|     + [GitHub 开放授权](https://github.com/settings/applications/new)。 | ||||
|     + 微信公众号授权(需要额外部署 [WeChat Server](https://github.com/songquanpeng/wechat-server))。 | ||||
| 11. 支持用户管理。 | ||||
| 12. 未来其他大模型开放 API 后,将第一时间支持,并将其封装成同样的 API 访问方式。 | ||||
|  | ||||
| ## 部署 | ||||
| ### 基于 Docker 进行部署 | ||||
| @@ -82,10 +86,13 @@ server{ | ||||
|           proxy_set_header X-Forwarded-For $remote_addr; | ||||
|           proxy_cache_bypass $http_upgrade; | ||||
|           proxy_set_header Accept-Encoding gzip; | ||||
|           proxy_buffering off;  # 重要:关闭代理缓冲 | ||||
|    } | ||||
| } | ||||
| ``` | ||||
|  | ||||
| 注意,为了 SSE 正常工作,需要关闭 Nginx 的代理缓冲。 | ||||
|  | ||||
| 之后使用 Let's Encrypt 的 certbot 配置 HTTPS: | ||||
| ```bash | ||||
| # Ubuntu 安装 certbot: | ||||
| @@ -135,6 +142,7 @@ sudo service nginx restart | ||||
| 之后就可以使用你的令牌访问 One API 了,使用方式与 [OpenAI API](https://platform.openai.com/docs/api-reference/introduction) 一致。 | ||||
|  | ||||
| 可以通过在令牌后面添加渠道 ID 的方式指定使用哪一个渠道处理本次请求,例如:`Authorization: Bearer ONE_API_KEY-CHANNEL_ID`。 | ||||
| 注意,需要是管理员用户创建的令牌才能指定渠道 ID。 | ||||
|  | ||||
| 不加的话将会使用负载均衡的方式使用多个渠道。 | ||||
|  | ||||
|   | ||||
| @@ -11,6 +11,7 @@ var Version = "v0.0.0"            // this hard coding will be replaced automatic | ||||
| var SystemName = "One API" | ||||
| var ServerAddress = "http://localhost:3000" | ||||
| var Footer = "" | ||||
| var TopUpLink = "" | ||||
|  | ||||
| var UsingSQLite = false | ||||
|  | ||||
| @@ -33,7 +34,9 @@ var TurnstileCheckEnabled = false | ||||
| var RegisterEnabled = true | ||||
|  | ||||
| var SMTPServer = "" | ||||
| var SMTPPort = 587 | ||||
| var SMTPAccount = "" | ||||
| var SMTPFrom = "" | ||||
| var SMTPToken = "" | ||||
|  | ||||
| var GitHubClientId = "" | ||||
| @@ -46,6 +49,8 @@ var WeChatAccountQRCodeImageURL = "" | ||||
| var TurnstileSiteKey = "" | ||||
| var TurnstileSecretKey = "" | ||||
|  | ||||
| var QuotaForNewUser = 100 | ||||
|  | ||||
| const ( | ||||
| 	RoleGuestUser  = 0 | ||||
| 	RoleCommonUser = 1 | ||||
| @@ -63,7 +68,7 @@ var ( | ||||
| // All duration's unit is seconds | ||||
| // Shouldn't larger then RateLimitKeyExpirationDuration | ||||
| var ( | ||||
| 	GlobalApiRateLimitNum            = 60000 // TODO: temporary set to 60000 | ||||
| 	GlobalApiRateLimitNum            = 180 | ||||
| 	GlobalApiRateLimitDuration int64 = 3 * 60 | ||||
|  | ||||
| 	GlobalWebRateLimitNum            = 60 | ||||
| @@ -93,6 +98,12 @@ const ( | ||||
| 	TokenStatusExhausted = 4 | ||||
| ) | ||||
|  | ||||
| const ( | ||||
| 	RedemptionCodeStatusEnabled  = 1 // don't use 0, 0 is the default value! | ||||
| 	RedemptionCodeStatusDisabled = 2 // also don't use 0 | ||||
| 	RedemptionCodeStatusUsed     = 3 // also don't use 0 | ||||
| ) | ||||
|  | ||||
| const ( | ||||
| 	ChannelStatusUnknown  = 0 | ||||
| 	ChannelStatusEnabled  = 1 // don't use 0, 0 is the default value! | ||||
|   | ||||
| @@ -1,14 +1,67 @@ | ||||
| package common | ||||
|  | ||||
| import "gopkg.in/gomail.v2" | ||||
| import ( | ||||
| 	"crypto/tls" | ||||
| 	"encoding/base64" | ||||
| 	"fmt" | ||||
| 	"net/smtp" | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| func SendEmail(subject string, receiver string, content string) error { | ||||
| 	m := gomail.NewMessage() | ||||
| 	m.SetHeader("From", SMTPAccount) | ||||
| 	m.SetHeader("To", receiver) | ||||
| 	m.SetHeader("Subject", subject) | ||||
| 	m.SetBody("text/html", content) | ||||
| 	d := gomail.NewDialer(SMTPServer, 587, SMTPAccount, SMTPToken) | ||||
| 	err := d.DialAndSend(m) | ||||
| 	if SMTPFrom == "" { // for compatibility | ||||
| 		SMTPFrom = SMTPAccount | ||||
| 	} | ||||
| 	encodedSubject := fmt.Sprintf("=?UTF-8?B?%s?=", base64.StdEncoding.EncodeToString([]byte(subject))) | ||||
| 	mail := []byte(fmt.Sprintf("To: %s\r\n"+ | ||||
| 		"From: %s<%s>\r\n"+ | ||||
| 		"Subject: %s\r\n"+ | ||||
| 		"Content-Type: text/html; charset=UTF-8\r\n\r\n%s\r\n", | ||||
| 		receiver, SystemName, SMTPFrom, encodedSubject, content)) | ||||
| 	auth := smtp.PlainAuth("", SMTPAccount, SMTPToken, SMTPServer) | ||||
| 	addr := fmt.Sprintf("%s:%d", SMTPServer, SMTPPort) | ||||
| 	to := strings.Split(receiver, ";") | ||||
| 	var err error | ||||
| 	if SMTPPort == 465 { | ||||
| 		tlsConfig := &tls.Config{ | ||||
| 			InsecureSkipVerify: true, | ||||
| 			ServerName:         SMTPServer, | ||||
| 		} | ||||
| 		conn, err := tls.Dial("tcp", fmt.Sprintf("%s:%d", SMTPServer, SMTPPort), tlsConfig) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		client, err := smtp.NewClient(conn, SMTPServer) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		defer client.Close() | ||||
| 		if err = client.Auth(auth); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		if err = client.Mail(SMTPFrom); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		receiverEmails := strings.Split(receiver, ";") | ||||
| 		for _, receiver := range receiverEmails { | ||||
| 			if err = client.Rcpt(receiver); err != nil { | ||||
| 				return err | ||||
| 			} | ||||
| 		} | ||||
| 		w, err := client.Data() | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		_, err = w.Write(mail) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		err = w.Close() | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 	} else { | ||||
| 		err = smtp.SendMail(addr, auth, SMTPAccount, to, mail) | ||||
| 	} | ||||
| 	return err | ||||
| } | ||||
|   | ||||
							
								
								
									
										52
									
								
								common/model-ratio.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										52
									
								
								common/model-ratio.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,52 @@ | ||||
| package common | ||||
|  | ||||
| import "encoding/json" | ||||
|  | ||||
| // https://platform.openai.com/docs/models/model-endpoint-compatibility | ||||
| // https://openai.com/pricing | ||||
| // TODO: when a new api is enabled, check the pricing here | ||||
| var ModelRatio = map[string]float64{ | ||||
| 	"gpt-4":                   15, | ||||
| 	"gpt-4-0314":              15, | ||||
| 	"gpt-4-32k":               30, | ||||
| 	"gpt-4-32k-0314":          30, | ||||
| 	"gpt-3.5-turbo":           1, | ||||
| 	"gpt-3.5-turbo-0301":      1, | ||||
| 	"text-ada-001":            0.2, | ||||
| 	"text-babbage-001":        0.25, | ||||
| 	"text-curie-001":          1, | ||||
| 	"text-davinci-002":        10, | ||||
| 	"text-davinci-003":        10, | ||||
| 	"text-davinci-edit-001":   10, | ||||
| 	"code-davinci-edit-001":   10, | ||||
| 	"whisper-1":               10, | ||||
| 	"davinci":                 10, | ||||
| 	"curie":                   10, | ||||
| 	"babbage":                 10, | ||||
| 	"ada":                     10, | ||||
| 	"text-embedding-ada-002":  0.2, | ||||
| 	"text-search-ada-doc-001": 10, | ||||
| 	"text-moderation-stable":  10, | ||||
| 	"text-moderation-latest":  10, | ||||
| } | ||||
|  | ||||
| func ModelRatio2JSONString() string { | ||||
| 	jsonBytes, err := json.Marshal(ModelRatio) | ||||
| 	if err != nil { | ||||
| 		SysError("Error marshalling model ratio: " + err.Error()) | ||||
| 	} | ||||
| 	return string(jsonBytes) | ||||
| } | ||||
|  | ||||
| func UpdateModelRatioByJSONString(jsonStr string) error { | ||||
| 	return json.Unmarshal([]byte(jsonStr), &ModelRatio) | ||||
| } | ||||
|  | ||||
| func GetModelRatio(name string) float64 { | ||||
| 	ratio, ok := ModelRatio[name] | ||||
| 	if !ok { | ||||
| 		SysError("Model ratio not found: " + name) | ||||
| 		return 1 | ||||
| 	} | ||||
| 	return ratio | ||||
| } | ||||
| @@ -6,6 +6,7 @@ import ( | ||||
| 	"one-api/common" | ||||
| 	"one-api/model" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| func GetAllChannels(c *gin.Context) { | ||||
| @@ -84,7 +85,17 @@ func AddChannel(c *gin.Context) { | ||||
| 	} | ||||
| 	channel.CreatedTime = common.GetTimestamp() | ||||
| 	channel.AccessedTime = common.GetTimestamp() | ||||
| 	err = channel.Insert() | ||||
| 	keys := strings.Split(channel.Key, "\n") | ||||
| 	channels := make([]model.Channel, 0) | ||||
| 	for _, key := range keys { | ||||
| 		if key == "" { | ||||
| 			continue | ||||
| 		} | ||||
| 		localChannel := channel | ||||
| 		localChannel.Key = key | ||||
| 		channels = append(channels, localChannel) | ||||
| 	} | ||||
| 	err = model.BatchInsertChannels(channels) | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
|   | ||||
| @@ -26,6 +26,7 @@ func GetStatus(c *gin.Context) { | ||||
| 			"server_address":     common.ServerAddress, | ||||
| 			"turnstile_check":    common.TurnstileCheckEnabled, | ||||
| 			"turnstile_site_key": common.TurnstileSiteKey, | ||||
| 			"top_up_link":        common.TopUpLink, | ||||
| 		}, | ||||
| 	}) | ||||
| 	return | ||||
| @@ -53,6 +54,17 @@ func GetAbout(c *gin.Context) { | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func GetHomePageContent(c *gin.Context) { | ||||
| 	common.OptionMapRWMutex.RLock() | ||||
| 	defer common.OptionMapRWMutex.RUnlock() | ||||
| 	c.JSON(http.StatusOK, gin.H{ | ||||
| 		"success": true, | ||||
| 		"message": "", | ||||
| 		"data":    common.OptionMap["HomePageContent"], | ||||
| 	}) | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func SendEmailVerification(c *gin.Context) { | ||||
| 	email := c.Query("email") | ||||
| 	if err := common.Validate.Var(email, "required,email"); err != nil { | ||||
|   | ||||
							
								
								
									
										192
									
								
								controller/redemption.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										192
									
								
								controller/redemption.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,192 @@ | ||||
| package controller | ||||
|  | ||||
| import ( | ||||
| 	"github.com/gin-gonic/gin" | ||||
| 	"net/http" | ||||
| 	"one-api/common" | ||||
| 	"one-api/model" | ||||
| 	"strconv" | ||||
| ) | ||||
|  | ||||
| func GetAllRedemptions(c *gin.Context) { | ||||
| 	p, _ := strconv.Atoi(c.Query("p")) | ||||
| 	if p < 0 { | ||||
| 		p = 0 | ||||
| 	} | ||||
| 	redemptions, err := model.GetAllRedemptions(p*common.ItemsPerPage, common.ItemsPerPage) | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	c.JSON(http.StatusOK, gin.H{ | ||||
| 		"success": true, | ||||
| 		"message": "", | ||||
| 		"data":    redemptions, | ||||
| 	}) | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func SearchRedemptions(c *gin.Context) { | ||||
| 	keyword := c.Query("keyword") | ||||
| 	redemptions, err := model.SearchRedemptions(keyword) | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	c.JSON(http.StatusOK, gin.H{ | ||||
| 		"success": true, | ||||
| 		"message": "", | ||||
| 		"data":    redemptions, | ||||
| 	}) | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func GetRedemption(c *gin.Context) { | ||||
| 	id, err := strconv.Atoi(c.Param("id")) | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	redemption, err := model.GetRedemptionById(id) | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	c.JSON(http.StatusOK, gin.H{ | ||||
| 		"success": true, | ||||
| 		"message": "", | ||||
| 		"data":    redemption, | ||||
| 	}) | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func AddRedemption(c *gin.Context) { | ||||
| 	redemption := model.Redemption{} | ||||
| 	err := c.ShouldBindJSON(&redemption) | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	if len(redemption.Name) == 0 || len(redemption.Name) > 20 { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": "兑换码名称长度必须在1-20之间", | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	if redemption.Count <= 0 { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": "兑换码个数必须大于0", | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	if redemption.Count > 100 { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": "一次兑换码批量生成的个数不能大于 100", | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	var keys []string | ||||
| 	for i := 0; i < redemption.Count; i++ { | ||||
| 		key := common.GetUUID() | ||||
| 		cleanRedemption := model.Redemption{ | ||||
| 			UserId:      c.GetInt("id"), | ||||
| 			Name:        redemption.Name, | ||||
| 			Key:         key, | ||||
| 			CreatedTime: common.GetTimestamp(), | ||||
| 			Quota:       redemption.Quota, | ||||
| 		} | ||||
| 		err = cleanRedemption.Insert() | ||||
| 		if err != nil { | ||||
| 			c.JSON(http.StatusOK, gin.H{ | ||||
| 				"success": false, | ||||
| 				"message": err.Error(), | ||||
| 				"data":    keys, | ||||
| 			}) | ||||
| 			return | ||||
| 		} | ||||
| 		keys = append(keys, key) | ||||
| 	} | ||||
| 	c.JSON(http.StatusOK, gin.H{ | ||||
| 		"success": true, | ||||
| 		"message": "", | ||||
| 		"data":    keys, | ||||
| 	}) | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func DeleteRedemption(c *gin.Context) { | ||||
| 	id, _ := strconv.Atoi(c.Param("id")) | ||||
| 	err := model.DeleteRedemptionById(id) | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	c.JSON(http.StatusOK, gin.H{ | ||||
| 		"success": true, | ||||
| 		"message": "", | ||||
| 	}) | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func UpdateRedemption(c *gin.Context) { | ||||
| 	statusOnly := c.Query("status_only") | ||||
| 	redemption := model.Redemption{} | ||||
| 	err := c.ShouldBindJSON(&redemption) | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	cleanRedemption, err := model.GetRedemptionById(redemption.Id) | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	if statusOnly != "" { | ||||
| 		cleanRedemption.Status = redemption.Status | ||||
| 	} else { | ||||
| 		// If you add more fields, please also update redemption.Update() | ||||
| 		cleanRedemption.Name = redemption.Name | ||||
| 		cleanRedemption.Quota = redemption.Quota | ||||
| 	} | ||||
| 	err = cleanRedemption.Update() | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	c.JSON(http.StatusOK, gin.H{ | ||||
| 		"success": true, | ||||
| 		"message": "", | ||||
| 		"data":    cleanRedemption, | ||||
| 	}) | ||||
| 	return | ||||
| } | ||||
| @@ -2,21 +2,58 @@ package controller | ||||
|  | ||||
| import ( | ||||
| 	"bufio" | ||||
| 	"bytes" | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"github.com/gin-gonic/gin" | ||||
| 	"github.com/pkoukk/tiktoken-go" | ||||
| 	"io" | ||||
| 	"net/http" | ||||
| 	"one-api/common" | ||||
| 	"one-api/model" | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| type Message struct { | ||||
| 	Role    string `json:"role"` | ||||
| 	Content string `json:"content"` | ||||
| } | ||||
|  | ||||
| type TextRequest struct { | ||||
| 	Model    string    `json:"model"` | ||||
| 	Messages []Message `json:"messages"` | ||||
| 	Prompt   string    `json:"prompt"` | ||||
| 	//Stream   bool      `json:"stream"` | ||||
| } | ||||
|  | ||||
| type Usage struct { | ||||
| 	PromptTokens     int `json:"prompt_tokens"` | ||||
| 	CompletionTokens int `json:"completion_tokens"` | ||||
| 	TotalTokens      int `json:"total_tokens"` | ||||
| } | ||||
|  | ||||
| type TextResponse struct { | ||||
| 	Usage `json:"usage"` | ||||
| } | ||||
|  | ||||
| type StreamResponse struct { | ||||
| 	Choices []struct { | ||||
| 		Delta struct { | ||||
| 			Content string `json:"content"` | ||||
| 		} `json:"delta"` | ||||
| 		FinishReason string `json:"finish_reason"` | ||||
| 	} `json:"choices"` | ||||
| } | ||||
|  | ||||
| var tokenEncoder, _ = tiktoken.GetEncoding("cl100k_base") | ||||
|  | ||||
| func countToken(text string) int { | ||||
| 	token := tokenEncoder.Encode(text, nil, nil) | ||||
| 	return len(token) | ||||
| } | ||||
|  | ||||
| func Relay(c *gin.Context) { | ||||
| 	channelType := c.GetInt("channel") | ||||
| 	baseURL := common.ChannelBaseURLs[channelType] | ||||
| 	if channelType == common.ChannelTypeCustom { | ||||
| 		baseURL = c.GetString("base_url") | ||||
| 	} | ||||
| 	req, err := http.NewRequest(c.Request.Method, fmt.Sprintf("%s%s", baseURL, c.Request.URL.String()), c.Request.Body) | ||||
| 	err := relayHelper(c) | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"error": gin.H{ | ||||
| @@ -24,30 +61,109 @@ func Relay(c *gin.Context) { | ||||
| 				"type":    "one_api_error", | ||||
| 			}, | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	//req.Header = c.Request.Header.Clone() | ||||
| 	// Fix HTTP Decompression failed | ||||
| 	// https://github.com/stoplightio/prism/issues/1064#issuecomment-824682360 | ||||
| 	//req.Header.Del("Accept-Encoding") | ||||
| 	req.Header.Set("Authorization", c.Request.Header.Get("Authorization")) | ||||
| } | ||||
|  | ||||
| func relayHelper(c *gin.Context) error { | ||||
| 	channelType := c.GetInt("channel") | ||||
| 	tokenId := c.GetInt("token_id") | ||||
| 	consumeQuota := c.GetBool("consume_quota") | ||||
| 	var textRequest TextRequest | ||||
| 	if consumeQuota || channelType == common.ChannelTypeAzure { | ||||
| 		requestBody, err := io.ReadAll(c.Request.Body) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		err = c.Request.Body.Close() | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		err = json.Unmarshal(requestBody, &textRequest) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		// Reset request body | ||||
| 		c.Request.Body = io.NopCloser(bytes.NewBuffer(requestBody)) | ||||
| 	} | ||||
| 	baseURL := common.ChannelBaseURLs[channelType] | ||||
| 	requestURL := c.Request.URL.String() | ||||
| 	if channelType == common.ChannelTypeCustom { | ||||
| 		baseURL = c.GetString("base_url") | ||||
| 	} | ||||
| 	fullRequestURL := fmt.Sprintf("%s%s", baseURL, requestURL) | ||||
| 	if channelType == common.ChannelTypeAzure { | ||||
| 		// https://learn.microsoft.com/en-us/azure/cognitive-services/openai/chatgpt-quickstart?pivots=rest-api&tabs=command-line#rest-api | ||||
| 		query := c.Request.URL.Query() | ||||
| 		apiVersion := query.Get("api-version") | ||||
| 		if apiVersion == "" { | ||||
| 			apiVersion = c.GetString("api_version") | ||||
| 		} | ||||
| 		requestURL := strings.Split(requestURL, "?")[0] | ||||
| 		requestURL = fmt.Sprintf("%s?api-version=%s", requestURL, apiVersion) | ||||
| 		baseURL = c.GetString("base_url") | ||||
| 		task := strings.TrimPrefix(requestURL, "/v1/") | ||||
| 		model_ := textRequest.Model | ||||
| 		model_ = strings.Replace(model_, ".", "", -1) | ||||
| 		fullRequestURL = fmt.Sprintf("%s/openai/deployments/%s/%s", baseURL, model_, task) | ||||
| 	} | ||||
| 	req, err := http.NewRequest(c.Request.Method, fullRequestURL, c.Request.Body) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	if channelType == common.ChannelTypeAzure { | ||||
| 		key := c.Request.Header.Get("Authorization") | ||||
| 		key = strings.TrimPrefix(key, "Bearer ") | ||||
| 		req.Header.Set("api-key", key) | ||||
| 	} else { | ||||
| 		req.Header.Set("Authorization", c.Request.Header.Get("Authorization")) | ||||
| 	} | ||||
| 	req.Header.Set("Content-Type", c.Request.Header.Get("Content-Type")) | ||||
| 	req.Header.Set("Accept", c.Request.Header.Get("Accept")) | ||||
| 	req.Header.Set("Connection", c.Request.Header.Get("Connection")) | ||||
| 	client := &http.Client{} | ||||
|  | ||||
| 	resp, err := client.Do(req) | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"error": gin.H{ | ||||
| 				"message": err.Error(), | ||||
| 				"type":    "one_api_error", | ||||
| 			}, | ||||
| 		}) | ||||
| 		return | ||||
| 		return err | ||||
| 	} | ||||
| 	defer resp.Body.Close() | ||||
| 	err = req.Body.Close() | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	err = c.Request.Body.Close() | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	var textResponse TextResponse | ||||
| 	isStream := resp.Header.Get("Content-Type") == "text/event-stream" | ||||
| 	var streamResponseText string | ||||
|  | ||||
| 	defer func() { | ||||
| 		if consumeQuota { | ||||
| 			quota := 0 | ||||
| 			usingGPT4 := strings.HasPrefix(textRequest.Model, "gpt-4") | ||||
| 			completionRatio := 1 | ||||
| 			if usingGPT4 { | ||||
| 				completionRatio = 2 | ||||
| 			} | ||||
| 			if isStream { | ||||
| 				var promptText string | ||||
| 				for _, message := range textRequest.Messages { | ||||
| 					promptText += fmt.Sprintf("%s: %s\n", message.Role, message.Content) | ||||
| 				} | ||||
| 				completionText := fmt.Sprintf("%s: %s\n", "assistant", streamResponseText) | ||||
| 				quota = countToken(promptText) + countToken(completionText)*completionRatio + 3 | ||||
| 			} else { | ||||
| 				quota = textResponse.Usage.PromptTokens + textResponse.Usage.CompletionTokens*completionRatio | ||||
| 			} | ||||
| 			ratio := common.GetModelRatio(textRequest.Model) | ||||
| 			quota = int(float64(quota) * ratio) | ||||
| 			err := model.DecreaseTokenQuota(tokenId, quota) | ||||
| 			if err != nil { | ||||
| 				common.SysError("Error consuming token remain quota: " + err.Error()) | ||||
| 			} | ||||
| 		} | ||||
| 	}() | ||||
|  | ||||
| 	if isStream { | ||||
| 		scanner := bufio.NewScanner(resp.Body) | ||||
| 		scanner.Split(func(data []byte, atEOF bool) (advance int, token []byte, err error) { | ||||
| @@ -71,6 +187,18 @@ func Relay(c *gin.Context) { | ||||
| 			for scanner.Scan() { | ||||
| 				data := scanner.Text() | ||||
| 				dataChan <- data | ||||
| 				data = data[6:] | ||||
| 				if !strings.HasPrefix(data, "[DONE]") { | ||||
| 					var streamResponse StreamResponse | ||||
| 					err = json.Unmarshal([]byte(data), &streamResponse) | ||||
| 					if err != nil { | ||||
| 						common.SysError("Error unmarshalling stream response: " + err.Error()) | ||||
| 						return | ||||
| 					} | ||||
| 					for _, choice := range streamResponse.Choices { | ||||
| 						streamResponseText += choice.Delta.Content | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 			stopChan <- true | ||||
| 		}() | ||||
| @@ -81,26 +209,57 @@ func Relay(c *gin.Context) { | ||||
| 		c.Stream(func(w io.Writer) bool { | ||||
| 			select { | ||||
| 			case data := <-dataChan: | ||||
| 				if strings.HasPrefix(data, "data: [DONE]") { | ||||
| 					data = "data: [DONE]" | ||||
| 				} | ||||
| 				c.Render(-1, common.CustomEvent{Data: data}) | ||||
| 				return true | ||||
| 			case <-stopChan: | ||||
| 				return false | ||||
| 			} | ||||
| 		}) | ||||
| 		return | ||||
| 		err = resp.Body.Close() | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		return nil | ||||
| 	} else { | ||||
| 		for k, v := range resp.Header { | ||||
| 			c.Writer.Header().Set(k, v[0]) | ||||
| 		} | ||||
| 		if consumeQuota { | ||||
| 			responseBody, err := io.ReadAll(resp.Body) | ||||
| 			if err != nil { | ||||
| 				return err | ||||
| 			} | ||||
| 			err = resp.Body.Close() | ||||
| 			if err != nil { | ||||
| 				return err | ||||
| 			} | ||||
| 			err = json.Unmarshal(responseBody, &textResponse) | ||||
| 			if err != nil { | ||||
| 				return err | ||||
| 			} | ||||
| 			// Reset response body | ||||
| 			resp.Body = io.NopCloser(bytes.NewBuffer(responseBody)) | ||||
| 		} | ||||
| 		_, err = io.Copy(c.Writer, resp.Body) | ||||
| 		if err != nil { | ||||
| 			c.JSON(http.StatusOK, gin.H{ | ||||
| 				"error": gin.H{ | ||||
| 					"message": err.Error(), | ||||
| 					"type":    "one_api_error", | ||||
| 				}, | ||||
| 			}) | ||||
| 			return | ||||
| 			return err | ||||
| 		} | ||||
| 		err = resp.Body.Close() | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		return nil | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func RelayNotImplemented(c *gin.Context) { | ||||
| 	c.JSON(http.StatusOK, gin.H{ | ||||
| 		"error": gin.H{ | ||||
| 			"message": "Not Implemented", | ||||
| 			"type":    "one_api_error", | ||||
| 		}, | ||||
| 	}) | ||||
| } | ||||
|   | ||||
| @@ -75,7 +75,32 @@ func GetToken(c *gin.Context) { | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func GetTokenStatus(c *gin.Context) { | ||||
| 	tokenId := c.GetInt("token_id") | ||||
| 	userId := c.GetInt("id") | ||||
| 	token, err := model.GetTokenByIds(tokenId, userId) | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	expiredAt := token.ExpiredTime | ||||
| 	if expiredAt == -1 { | ||||
| 		expiredAt = 0 | ||||
| 	} | ||||
| 	c.JSON(http.StatusOK, gin.H{ | ||||
| 		"object":          "credit_summary", | ||||
| 		"total_granted":   token.RemainQuota, | ||||
| 		"total_used":      0, // not supported currently | ||||
| 		"total_available": token.RemainQuota, | ||||
| 		"expires_at":      expiredAt * 1000, | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func AddToken(c *gin.Context) { | ||||
| 	isAdmin := c.GetInt("role") >= common.RoleAdminUser | ||||
| 	token := model.Token{} | ||||
| 	err := c.ShouldBindJSON(&token) | ||||
| 	if err != nil { | ||||
| @@ -99,7 +124,21 @@ func AddToken(c *gin.Context) { | ||||
| 		CreatedTime:  common.GetTimestamp(), | ||||
| 		AccessedTime: common.GetTimestamp(), | ||||
| 		ExpiredTime:  token.ExpiredTime, | ||||
| 		RemainTimes:  token.RemainTimes, | ||||
| 	} | ||||
| 	if isAdmin { | ||||
| 		cleanToken.RemainQuota = token.RemainQuota | ||||
| 		cleanToken.UnlimitedQuota = token.UnlimitedQuota | ||||
| 	} else { | ||||
| 		userId := c.GetInt("id") | ||||
| 		quota, err := model.GetUserQuota(userId) | ||||
| 		if err != nil { | ||||
| 			c.JSON(http.StatusOK, gin.H{ | ||||
| 				"success": false, | ||||
| 				"message": err.Error(), | ||||
| 			}) | ||||
| 			return | ||||
| 		} | ||||
| 		cleanToken.RemainQuota = quota | ||||
| 	} | ||||
| 	err = cleanToken.Insert() | ||||
| 	if err != nil { | ||||
| @@ -109,6 +148,10 @@ func AddToken(c *gin.Context) { | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	if !isAdmin { | ||||
| 		// update user quota | ||||
| 		err = model.DecreaseUserQuota(c.GetInt("id"), cleanToken.RemainQuota) | ||||
| 	} | ||||
| 	c.JSON(http.StatusOK, gin.H{ | ||||
| 		"success": true, | ||||
| 		"message": "", | ||||
| @@ -135,7 +178,9 @@ func DeleteToken(c *gin.Context) { | ||||
| } | ||||
|  | ||||
| func UpdateToken(c *gin.Context) { | ||||
| 	isAdmin := c.GetInt("role") >= common.RoleAdminUser | ||||
| 	userId := c.GetInt("id") | ||||
| 	statusOnly := c.Query("status_only") | ||||
| 	token := model.Token{} | ||||
| 	err := c.ShouldBindJSON(&token) | ||||
| 	if err != nil { | ||||
| @@ -161,19 +206,25 @@ func UpdateToken(c *gin.Context) { | ||||
| 			}) | ||||
| 			return | ||||
| 		} | ||||
| 		if cleanToken.Status == common.TokenStatusExhausted && cleanToken.RemainTimes == 0 { | ||||
| 		if cleanToken.Status == common.TokenStatusExhausted && cleanToken.RemainQuota <= 0 && !cleanToken.UnlimitedQuota { | ||||
| 			c.JSON(http.StatusOK, gin.H{ | ||||
| 				"success": false, | ||||
| 				"message": "令牌可用次数已用尽,无法启用,请先修改令牌剩余次数", | ||||
| 				"message": "令牌可用次数已用尽,无法启用,请先修改令牌剩余次数,或者设置为无限次数", | ||||
| 			}) | ||||
| 			return | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	cleanToken.Name = token.Name | ||||
| 	cleanToken.Status = token.Status | ||||
| 	cleanToken.ExpiredTime = token.ExpiredTime | ||||
| 	cleanToken.RemainTimes = token.RemainTimes | ||||
| 	if statusOnly != "" { | ||||
| 		cleanToken.Status = token.Status | ||||
| 	} else { | ||||
| 		// If you add more fields, please also update token.Update() | ||||
| 		cleanToken.Name = token.Name | ||||
| 		cleanToken.ExpiredTime = token.ExpiredTime | ||||
| 		if isAdmin { | ||||
| 			cleanToken.RemainQuota = token.RemainQuota | ||||
| 			cleanToken.UnlimitedQuota = token.UnlimitedQuota | ||||
| 		} | ||||
| 	} | ||||
| 	err = cleanToken.Update() | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| @@ -189,3 +240,34 @@ func UpdateToken(c *gin.Context) { | ||||
| 	}) | ||||
| 	return | ||||
| } | ||||
|  | ||||
| type topUpRequest struct { | ||||
| 	Id  int    `json:"id"` | ||||
| 	Key string `json:"key"` | ||||
| } | ||||
|  | ||||
| func TopUp(c *gin.Context) { | ||||
| 	req := topUpRequest{} | ||||
| 	err := c.ShouldBindJSON(&req) | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	quota, err := model.Redeem(req.Key, req.Id) | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	c.JSON(http.StatusOK, gin.H{ | ||||
| 		"success": true, | ||||
| 		"message": "", | ||||
| 		"data":    quota, | ||||
| 	}) | ||||
| 	return | ||||
| } | ||||
|   | ||||
| @@ -243,6 +243,42 @@ func GetUser(c *gin.Context) { | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func GenerateAccessToken(c *gin.Context) { | ||||
| 	id := c.GetInt("id") | ||||
| 	user, err := model.GetUserById(id, true) | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	user.AccessToken = common.GetUUID() | ||||
|  | ||||
| 	if model.DB.Where("token = ?", user.AccessToken).First(user).RowsAffected != 0 { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": "请重试,系统生成的 UUID 竟然重复了!", | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	if err := user.Update(false); err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": err.Error(), | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	c.JSON(http.StatusOK, gin.H{ | ||||
| 		"success": true, | ||||
| 		"message": "", | ||||
| 		"data":    user.AccessToken, | ||||
| 	}) | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func GetSelf(c *gin.Context) { | ||||
| 	id := c.GetInt("id") | ||||
| 	user, err := model.GetUserById(id, false) | ||||
| @@ -503,9 +539,23 @@ func ManageUser(c *gin.Context) { | ||||
| 	switch req.Action { | ||||
| 	case "disable": | ||||
| 		user.Status = common.UserStatusDisabled | ||||
| 		if user.Role == common.RoleRootUser { | ||||
| 			c.JSON(http.StatusOK, gin.H{ | ||||
| 				"success": false, | ||||
| 				"message": "无法禁用超级管理员用户", | ||||
| 			}) | ||||
| 			return | ||||
| 		} | ||||
| 	case "enable": | ||||
| 		user.Status = common.UserStatusEnabled | ||||
| 	case "delete": | ||||
| 		if user.Role == common.RoleRootUser { | ||||
| 			c.JSON(http.StatusOK, gin.H{ | ||||
| 				"success": false, | ||||
| 				"message": "无法删除超级管理员用户", | ||||
| 			}) | ||||
| 			return | ||||
| 		} | ||||
| 		if err := user.Delete(); err != nil { | ||||
| 			c.JSON(http.StatusOK, gin.H{ | ||||
| 				"success": false, | ||||
| @@ -521,8 +571,29 @@ func ManageUser(c *gin.Context) { | ||||
| 			}) | ||||
| 			return | ||||
| 		} | ||||
| 		if user.Role >= common.RoleAdminUser { | ||||
| 			c.JSON(http.StatusOK, gin.H{ | ||||
| 				"success": false, | ||||
| 				"message": "该用户已经是管理员", | ||||
| 			}) | ||||
| 			return | ||||
| 		} | ||||
| 		user.Role = common.RoleAdminUser | ||||
| 	case "demote": | ||||
| 		if user.Role == common.RoleRootUser { | ||||
| 			c.JSON(http.StatusOK, gin.H{ | ||||
| 				"success": false, | ||||
| 				"message": "无法降级超级管理员用户", | ||||
| 			}) | ||||
| 			return | ||||
| 		} | ||||
| 		if user.Role == common.RoleCommonUser { | ||||
| 			c.JSON(http.StatusOK, gin.H{ | ||||
| 				"success": false, | ||||
| 				"message": "该用户已经是普通用户", | ||||
| 			}) | ||||
| 			return | ||||
| 		} | ||||
| 		user.Role = common.RoleCommonUser | ||||
| 	} | ||||
|  | ||||
|   | ||||
							
								
								
									
										22
									
								
								docker-compose.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								docker-compose.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,22 @@ | ||||
| version: '3.4' | ||||
|  | ||||
| services: | ||||
|   one-api: | ||||
|     image: ghcr.io/songquanpeng/one-api:latest | ||||
|     container_name: one-api | ||||
|     restart: always | ||||
|     command: --log-dir /app/logs | ||||
|     ports: | ||||
|       - "3000:3000" | ||||
|     volumes: | ||||
|       - /home/ubuntu/data/one-api:/data | ||||
|       - /home/ubuntu/data/one-api/logs:/app/logs | ||||
|     # environment: | ||||
|     #   REDIS_CONN_STRING: redis://default:redispw@localhost:49153 | ||||
|     #   SESSION_SECRET: random_string | ||||
|     #   SQL_DSN: root:123456@tcp(localhost:3306)/one-api | ||||
|     healthcheck: | ||||
|       test: ["CMD-SHELL", "curl -s http://localhost:3000/api/status | grep -o '\"success\":\\s*true' | awk '{print $2}' | grep 'true'"] | ||||
|       interval: 30s | ||||
|       timeout: 10s | ||||
|       retries: 3 | ||||
							
								
								
									
										5
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										5
									
								
								go.mod
									
									
									
									
									
								
							| @@ -12,8 +12,8 @@ require ( | ||||
| 	github.com/go-playground/validator/v10 v10.12.0 | ||||
| 	github.com/go-redis/redis/v8 v8.11.5 | ||||
| 	github.com/google/uuid v1.3.0 | ||||
| 	github.com/pkoukk/tiktoken-go v0.1.1 | ||||
| 	golang.org/x/crypto v0.8.0 | ||||
| 	gopkg.in/gomail.v2 v2.0.0-20160411212932-81ebce5c23df | ||||
| 	gorm.io/driver/mysql v1.4.3 | ||||
| 	gorm.io/driver/sqlite v1.4.3 | ||||
| 	gorm.io/gorm v1.24.0 | ||||
| @@ -25,6 +25,7 @@ require ( | ||||
| 	github.com/cespare/xxhash/v2 v2.1.2 // indirect | ||||
| 	github.com/chenzhuoyu/base64x v0.0.0-20221115062448-fe3a3abad311 // indirect | ||||
| 	github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f // indirect | ||||
| 	github.com/dlclark/regexp2 v1.8.1 // indirect | ||||
| 	github.com/gin-contrib/sse v0.1.0 // indirect | ||||
| 	github.com/go-playground/locales v0.14.1 // indirect | ||||
| 	github.com/go-playground/universal-translator v0.18.1 // indirect | ||||
| @@ -51,7 +52,5 @@ require ( | ||||
| 	golang.org/x/sys v0.7.0 // indirect | ||||
| 	golang.org/x/text v0.9.0 // indirect | ||||
| 	google.golang.org/protobuf v1.30.0 // indirect | ||||
| 	gopkg.in/alexcesaro/quotedprintable.v3 v3.0.0-20150716171945-2caba252f4dc // indirect | ||||
| 	gopkg.in/yaml.v2 v2.4.0 // indirect | ||||
| 	gopkg.in/yaml.v3 v3.0.1 // indirect | ||||
| ) | ||||
|   | ||||
							
								
								
									
										35
									
								
								go.sum
									
									
									
									
									
								
							
							
						
						
									
										35
									
								
								go.sum
									
									
									
									
									
								
							| @@ -14,6 +14,8 @@ github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c | ||||
| github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | ||||
| github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f h1:lO4WD4F/rVNCu3HqELle0jiPLLBs70cWOduZpkS1E78= | ||||
| github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f/go.mod h1:cuUVRXasLTGF7a8hSLbxyZXjz+1KgoB3wDUb6vlszIc= | ||||
| github.com/dlclark/regexp2 v1.8.1 h1:6Lcdwya6GjPUNsBct8Lg/yRPwMhABj269AAzdGSiR+0= | ||||
| github.com/dlclark/regexp2 v1.8.1/go.mod h1:DHkYz0B9wPfa6wondMfaivmHpzrQ3v9q8cnmRbL6yW8= | ||||
| github.com/fsnotify/fsnotify v1.4.9 h1:hsms1Qyu0jgnwNXIxa+/V/PDsU6CfLf6CNO8H7IWoS4= | ||||
| github.com/gin-contrib/cors v1.4.0 h1:oJ6gwtUl3lqV0WEIwM/LxPF1QZ5qe2lGWdY2+bz7y0g= | ||||
| github.com/gin-contrib/cors v1.4.0/go.mod h1:bs9pNM0x/UsmHPBWT2xZz9ROh8xYjYkiURUfmBoMlcs= | ||||
| @@ -26,33 +28,27 @@ github.com/gin-contrib/sse v0.1.0/go.mod h1:RHrZQHXnP2xjPF+u1gW/2HnVO7nvIa9PG3Gm | ||||
| github.com/gin-contrib/static v0.0.1 h1:JVxuvHPuUfkoul12N7dtQw7KRn/pSMq7Ue1Va9Swm1U= | ||||
| github.com/gin-contrib/static v0.0.1/go.mod h1:CSxeF+wep05e0kCOsqWdAWbSszmc31zTIbD8TvWl7Hs= | ||||
| github.com/gin-gonic/gin v1.6.3/go.mod h1:75u5sXoLsGZoRN5Sgbi1eraJ4GU3++wFwWzhwvtwp4M= | ||||
| github.com/gin-gonic/gin v1.8.1 h1:4+fr/el88TOO3ewCmQr8cx/CtZ/umlIRIs5M4NTNjf8= | ||||
| github.com/gin-gonic/gin v1.8.1/go.mod h1:ji8BvRH1azfM+SYow9zQ6SZMvR8qOMZHmsCuWR9tTTk= | ||||
| github.com/gin-gonic/gin v1.9.0 h1:OjyFBKICoexlu99ctXNR2gg+c5pKrKMuyjgARg9qeY8= | ||||
| github.com/gin-gonic/gin v1.9.0/go.mod h1:W1Me9+hsUSyj3CePGrd1/QrKJMSJ1Tu/0hFEH89961k= | ||||
| github.com/go-playground/assert/v2 v2.0.1 h1:MsBgLAaY856+nPRTKrp3/OZK38U/wa0CcBYNjji3q3A= | ||||
| github.com/go-playground/assert/v2 v2.0.1/go.mod h1:VDjEfimB/XKnb+ZQfWdccd7VUvScMdVu0Titje2rxJ4= | ||||
| github.com/go-playground/assert/v2 v2.2.0 h1:JvknZsQTYeFEAhQwI4qEt9cyV5ONwRHC+lYKSsYSR8s= | ||||
| github.com/go-playground/locales v0.13.0/go.mod h1:taPMhCMXrRLJO55olJkUXHZBHCxTMfnGwq/HNwmWNS8= | ||||
| github.com/go-playground/locales v0.14.0 h1:u50s323jtVGugKlcYeyzC0etD1HifMjqmJqb8WugfUU= | ||||
| github.com/go-playground/locales v0.14.0/go.mod h1:sawfccIbzZTqEDETgFXqTho0QybSa7l++s0DH+LDiLs= | ||||
| github.com/go-playground/locales v0.14.1 h1:EWaQ/wswjilfKLTECiXz7Rh+3BjFhfDFKv/oXslEjJA= | ||||
| github.com/go-playground/locales v0.14.1/go.mod h1:hxrqLVvrK65+Rwrd5Fc6F2O76J/NuW9t0sjnWqG1slY= | ||||
| github.com/go-playground/universal-translator v0.17.0/go.mod h1:UkSxE5sNxxRwHyU+Scu5vgOQjsIJAF8j9muTVoKLVtA= | ||||
| github.com/go-playground/universal-translator v0.18.0 h1:82dyy6p4OuJq4/CByFNOn/jYrnRPArHwAcmLoJZxyho= | ||||
| github.com/go-playground/universal-translator v0.18.0/go.mod h1:UvRDBj+xPUEGrFYl+lu/H90nyDXpg0fqeB/AQUGNTVA= | ||||
| github.com/go-playground/universal-translator v0.18.1 h1:Bcnm0ZwsGyWbCzImXv+pAJnYK9S473LQFuzCbDbfSFY= | ||||
| github.com/go-playground/universal-translator v0.18.1/go.mod h1:xekY+UJKNuX9WP91TpwSH2VMlDf28Uj24BCp08ZFTUY= | ||||
| github.com/go-playground/validator/v10 v10.2.0/go.mod h1:uOYAAleCW8F/7oMFd6aG0GOhaH6EGOAJShg8Id5JGkI= | ||||
| github.com/go-playground/validator/v10 v10.10.0/go.mod h1:74x4gJWsvQexRdW8Pn3dXSGrTK4nAUsbPlLADvpJkos= | ||||
| github.com/go-playground/validator/v10 v10.11.1 h1:prmOlTVv+YjZjmRmNSF3VmspqJIxJWXmqUsHwfTRRkQ= | ||||
| github.com/go-playground/validator/v10 v10.11.1/go.mod h1:i+3WkQ1FvaUjjxh1kSvIA4dMGDBiPU55YFDl0WbKdWU= | ||||
| github.com/go-playground/validator/v10 v10.12.0 h1:E4gtWgxWxp8YSxExrQFv5BpCahla0PVF2oTTEYaWQGI= | ||||
| github.com/go-playground/validator/v10 v10.12.0/go.mod h1:hCAPuzYvKdP33pxWa+2+6AIKXEKqjIUyqsNCtbsSJrA= | ||||
| github.com/go-redis/redis/v8 v8.11.5 h1:AcZZR7igkdvfVmQTPnu9WE37LRrO/YrBH5zWyjDC0oI= | ||||
| github.com/go-redis/redis/v8 v8.11.5/go.mod h1:gREzHqY1hg6oD9ngVRbLStwAWKhA0FEgq8Jd4h5lpwo= | ||||
| github.com/go-sql-driver/mysql v1.6.0 h1:BCTh4TKNUYmOmMUcQ3IipzF5prigylS7XXjEkfCHuOE= | ||||
| github.com/go-sql-driver/mysql v1.6.0/go.mod h1:DCzpHaOWr8IXmIStZouvnhqoel9Qv2LBy8hT2VhHyBg= | ||||
| github.com/goccy/go-json v0.9.7 h1:IcB+Aqpx/iMHu5Yooh7jEzJk1JZ7Pjtmys2ukPr7EeM= | ||||
| github.com/goccy/go-json v0.9.7/go.mod h1:6MelG93GURQebXPDq3khkgXZkazVtN9CRI+MGFi0w8I= | ||||
| github.com/goccy/go-json v0.10.2 h1:CrxCmQqYDkv1z7lO7Wbh2HN93uovUHgrECaO5ZrCXAU= | ||||
| github.com/goccy/go-json v0.10.2/go.mod h1:6MelG93GURQebXPDq3khkgXZkazVtN9CRI+MGFi0w8I= | ||||
| @@ -92,19 +88,16 @@ github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= | ||||
| github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= | ||||
| github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= | ||||
| github.com/leodido/go-urn v1.2.0/go.mod h1:+8+nEpDfqqsY+g338gtMEUOtuK+4dEMhiQEgxpxOKII= | ||||
| github.com/leodido/go-urn v1.2.1 h1:BqpAaACuzVSgi/VLzGZIobT2z4v53pjosyNd9Yv6n/w= | ||||
| github.com/leodido/go-urn v1.2.1/go.mod h1:zt4jvISO2HfUBqxjfIshjdMTYS56ZS/qv49ictyFfxY= | ||||
| github.com/leodido/go-urn v1.2.3 h1:6BE2vPT0lqoz3fmOesHZiaiFh7889ssCo2GMvLCfiuA= | ||||
| github.com/leodido/go-urn v1.2.3/go.mod h1:7ZrI8mTSeBSHl/UaRyKQW1qZeMgak41ANeCNaVckg+4= | ||||
| github.com/mattn/go-isatty v0.0.12/go.mod h1:cbi8OIDigv2wuxKPP5vlRcQ1OAZbq2CE4Kysco4FUpU= | ||||
| github.com/mattn/go-isatty v0.0.14 h1:yVuAays6BHfxijgZPzw+3Zlu5yQgKGP2/hcQbHb7S9Y= | ||||
| github.com/mattn/go-isatty v0.0.14/go.mod h1:7GGIvUiUoEMVVmxf/4nioHXj79iQHKdU27kJ6hsGG94= | ||||
| github.com/mattn/go-isatty v0.0.18 h1:DOKFKCQ7FNG2L1rbrmstDN4QVRdS89Nkh85u68Uwp98= | ||||
| github.com/mattn/go-isatty v0.0.18/go.mod h1:W+V8PltTTMOvKvAeJH7IuucS94S2C6jfK/D7dTCTo3Y= | ||||
| github.com/mattn/go-sqlite3 v1.14.15/go.mod h1:2eHXhiwb8IkHr+BDWZGa96P6+rkvnG63S2DGjv9HUNg= | ||||
| github.com/mattn/go-sqlite3 v2.0.3+incompatible h1:gXHsfypPkaMZrKbD5209QV9jbUTJKjyR5WD3HYQSd+U= | ||||
| github.com/mattn/go-sqlite3 v2.0.3+incompatible/go.mod h1:FPy6KqzDD04eiIsT53CuJW3U88zkxoIYsOqkbpncsNc= | ||||
| github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421 h1:ZqeYNhU3OHLH3mGKHDcjJRFFRrJa6eAM5H+CtDdOsPc= | ||||
| github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q= | ||||
| github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd h1:TRLaZ9cD/w8PVh93nsPXa1VrQ6jlwL5oN8l14QlcNfg= | ||||
| github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q= | ||||
| @@ -114,11 +107,12 @@ github.com/modern-go/reflect2 v1.0.2/go.mod h1:yWuevngMOJpCy52FWWMvUC8ws7m/LJsjY | ||||
| github.com/nxadm/tail v1.4.8 h1:nPr65rt6Y5JFSKQO7qToXr7pePgD6Gwiw05lkbyAQTE= | ||||
| github.com/onsi/ginkgo v1.16.5 h1:8xi0RTUf59SOSfEtZMvwTvXYMzG4gV23XVHOZiXNtnE= | ||||
| github.com/onsi/gomega v1.18.1 h1:M1GfJqGRrBrrGGsbxzV5dqM2U2ApXefZCQpkukxYRLE= | ||||
| github.com/pelletier/go-toml/v2 v2.0.1 h1:8e3L2cCQzLFi2CR4g7vGFuFxX7Jl1kKX8gW+iV0GUKU= | ||||
| github.com/pelletier/go-toml/v2 v2.0.1/go.mod h1:r9LEWfGN8R5k0VXJ+0BkIe7MYkRdwZOjgMj2KwnJFUo= | ||||
| github.com/pelletier/go-toml/v2 v2.0.7 h1:muncTPStnKRos5dpVKULv2FVd4bMOhNePj9CjgDb8Us= | ||||
| github.com/pelletier/go-toml/v2 v2.0.7/go.mod h1:eumQOmlWiOPt5WriQQqoM5y18pDHwha2N+QD+EUNTek= | ||||
| github.com/pkg/diff v0.0.0-20210226163009-20ebb0f2a09e/go.mod h1:pJLUxLENpZxwdsKMEsNbx1VGcRFpLqf3715MtcvvzbA= | ||||
| github.com/pkoukk/tiktoken-go v0.1.1 h1:jtkYlIECjyM9OW1w4rjPmTohK4arORP9V25y6TM6nXo= | ||||
| github.com/pkoukk/tiktoken-go v0.1.1/go.mod h1:boMWvk9pQCOTx11pgu0DrIdrAKgQzzJKUP6vLXaz7Rw= | ||||
| github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= | ||||
| github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= | ||||
| github.com/rogpeppe/go-internal v1.6.1/go.mod h1:xXDCJY+GAPziupqXw64V24skbSoqbTEfhy4qGm1nDQc= | ||||
| @@ -132,7 +126,6 @@ github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81P | ||||
| github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= | ||||
| github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= | ||||
| github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= | ||||
| github.com/stretchr/testify v1.8.0 h1:pSgiaMZlXftHpm5L7V1+rVB+AZJydKsMxsQBIJw4PKk= | ||||
| github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU= | ||||
| github.com/stretchr/testify v1.8.1/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4= | ||||
| github.com/stretchr/testify v1.8.2 h1:+h33VjcLVPDHtOdpUCuF+7gSuG3yGIftsP1YvFihtJ8= | ||||
| @@ -142,7 +135,6 @@ github.com/twitchyliquid64/golang-asm v0.15.1/go.mod h1:a1lVb/DtPvCB8fslRZhAngC2 | ||||
| github.com/ugorji/go v1.1.7/go.mod h1:kZn38zHttfInRq0xu/PH0az30d+z6vm202qpg1oXVMw= | ||||
| github.com/ugorji/go v1.2.7/go.mod h1:nF9osbDWLy6bDVv/Rtoh6QgnvNDpmCalQV5urGCCS6M= | ||||
| github.com/ugorji/go/codec v1.1.7/go.mod h1:Ax+UKWsSmolVDwsd+7N3ZtXu+yMGCf907BLYF3GoBXY= | ||||
| github.com/ugorji/go/codec v1.2.7 h1:YPXUKf7fYbp/y8xloBqZOw2qaVggbfwMlI8WM3wZUJ0= | ||||
| github.com/ugorji/go/codec v1.2.7/go.mod h1:WGN1fab3R1fzQlVQTkfxVtIBhWDRqOviHU95kRgeqEY= | ||||
| github.com/ugorji/go/codec v1.2.11 h1:BMaWp1Bb6fHwEtbplGBGJ498wD+LKlNSl25MjdZY4dU= | ||||
| github.com/ugorji/go/codec v1.2.11/go.mod h1:UNopzCgEMSXjBc6AOMqYvWC1ktqTAfzJZUZgYf6w6lg= | ||||
| @@ -150,26 +142,17 @@ golang.org/x/arch v0.0.0-20210923205945-b76863e36670/go.mod h1:5om86z9Hs0C8fWVUu | ||||
| golang.org/x/arch v0.3.0 h1:02VY4/ZcO/gBOH6PUaoiptASxtXU10jazRCP865E97k= | ||||
| golang.org/x/arch v0.3.0/go.mod h1:5om86z9Hs0C8fWVUuoMHwpExlXzs5Tkyp9hOrfG7pp8= | ||||
| golang.org/x/crypto v0.0.0-20210711020723-a769d52b0f97/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= | ||||
| golang.org/x/crypto v0.0.0-20211215153901-e495a2d5b3d3/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4= | ||||
| golang.org/x/crypto v0.1.0 h1:MDRAIl0xIo9Io2xV565hzXHw3zVseKrJKodhohM5CjU= | ||||
| golang.org/x/crypto v0.1.0/go.mod h1:RecgLatLF4+eUMCP1PoPZQb+cVrJcOPbHkTkbkB9sbw= | ||||
| golang.org/x/crypto v0.8.0 h1:pd9TJtTueMTVQXzk8E2XESSMQDj/U7OUu0PqJqPXQjQ= | ||||
| golang.org/x/crypto v0.8.0/go.mod h1:mRqEX+O9/h5TFCrQhkgjo2yKi0yYA+9ecGkdQoHrywE= | ||||
| golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= | ||||
| golang.org/x/net v0.0.0-20211112202133-69e39bad7dc2/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= | ||||
| golang.org/x/net v0.7.0 h1:rJrUqqhjsgNp7KqAIc25s9pZnjU7TUcSY7HcVZjdn1g= | ||||
| golang.org/x/net v0.7.0/go.mod h1:2Tu9+aMcznHK/AK1HMvgo6xiTLG5rD5rZLDS+rp2Bjs= | ||||
| golang.org/x/net v0.9.0 h1:aWJ/m6xSmxWBx+V0XRHTlrYrPG56jKsLdTFmsSsCzOM= | ||||
| golang.org/x/net v0.9.0/go.mod h1:d48xBJpPfHeWQsugry2m+kC02ZBRGRgulfHnEXEuWns= | ||||
| golang.org/x/sys v0.0.0-20200116001909-b77594299b42/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210423082822-04245dca01da/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.0.0-20210630005230-0f9fa26af87c/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.0.0-20210806184541-e5e7981a1069/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.0.0-20220704084225-05e143d24a9e/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.5.0 h1:MUK/U/4lj1t1oPg0HfuXDN/Z1wv31ZJ/YcPiGccS4DU= | ||||
| golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.7.0 h1:3jlCCIQZPdOYu1h8BkNvLz8Kgwtae2cagcG/VamtZRU= | ||||
| golang.org/x/sys v0.7.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| @@ -177,28 +160,20 @@ golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9sn | ||||
| golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk= | ||||
| golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||
| golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||
| golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ= | ||||
| golang.org/x/text v0.7.0 h1:4BRB4x83lYWy72KwLD/qYDuTu7q9PjSagHvijDw7cLo= | ||||
| golang.org/x/text v0.7.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8= | ||||
| golang.org/x/text v0.9.0 h1:2sjJmO8cDvYveuX97RDLsxlyUxLl+GHoLxBiRdHllBE= | ||||
| golang.org/x/text v0.9.0/go.mod h1:e1OnstbJyHTd6l/uOt8jFFHp6TRDWZR/bV3emEE/zU8= | ||||
| golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= | ||||
| golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543 h1:E7g+9GITq07hpfrRu66IVDexMakfv52eLZ2CXBWiKr4= | ||||
| golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| google.golang.org/protobuf v1.26.0-rc.1/go.mod h1:jlhhOSvTdKEhbULTjvd4ARK9grFBp09yW+WbY/TyQbw= | ||||
| google.golang.org/protobuf v1.28.0 h1:w43yiav+6bVFTBQFZX0r7ipe9JQ1QsbMgHwbBziscLw= | ||||
| google.golang.org/protobuf v1.28.0/go.mod h1:HV8QOd/L58Z+nl8r43ehVNZIU/HEI6OcFqwMG9pJV4I= | ||||
| google.golang.org/protobuf v1.30.0 h1:kPPoIgf3TsEvrm0PFe15JQ+570QVxYzEvvHqChK+cng= | ||||
| google.golang.org/protobuf v1.30.0/go.mod h1:HV8QOd/L58Z+nl8r43ehVNZIU/HEI6OcFqwMG9pJV4I= | ||||
| gopkg.in/alexcesaro/quotedprintable.v3 v3.0.0-20150716171945-2caba252f4dc h1:2gGKlE2+asNV9m7xrywl36YYNnBG5ZQ0r/BOOxqPpmk= | ||||
| gopkg.in/alexcesaro/quotedprintable.v3 v3.0.0-20150716171945-2caba252f4dc/go.mod h1:m7x9LTH6d71AHyAX77c9yqWCCa3UKHcVEj9y7hAtKDk= | ||||
| gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||
| gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||
| gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c h1:Hei/4ADfdWqJk1ZMxUNpqntNwaWcugrBjAiHlqqRiVk= | ||||
| gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c/go.mod h1:JHkPIbrfpd72SG/EVd6muEfDQjcINNoR0C8j2r3qZ4Q= | ||||
| gopkg.in/errgo.v2 v2.1.0/go.mod h1:hNsd1EY+bozCKY1Ytp96fpM3vjJbqLJn88ws8XvfDNI= | ||||
| gopkg.in/gomail.v2 v2.0.0-20160411212932-81ebce5c23df h1:n7WqCuqOuCbNr617RXOY0AWRXxgwEyPp2z+p0+hgMuE= | ||||
| gopkg.in/gomail.v2 v2.0.0-20160411212932-81ebce5c23df/go.mod h1:LRQQ+SO6ZHR7tOkpBDuZnXENFzX8qRjMDMyPD6BRkCw= | ||||
| gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7 h1:uRGJdciOHaEIrze2W8Q3AKkepLTh2hOroT7a+7czfdQ= | ||||
| gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||
| gopkg.in/yaml.v2 v2.2.8/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||
|   | ||||
| @@ -16,12 +16,31 @@ func authHelper(c *gin.Context, minRole int) { | ||||
| 	id := session.Get("id") | ||||
| 	status := session.Get("status") | ||||
| 	if username == nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": "无权进行此操作,未登录", | ||||
| 		}) | ||||
| 		c.Abort() | ||||
| 		return | ||||
| 		// Check access token | ||||
| 		accessToken := c.Request.Header.Get("Authorization") | ||||
| 		if accessToken == "" { | ||||
| 			c.JSON(http.StatusUnauthorized, gin.H{ | ||||
| 				"success": false, | ||||
| 				"message": "无权进行此操作,未登录且未提供 access token", | ||||
| 			}) | ||||
| 			c.Abort() | ||||
| 			return | ||||
| 		} | ||||
| 		user := model.ValidateAccessToken(accessToken) | ||||
| 		if user != nil && user.Username != "" { | ||||
| 			// Token is valid | ||||
| 			username = user.Username | ||||
| 			role = user.Role | ||||
| 			id = user.Id | ||||
| 			status = user.Status | ||||
| 		} else { | ||||
| 			c.JSON(http.StatusOK, gin.H{ | ||||
| 				"success": false, | ||||
| 				"message": "无权进行此操作,access token 无效", | ||||
| 			}) | ||||
| 			c.Abort() | ||||
| 			return | ||||
| 		} | ||||
| 	} | ||||
| 	if status.(int) == common.UserStatusDisabled { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| @@ -79,9 +98,42 @@ func TokenAuth() func(c *gin.Context) { | ||||
| 			c.Abort() | ||||
| 			return | ||||
| 		} | ||||
| 		if !model.IsUserEnabled(token.UserId) { | ||||
| 			c.JSON(http.StatusOK, gin.H{ | ||||
| 				"error": gin.H{ | ||||
| 					"message": "用户已被封禁", | ||||
| 					"type":    "one_api_error", | ||||
| 				}, | ||||
| 			}) | ||||
| 			c.Abort() | ||||
| 			return | ||||
| 		} | ||||
| 		c.Set("id", token.UserId) | ||||
| 		c.Set("token_id", token.Id) | ||||
| 		requestURL := c.Request.URL.String() | ||||
| 		consumeQuota := false | ||||
| 		switch requestURL { | ||||
| 		case "/v1/chat/completions": | ||||
| 			consumeQuota = !token.UnlimitedQuota | ||||
| 		case "/v1/completions": | ||||
| 			consumeQuota = !token.UnlimitedQuota | ||||
| 		case "/v1/edits": | ||||
| 			consumeQuota = !token.UnlimitedQuota | ||||
| 		} | ||||
| 		c.Set("consume_quota", consumeQuota) | ||||
| 		if len(parts) > 1 { | ||||
| 			c.Set("channelId", parts[1]) | ||||
| 			if model.IsAdmin(token.UserId) { | ||||
| 				c.Set("channelId", parts[1]) | ||||
| 			} else { | ||||
| 				c.JSON(http.StatusOK, gin.H{ | ||||
| 					"error": gin.H{ | ||||
| 						"message": "普通用户不支持指定渠道", | ||||
| 						"type":    "one_api_error", | ||||
| 					}, | ||||
| 				}) | ||||
| 				c.Abort() | ||||
| 				return | ||||
| 			} | ||||
| 		} | ||||
| 		c.Next() | ||||
| 	} | ||||
|   | ||||
| @@ -7,6 +7,9 @@ import ( | ||||
|  | ||||
| func CORS() gin.HandlerFunc { | ||||
| 	config := cors.DefaultConfig() | ||||
| 	config.AllowOrigins = []string{"https://one-api.vercel.app", "http://localhost:3000/"} | ||||
| 	config.AllowAllOrigins = true | ||||
| 	config.AllowCredentials = true | ||||
| 	config.AllowMethods = []string{"GET", "POST", "PUT", "DELETE", "OPTIONS"} | ||||
| 	config.AllowHeaders = []string{"Origin", "Content-Length", "Content-Type", "Authorization", "Accept", "Connection"} | ||||
| 	return cors.New(config) | ||||
| } | ||||
|   | ||||
| @@ -63,8 +63,11 @@ func Distribute() func(c *gin.Context) { | ||||
| 		} | ||||
| 		c.Set("channel", channel.Type) | ||||
| 		c.Request.Header.Set("Authorization", fmt.Sprintf("Bearer %s", channel.Key)) | ||||
| 		if channel.Type == common.ChannelTypeCustom { | ||||
| 		if channel.Type == common.ChannelTypeCustom || channel.Type == common.ChannelTypeAzure { | ||||
| 			c.Set("base_url", channel.BaseURL) | ||||
| 			if channel.Type == common.ChannelTypeAzure { | ||||
| 				c.Set("api_version", channel.Other) | ||||
| 			} | ||||
| 		} | ||||
| 		c.Next() | ||||
| 	} | ||||
|   | ||||
| @@ -15,6 +15,7 @@ type Channel struct { | ||||
| 	CreatedTime  int64  `json:"created_time" gorm:"bigint"` | ||||
| 	AccessedTime int64  `json:"accessed_time" gorm:"bigint"` | ||||
| 	BaseURL      string `json:"base_url" gorm:"column:base_url"` | ||||
| 	Other        string `json:"other"` | ||||
| } | ||||
|  | ||||
| func GetAllChannels(startIdx int, num int) ([]*Channel, error) { | ||||
| @@ -52,6 +53,12 @@ func GetRandomChannel() (*Channel, error) { | ||||
| 	return &channel, err | ||||
| } | ||||
|  | ||||
| func BatchInsertChannels(channels []Channel) error { | ||||
| 	var err error | ||||
| 	err = DB.Create(&channels).Error | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func (channel *Channel) Insert() error { | ||||
| 	var err error | ||||
| 	err = DB.Create(channel).Error | ||||
|   | ||||
| @@ -25,6 +25,7 @@ func createRootAccountIfNeed() error { | ||||
| 			Role:        common.RoleRootUser, | ||||
| 			Status:      common.UserStatusEnabled, | ||||
| 			DisplayName: "Root User", | ||||
| 			AccessToken: common.GetUUID(), | ||||
| 		} | ||||
| 		DB.Create(&rootUser) | ||||
| 	} | ||||
| @@ -69,6 +70,10 @@ func InitDB() (err error) { | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		err = db.AutoMigrate(&Redemption{}) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		err = createRootAccountIfNeed() | ||||
| 		return err | ||||
| 	} else { | ||||
|   | ||||
| @@ -33,10 +33,13 @@ func InitOptionMap() { | ||||
| 	common.OptionMap["TurnstileCheckEnabled"] = strconv.FormatBool(common.TurnstileCheckEnabled) | ||||
| 	common.OptionMap["RegisterEnabled"] = strconv.FormatBool(common.RegisterEnabled) | ||||
| 	common.OptionMap["SMTPServer"] = "" | ||||
| 	common.OptionMap["SMTPFrom"] = "" | ||||
| 	common.OptionMap["SMTPPort"] = strconv.Itoa(common.SMTPPort) | ||||
| 	common.OptionMap["SMTPAccount"] = "" | ||||
| 	common.OptionMap["SMTPToken"] = "" | ||||
| 	common.OptionMap["Notice"] = "" | ||||
| 	common.OptionMap["About"] = "" | ||||
| 	common.OptionMap["HomePageContent"] = "" | ||||
| 	common.OptionMap["Footer"] = common.Footer | ||||
| 	common.OptionMap["ServerAddress"] = "" | ||||
| 	common.OptionMap["GitHubClientId"] = "" | ||||
| @@ -46,10 +49,16 @@ func InitOptionMap() { | ||||
| 	common.OptionMap["WeChatAccountQRCodeImageURL"] = "" | ||||
| 	common.OptionMap["TurnstileSiteKey"] = "" | ||||
| 	common.OptionMap["TurnstileSecretKey"] = "" | ||||
| 	common.OptionMap["QuotaForNewUser"] = strconv.Itoa(common.QuotaForNewUser) | ||||
| 	common.OptionMap["ModelRatio"] = common.ModelRatio2JSONString() | ||||
| 	common.OptionMap["TopUpLink"] = common.TopUpLink | ||||
| 	common.OptionMapRWMutex.Unlock() | ||||
| 	options, _ := AllOption() | ||||
| 	for _, option := range options { | ||||
| 		updateOptionMap(option.Key, option.Value) | ||||
| 		err := updateOptionMap(option.Key, option.Value) | ||||
| 		if err != nil { | ||||
| 			common.SysError("Failed to update option map: " + err.Error()) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -66,11 +75,10 @@ func UpdateOption(key string, value string) error { | ||||
| 	// otherwise it will execute Update (with all fields). | ||||
| 	DB.Save(&option) | ||||
| 	// Update OptionMap | ||||
| 	updateOptionMap(key, value) | ||||
| 	return nil | ||||
| 	return updateOptionMap(key, value) | ||||
| } | ||||
|  | ||||
| func updateOptionMap(key string, value string) { | ||||
| func updateOptionMap(key string, value string) (err error) { | ||||
| 	common.OptionMapRWMutex.Lock() | ||||
| 	defer common.OptionMapRWMutex.Unlock() | ||||
| 	common.OptionMap[key] = value | ||||
| @@ -109,8 +117,13 @@ func updateOptionMap(key string, value string) { | ||||
| 	switch key { | ||||
| 	case "SMTPServer": | ||||
| 		common.SMTPServer = value | ||||
| 	case "SMTPPort": | ||||
| 		intValue, _ := strconv.Atoi(value) | ||||
| 		common.SMTPPort = intValue | ||||
| 	case "SMTPAccount": | ||||
| 		common.SMTPAccount = value | ||||
| 	case "SMTPFrom": | ||||
| 		common.SMTPFrom = value | ||||
| 	case "SMTPToken": | ||||
| 		common.SMTPToken = value | ||||
| 	case "ServerAddress": | ||||
| @@ -131,5 +144,12 @@ func updateOptionMap(key string, value string) { | ||||
| 		common.TurnstileSiteKey = value | ||||
| 	case "TurnstileSecretKey": | ||||
| 		common.TurnstileSecretKey = value | ||||
| 	case "QuotaForNewUser": | ||||
| 		common.QuotaForNewUser, _ = strconv.Atoi(value) | ||||
| 	case "ModelRatio": | ||||
| 		err = common.UpdateModelRatioByJSONString(value) | ||||
| 	case "TopUpLink": | ||||
| 		common.TopUpLink = value | ||||
| 	} | ||||
| 	return err | ||||
| } | ||||
|   | ||||
							
								
								
									
										107
									
								
								model/redemption.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										107
									
								
								model/redemption.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,107 @@ | ||||
| package model | ||||
|  | ||||
| import ( | ||||
| 	"errors" | ||||
| 	_ "gorm.io/driver/sqlite" | ||||
| 	"one-api/common" | ||||
| ) | ||||
|  | ||||
| type Redemption struct { | ||||
| 	Id           int    `json:"id"` | ||||
| 	UserId       int    `json:"user_id"` | ||||
| 	Key          string `json:"key" gorm:"type:char(32);uniqueIndex"` | ||||
| 	Status       int    `json:"status" gorm:"default:1"` | ||||
| 	Name         string `json:"name" gorm:"index"` | ||||
| 	Quota        int    `json:"quota" gorm:"default:100"` | ||||
| 	CreatedTime  int64  `json:"created_time" gorm:"bigint"` | ||||
| 	RedeemedTime int64  `json:"redeemed_time" gorm:"bigint"` | ||||
| 	Count        int    `json:"count" gorm:"-:all"` // only for api request | ||||
| } | ||||
|  | ||||
| func GetAllRedemptions(startIdx int, num int) ([]*Redemption, error) { | ||||
| 	var redemptions []*Redemption | ||||
| 	var err error | ||||
| 	err = DB.Order("id desc").Limit(num).Offset(startIdx).Find(&redemptions).Error | ||||
| 	return redemptions, err | ||||
| } | ||||
|  | ||||
| func SearchRedemptions(keyword string) (redemptions []*Redemption, err error) { | ||||
| 	err = DB.Where("id = ? or name LIKE ?", keyword, keyword+"%").Find(&redemptions).Error | ||||
| 	return redemptions, err | ||||
| } | ||||
|  | ||||
| func GetRedemptionById(id int) (*Redemption, error) { | ||||
| 	if id == 0 { | ||||
| 		return nil, errors.New("id 为空!") | ||||
| 	} | ||||
| 	redemption := Redemption{Id: id} | ||||
| 	var err error = nil | ||||
| 	err = DB.First(&redemption, "id = ?", id).Error | ||||
| 	return &redemption, err | ||||
| } | ||||
|  | ||||
| func Redeem(key string, tokenId int) (quota int, err error) { | ||||
| 	if key == "" { | ||||
| 		return 0, errors.New("未提供兑换码") | ||||
| 	} | ||||
| 	if tokenId == 0 { | ||||
| 		return 0, errors.New("未提供 token id") | ||||
| 	} | ||||
| 	redemption := &Redemption{} | ||||
| 	err = DB.Where("`key` = ?", key).First(redemption).Error | ||||
| 	if err != nil { | ||||
| 		return 0, errors.New("无效的兑换码") | ||||
| 	} | ||||
| 	if redemption.Status != common.RedemptionCodeStatusEnabled { | ||||
| 		return 0, errors.New("该兑换码已被使用") | ||||
| 	} | ||||
| 	err = IncreaseTokenQuota(tokenId, redemption.Quota) | ||||
| 	if err != nil { | ||||
| 		return 0, err | ||||
| 	} | ||||
| 	go func() { | ||||
| 		redemption.RedeemedTime = common.GetTimestamp() | ||||
| 		redemption.Status = common.RedemptionCodeStatusUsed | ||||
| 		err := redemption.SelectUpdate() | ||||
| 		if err != nil { | ||||
| 			common.SysError("更新兑换码状态失败:" + err.Error()) | ||||
| 		} | ||||
| 	}() | ||||
| 	return redemption.Quota, nil | ||||
| } | ||||
|  | ||||
| func (redemption *Redemption) Insert() error { | ||||
| 	var err error | ||||
| 	err = DB.Create(redemption).Error | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func (redemption *Redemption) SelectUpdate() error { | ||||
| 	// This can update zero values | ||||
| 	return DB.Model(redemption).Select("redeemed_time", "status").Updates(redemption).Error | ||||
| } | ||||
|  | ||||
| // Update Make sure your token's fields is completed, because this will update non-zero values | ||||
| func (redemption *Redemption) Update() error { | ||||
| 	var err error | ||||
| 	err = DB.Model(redemption).Select("name", "status", "redeemed_time").Updates(redemption).Error | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func (redemption *Redemption) Delete() error { | ||||
| 	var err error | ||||
| 	err = DB.Delete(redemption).Error | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func DeleteRedemptionById(id int) (err error) { | ||||
| 	if id == 0 { | ||||
| 		return errors.New("id 为空!") | ||||
| 	} | ||||
| 	redemption := Redemption{Id: id} | ||||
| 	err = DB.Where(redemption).First(&redemption).Error | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	return redemption.Delete() | ||||
| } | ||||
| @@ -3,20 +3,22 @@ package model | ||||
| import ( | ||||
| 	"errors" | ||||
| 	_ "gorm.io/driver/sqlite" | ||||
| 	"gorm.io/gorm" | ||||
| 	"one-api/common" | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| type Token struct { | ||||
| 	Id           int    `json:"id"` | ||||
| 	UserId       int    `json:"user_id"` | ||||
| 	Key          string `json:"key" gorm:"uniqueIndex"` | ||||
| 	Status       int    `json:"status" gorm:"default:1"` | ||||
| 	Name         string `json:"name" gorm:"index" ` | ||||
| 	CreatedTime  int64  `json:"created_time" gorm:"bigint"` | ||||
| 	AccessedTime int64  `json:"accessed_time" gorm:"bigint"` | ||||
| 	ExpiredTime  int64  `json:"expired_time" gorm:"bigint;default:-1"` // -1 means never expired | ||||
| 	RemainTimes  int    `json:"remain_times" gorm:"default:-1"`        // -1 means infinite times | ||||
| 	Id             int    `json:"id"` | ||||
| 	UserId         int    `json:"user_id"` | ||||
| 	Key            string `json:"key" gorm:"type:char(32);uniqueIndex"` | ||||
| 	Status         int    `json:"status" gorm:"default:1"` | ||||
| 	Name           string `json:"name" gorm:"index" ` | ||||
| 	CreatedTime    int64  `json:"created_time" gorm:"bigint"` | ||||
| 	AccessedTime   int64  `json:"accessed_time" gorm:"bigint"` | ||||
| 	ExpiredTime    int64  `json:"expired_time" gorm:"bigint;default:-1"` // -1 means never expired | ||||
| 	RemainQuota    int    `json:"remain_quota" gorm:"default:0"` | ||||
| 	UnlimitedQuota bool   `json:"unlimited_quota" gorm:"default:false"` | ||||
| } | ||||
|  | ||||
| func GetAllUserTokens(userId int, startIdx int, num int) ([]*Token, error) { | ||||
| @@ -37,7 +39,7 @@ func ValidateUserToken(key string) (token *Token, err error) { | ||||
| 	} | ||||
| 	key = strings.Replace(key, "Bearer ", "", 1) | ||||
| 	token = &Token{} | ||||
| 	err = DB.Where("key = ?", key).First(token).Error | ||||
| 	err = DB.Where("`key` = ?", key).First(token).Error | ||||
| 	if err == nil { | ||||
| 		if token.Status != common.TokenStatusEnabled { | ||||
| 			return nil, errors.New("该 token 状态不可用") | ||||
| @@ -50,14 +52,16 @@ func ValidateUserToken(key string) (token *Token, err error) { | ||||
| 			} | ||||
| 			return nil, errors.New("该 token 已过期") | ||||
| 		} | ||||
| 		if !token.UnlimitedQuota && token.RemainQuota <= 0 { | ||||
| 			token.Status = common.TokenStatusExhausted | ||||
| 			err := token.SelectUpdate() | ||||
| 			if err != nil { | ||||
| 				common.SysError("更新 token 状态失败:" + err.Error()) | ||||
| 			} | ||||
| 			return nil, errors.New("该 token 额度已用尽") | ||||
| 		} | ||||
| 		go func() { | ||||
| 			token.AccessedTime = common.GetTimestamp() | ||||
| 			if token.RemainTimes > 0 { | ||||
| 				token.RemainTimes-- | ||||
| 				if token.RemainTimes == 0 { | ||||
| 					token.Status = common.TokenStatusExhausted | ||||
| 				} | ||||
| 			} | ||||
| 			err := token.SelectUpdate() | ||||
| 			if err != nil { | ||||
| 				common.SysError("更新 token 失败:" + err.Error()) | ||||
| @@ -84,15 +88,16 @@ func (token *Token) Insert() error { | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| // Update Make sure your token's fields is completed, because this will update non-zero values | ||||
| func (token *Token) Update() error { | ||||
| 	var err error | ||||
| 	err = DB.Model(token).Updates(token).Error | ||||
| 	err = DB.Model(token).Select("name", "status", "expired_time", "remain_quota", "unlimited_quota").Updates(token).Error | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func (token *Token) SelectUpdate() error { | ||||
| 	// This can update zero values | ||||
| 	return DB.Model(token).Select("accessed_time", "remain_times", "status").Updates(token).Error | ||||
| 	return DB.Model(token).Select("accessed_time", "status").Updates(token).Error | ||||
| } | ||||
|  | ||||
| func (token *Token) Delete() error { | ||||
| @@ -111,5 +116,26 @@ func DeleteTokenById(id int, userId int) (err error) { | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	quota := token.RemainQuota | ||||
| 	if quota != 0 { | ||||
| 		if quota > 0 { | ||||
| 			err = IncreaseUserQuota(userId, quota) | ||||
| 		} else { | ||||
| 			err = DecreaseUserQuota(userId, -quota) | ||||
| 		} | ||||
| 	} | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	return token.Delete() | ||||
| } | ||||
|  | ||||
| func IncreaseTokenQuota(id int, quota int) (err error) { | ||||
| 	err = DB.Model(&Token{}).Where("id = ?", id).Update("remain_quota", gorm.Expr("remain_quota + ?", quota)).Error | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func DecreaseTokenQuota(id int, quota int) (err error) { | ||||
| 	err = DB.Model(&Token{}).Where("id = ?", id).Update("remain_quota", gorm.Expr("remain_quota - ?", quota)).Error | ||||
| 	return err | ||||
| } | ||||
|   | ||||
| @@ -2,7 +2,9 @@ package model | ||||
|  | ||||
| import ( | ||||
| 	"errors" | ||||
| 	"gorm.io/gorm" | ||||
| 	"one-api/common" | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| // User if you add sensitive fields, don't forget to clean them in setupLogin function. | ||||
| @@ -19,6 +21,8 @@ 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! | ||||
| 	Balance          int    `json:"balance" gorm:"type:int;default:0"` | ||||
| 	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"` | ||||
| } | ||||
|  | ||||
| func GetMaxUserId() int { | ||||
| @@ -67,6 +71,8 @@ func (user *User) Insert() error { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
| 	user.Quota = common.QuotaForNewUser | ||||
| 	user.AccessToken = common.GetUUID() | ||||
| 	err = DB.Create(user).Error | ||||
| 	return err | ||||
| } | ||||
| @@ -175,3 +181,56 @@ func ResetUserPasswordByEmail(email string, password string) error { | ||||
| 	err = DB.Model(&User{}).Where("email = ?", email).Update("password", hashedPassword).Error | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func IsAdmin(userId int) bool { | ||||
| 	if userId == 0 { | ||||
| 		return false | ||||
| 	} | ||||
| 	var user User | ||||
| 	err := DB.Where("id = ?", userId).Select("role").Find(&user).Error | ||||
| 	if err != nil { | ||||
| 		common.SysError("No such user " + err.Error()) | ||||
| 		return false | ||||
| 	} | ||||
| 	return user.Role >= common.RoleAdminUser | ||||
| } | ||||
|  | ||||
| func IsUserEnabled(userId int) bool { | ||||
| 	if userId == 0 { | ||||
| 		return false | ||||
| 	} | ||||
| 	var user User | ||||
| 	err := DB.Where("id = ?", userId).Select("status").Find(&user).Error | ||||
| 	if err != nil { | ||||
| 		common.SysError("No such user " + err.Error()) | ||||
| 		return false | ||||
| 	} | ||||
| 	return user.Status == common.UserStatusEnabled | ||||
| } | ||||
|  | ||||
| func ValidateAccessToken(token string) (user *User) { | ||||
| 	if token == "" { | ||||
| 		return nil | ||||
| 	} | ||||
| 	token = strings.Replace(token, "Bearer ", "", 1) | ||||
| 	user = &User{} | ||||
| 	if DB.Where("access_token = ?", token).First(user).RowsAffected == 1 { | ||||
| 		return user | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func GetUserQuota(id int) (quota int, err error) { | ||||
| 	err = DB.Model(&User{}).Where("id = ?", id).Select("quota").Find("a).Error | ||||
| 	return quota, err | ||||
| } | ||||
|  | ||||
| func IncreaseUserQuota(id int, quota int) (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) { | ||||
| 	err = DB.Model(&User{}).Where("id = ?", id).Update("quota", gorm.Expr("quota - ?", quota)).Error | ||||
| 	return err | ||||
| } | ||||
|   | ||||
							
								
								
									
										13
									
								
								one-api.service
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								one-api.service
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | ||||
| [Unit] | ||||
| Description=One API Service | ||||
| After=network.target | ||||
|  | ||||
| [Service] | ||||
| User=yourusername                  # 守护进程用户名 | ||||
| WorkingDirectory=/path/to/One-API  # One API运行路径 | ||||
| ExecStart=/path/to/One-API/one-api --port 3000 --log-dir /path/to/One-API/logs  # 端口 | ||||
| Restart=always | ||||
| RestartSec=5 | ||||
|  | ||||
| [Install] | ||||
| WantedBy=multi-user.target | ||||
| @@ -15,6 +15,7 @@ func SetApiRouter(router *gin.Engine) { | ||||
| 		apiRouter.GET("/status", controller.GetStatus) | ||||
| 		apiRouter.GET("/notice", controller.GetNotice) | ||||
| 		apiRouter.GET("/about", controller.GetAbout) | ||||
| 		apiRouter.GET("/home_page_content", controller.GetHomePageContent) | ||||
| 		apiRouter.GET("/verification", middleware.CriticalRateLimit(), middleware.TurnstileCheck(), controller.SendEmailVerification) | ||||
| 		apiRouter.GET("/reset_password", middleware.CriticalRateLimit(), middleware.TurnstileCheck(), controller.SendPasswordResetEmail) | ||||
| 		apiRouter.POST("/user/reset", middleware.CriticalRateLimit(), controller.ResetPassword) | ||||
| @@ -35,6 +36,7 @@ func SetApiRouter(router *gin.Engine) { | ||||
| 				selfRoute.GET("/self", controller.GetSelf) | ||||
| 				selfRoute.PUT("/self", controller.UpdateSelf) | ||||
| 				selfRoute.DELETE("/self", controller.DeleteSelf) | ||||
| 				selfRoute.GET("/token", controller.GenerateAccessToken) | ||||
| 			} | ||||
|  | ||||
| 			adminRoute := userRoute.Group("/") | ||||
| @@ -70,10 +72,21 @@ func SetApiRouter(router *gin.Engine) { | ||||
| 		{ | ||||
| 			tokenRoute.GET("/", controller.GetAllTokens) | ||||
| 			tokenRoute.GET("/search", controller.SearchTokens) | ||||
| 			tokenRoute.POST("/topup", controller.TopUp) | ||||
| 			tokenRoute.GET("/:id", controller.GetToken) | ||||
| 			tokenRoute.POST("/", controller.AddToken) | ||||
| 			tokenRoute.PUT("/", controller.UpdateToken) | ||||
| 			tokenRoute.DELETE("/:id", controller.DeleteToken) | ||||
| 		} | ||||
| 		redemptionRoute := apiRouter.Group("/redemption") | ||||
| 		redemptionRoute.Use(middleware.AdminAuth()) | ||||
| 		{ | ||||
| 			redemptionRoute.GET("/", controller.GetAllRedemptions) | ||||
| 			redemptionRoute.GET("/search", controller.SearchRedemptions) | ||||
| 			redemptionRoute.GET("/:id", controller.GetRedemption) | ||||
| 			redemptionRoute.POST("/", controller.AddRedemption) | ||||
| 			redemptionRoute.PUT("/", controller.UpdateRedemption) | ||||
| 			redemptionRoute.DELETE("/:id", controller.DeleteRedemption) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|   | ||||
							
								
								
									
										18
									
								
								router/dashboard.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								router/dashboard.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| package router | ||||
|  | ||||
| import ( | ||||
| 	"github.com/gin-contrib/gzip" | ||||
| 	"github.com/gin-gonic/gin" | ||||
| 	"one-api/controller" | ||||
| 	"one-api/middleware" | ||||
| ) | ||||
|  | ||||
| func SetDashboardRouter(router *gin.Engine) { | ||||
| 	apiRouter := router.Group("/dashboard") | ||||
| 	apiRouter.Use(gzip.Gzip(gzip.DefaultCompression)) | ||||
| 	apiRouter.Use(middleware.GlobalAPIRateLimit()) | ||||
| 	apiRouter.Use(middleware.TokenAuth()) | ||||
| 	{ | ||||
| 		apiRouter.GET("/billing/credit_grants", controller.GetTokenStatus) | ||||
| 	} | ||||
| } | ||||
| @@ -7,6 +7,7 @@ import ( | ||||
|  | ||||
| func SetRouter(router *gin.Engine, buildFS embed.FS, indexPage []byte) { | ||||
| 	SetApiRouter(router) | ||||
| 	SetDashboardRouter(router) | ||||
| 	SetRelayRouter(router) | ||||
| 	setWebRouter(router, buildFS, indexPage) | ||||
| } | ||||
|   | ||||
| @@ -7,9 +7,32 @@ import ( | ||||
| ) | ||||
|  | ||||
| func SetRelayRouter(router *gin.Engine) { | ||||
| 	relayRouter := router.Group("/v1") | ||||
| 	relayRouter.Use(middleware.GlobalAPIRateLimit(), middleware.TokenAuth(), middleware.Distribute()) | ||||
| 	// https://platform.openai.com/docs/api-reference/introduction | ||||
| 	relayV1Router := router.Group("/v1") | ||||
| 	relayV1Router.Use(middleware.TokenAuth(), middleware.Distribute()) | ||||
| 	{ | ||||
| 		relayRouter.Any("/*path", controller.Relay) | ||||
| 		relayV1Router.GET("/models", controller.Relay) | ||||
| 		relayV1Router.GET("/models/:model", controller.Relay) | ||||
| 		relayV1Router.POST("/completions", controller.RelayNotImplemented) | ||||
| 		relayV1Router.POST("/chat/completions", controller.Relay) | ||||
| 		relayV1Router.POST("/edits", controller.RelayNotImplemented) | ||||
| 		relayV1Router.POST("/images/generations", controller.RelayNotImplemented) | ||||
| 		relayV1Router.POST("/images/edits", controller.RelayNotImplemented) | ||||
| 		relayV1Router.POST("/images/variations", controller.RelayNotImplemented) | ||||
| 		relayV1Router.POST("/embeddings", controller.Relay) | ||||
| 		relayV1Router.POST("/audio/transcriptions", controller.RelayNotImplemented) | ||||
| 		relayV1Router.POST("/audio/translations", controller.RelayNotImplemented) | ||||
| 		relayV1Router.GET("/files", controller.RelayNotImplemented) | ||||
| 		relayV1Router.POST("/files", controller.RelayNotImplemented) | ||||
| 		relayV1Router.DELETE("/files/:id", controller.RelayNotImplemented) | ||||
| 		relayV1Router.GET("/files/:id", controller.RelayNotImplemented) | ||||
| 		relayV1Router.GET("/files/:id/content", controller.RelayNotImplemented) | ||||
| 		relayV1Router.POST("/fine-tunes", controller.RelayNotImplemented) | ||||
| 		relayV1Router.GET("/fine-tunes", controller.RelayNotImplemented) | ||||
| 		relayV1Router.GET("/fine-tunes/:id", controller.RelayNotImplemented) | ||||
| 		relayV1Router.POST("/fine-tunes/:id/cancel", controller.RelayNotImplemented) | ||||
| 		relayV1Router.GET("/fine-tunes/:id/events", controller.RelayNotImplemented) | ||||
| 		relayV1Router.DELETE("/models/:model", controller.RelayNotImplemented) | ||||
| 		relayV1Router.POST("/moderations", controller.RelayNotImplemented) | ||||
| 	} | ||||
| } | ||||
|   | ||||
| @@ -19,7 +19,8 @@ import Channel from './pages/Channel'; | ||||
| import Token from './pages/Token'; | ||||
| import EditToken from './pages/Token/EditToken'; | ||||
| import EditChannel from './pages/Channel/EditChannel'; | ||||
| import AddChannel from './pages/Channel/AddChannel'; | ||||
| import Redemption from './pages/Redemption'; | ||||
| import EditRedemption from './pages/Redemption/EditRedemption'; | ||||
|  | ||||
| const Home = lazy(() => import('./pages/Home')); | ||||
| const About = lazy(() => import('./pages/About')); | ||||
| @@ -91,7 +92,7 @@ function App() { | ||||
|         path='/channel/add' | ||||
|         element={ | ||||
|           <Suspense fallback={<Loading></Loading>}> | ||||
|             <AddChannel /> | ||||
|             <EditChannel /> | ||||
|           </Suspense> | ||||
|         } | ||||
|       /> | ||||
| @@ -119,6 +120,30 @@ function App() { | ||||
|           </Suspense> | ||||
|         } | ||||
|       /> | ||||
|       <Route | ||||
|         path='/redemption' | ||||
|         element={ | ||||
|           <PrivateRoute> | ||||
|             <Redemption /> | ||||
|           </PrivateRoute> | ||||
|         } | ||||
|       /> | ||||
|       <Route | ||||
|         path='/redemption/edit/:id' | ||||
|         element={ | ||||
|           <Suspense fallback={<Loading></Loading>}> | ||||
|             <EditRedemption /> | ||||
|           </Suspense> | ||||
|         } | ||||
|       /> | ||||
|       <Route | ||||
|         path='/redemption/add' | ||||
|         element={ | ||||
|           <Suspense fallback={<Loading></Loading>}> | ||||
|             <EditRedemption /> | ||||
|           </Suspense> | ||||
|         } | ||||
|       /> | ||||
|       <Route | ||||
|         path='/user' | ||||
|         element={ | ||||
|   | ||||
| @@ -24,6 +24,12 @@ const headerButtons = [ | ||||
|     to: '/token', | ||||
|     icon: 'key', | ||||
|   }, | ||||
|   { | ||||
|     name: '兑换', | ||||
|     to: '/redemption', | ||||
|     icon: 'dollar sign', | ||||
|     admin: true, | ||||
|   }, | ||||
|   { | ||||
|     name: '用户', | ||||
|     to: '/user', | ||||
|   | ||||
| @@ -10,7 +10,7 @@ import { | ||||
|   Modal, | ||||
|   Segment, | ||||
| } from 'semantic-ui-react'; | ||||
| import { Link, useNavigate } from 'react-router-dom'; | ||||
| import { Link, useNavigate, useSearchParams } from 'react-router-dom'; | ||||
| import { UserContext } from '../context/User'; | ||||
| import { API, showError, showSuccess } from '../helpers'; | ||||
|  | ||||
| @@ -20,6 +20,7 @@ const LoginForm = () => { | ||||
|     password: '', | ||||
|     wechat_verification_code: '', | ||||
|   }); | ||||
|   const [searchParams, setSearchParams] = useSearchParams(); | ||||
|   const [submitted, setSubmitted] = useState(false); | ||||
|   const { username, password } = inputs; | ||||
|   const [userState, userDispatch] = useContext(UserContext); | ||||
| @@ -28,6 +29,9 @@ const LoginForm = () => { | ||||
|   const [status, setStatus] = useState({}); | ||||
|  | ||||
|   useEffect(() => { | ||||
|     if (searchParams.get("expired")) { | ||||
|       showError('未登录或登录已过期,请重新登录!'); | ||||
|     } | ||||
|     let status = localStorage.getItem('status'); | ||||
|     if (status) { | ||||
|       status = JSON.parse(status); | ||||
|   | ||||
| @@ -8,6 +8,7 @@ const OtherSetting = () => { | ||||
|     Footer: '', | ||||
|     Notice: '', | ||||
|     About: '', | ||||
|     HomePageContent: '', | ||||
|   }); | ||||
|   let originInputs = {}; | ||||
|   let [loading, setLoading] = useState(false); | ||||
| @@ -69,6 +70,10 @@ const OtherSetting = () => { | ||||
|     await updateOption('About', inputs.About); | ||||
|   }; | ||||
|  | ||||
|   const submitOption = async (key) => { | ||||
|     await updateOption(key, inputs[key]); | ||||
|   }; | ||||
|  | ||||
|   const openGitHubRelease = () => { | ||||
|     window.location = | ||||
|       'https://github.com/songquanpeng/one-api/releases/latest'; | ||||
| @@ -109,6 +114,17 @@ const OtherSetting = () => { | ||||
|           <Form.Button onClick={submitNotice}>保存公告</Form.Button> | ||||
|           <Divider /> | ||||
|           <Header as='h3'>个性化设置</Header> | ||||
|           <Form.Group widths='equal'> | ||||
|             <Form.TextArea | ||||
|               label='首页内容' | ||||
|               placeholder='在此输入首页内容,支持 Markdown & HTML 代码,设置后首页的状态信息将不再显示' | ||||
|               value={inputs.HomePageContent} | ||||
|               name='HomePageContent' | ||||
|               onChange={handleInputChange} | ||||
|               style={{ minHeight: 300, fontFamily: 'JetBrains Mono, Consolas' }} | ||||
|             /> | ||||
|           </Form.Group> | ||||
|           <Form.Button onClick={()=>submitOption('HomePageContent')}>保存首页内容</Form.Button> | ||||
|           <Form.Group widths='equal'> | ||||
|             <Form.TextArea | ||||
|               label='关于' | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| import React, { useEffect, useState } from 'react'; | ||||
| import { Button, Divider, Form, Header, Image, Modal } from 'semantic-ui-react'; | ||||
| import { Button, Divider, Form, Header, Image, Message, Modal } from 'semantic-ui-react'; | ||||
| import { Link } from 'react-router-dom'; | ||||
| import { API, copy, showError, showInfo, showSuccess } from '../helpers'; | ||||
| import Turnstile from 'react-turnstile'; | ||||
| @@ -34,6 +34,17 @@ const PersonalSetting = () => { | ||||
|     setInputs((inputs) => ({ ...inputs, [name]: value })); | ||||
|   }; | ||||
|  | ||||
|   const generateAccessToken = async () => { | ||||
|     const res = await API.get('/api/user/token'); | ||||
|     const { success, message, data } = res.data; | ||||
|     if (success) { | ||||
|       await copy(data); | ||||
|       showSuccess(`令牌已重置并已复制到剪贴板:${data}`); | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   const bindWeChat = async () => { | ||||
|     if (inputs.wechat_verification_code === '') return; | ||||
|     const res = await API.get( | ||||
| @@ -92,9 +103,13 @@ const PersonalSetting = () => { | ||||
|   return ( | ||||
|     <div style={{ lineHeight: '40px' }}> | ||||
|       <Header as='h3'>通用设置</Header> | ||||
|       <Message> | ||||
|         注意,此处生成的令牌用于系统管理,而非用于请求 OpenAI 相关的服务,请知悉。 | ||||
|       </Message> | ||||
|       <Button as={Link} to={`/user/edit/`}> | ||||
|         更新个人信息 | ||||
|       </Button> | ||||
|       <Button onClick={generateAccessToken}>生成系统访问令牌</Button> | ||||
|       <Divider /> | ||||
|       <Header as='h3'>账号绑定</Header> | ||||
|       <Button | ||||
|   | ||||
							
								
								
									
										303
									
								
								web/src/components/RedemptionsTable.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										303
									
								
								web/src/components/RedemptionsTable.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,303 @@ | ||||
| import React, { useEffect, useState } from 'react'; | ||||
| import { Button, Form, Label, Message, Pagination, Table } from 'semantic-ui-react'; | ||||
| import { Link } from 'react-router-dom'; | ||||
| import { API, copy, showError, showInfo, showSuccess, showWarning, timestamp2string } from '../helpers'; | ||||
|  | ||||
| import { ITEMS_PER_PAGE } from '../constants'; | ||||
|  | ||||
| function renderTimestamp(timestamp) { | ||||
|   return ( | ||||
|     <> | ||||
|       {timestamp2string(timestamp)} | ||||
|     </> | ||||
|   ); | ||||
| } | ||||
|  | ||||
| function renderStatus(status) { | ||||
|   switch (status) { | ||||
|     case 1: | ||||
|       return <Label basic color='green'>未使用</Label>; | ||||
|     case 2: | ||||
|       return <Label basic color='red'> 已禁用 </Label>; | ||||
|     case 3: | ||||
|       return <Label basic color='grey'> 已使用 </Label>; | ||||
|     default: | ||||
|       return <Label basic color='black'> 未知状态 </Label>; | ||||
|   } | ||||
| } | ||||
|  | ||||
| const RedemptionsTable = () => { | ||||
|   const [redemptions, setRedemptions] = useState([]); | ||||
|   const [loading, setLoading] = useState(true); | ||||
|   const [activePage, setActivePage] = useState(1); | ||||
|   const [searchKeyword, setSearchKeyword] = useState(''); | ||||
|   const [searching, setSearching] = useState(false); | ||||
|  | ||||
|   const loadRedemptions = async (startIdx) => { | ||||
|     const res = await API.get(`/api/redemption/?p=${startIdx}`); | ||||
|     const { success, message, data } = res.data; | ||||
|     if (success) { | ||||
|       if (startIdx === 0) { | ||||
|         setRedemptions(data); | ||||
|       } else { | ||||
|         let newRedemptions = redemptions; | ||||
|         newRedemptions.push(...data); | ||||
|         setRedemptions(newRedemptions); | ||||
|       } | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
|     setLoading(false); | ||||
|   }; | ||||
|  | ||||
|   const onPaginationChange = (e, { activePage }) => { | ||||
|     (async () => { | ||||
|       if (activePage === Math.ceil(redemptions.length / ITEMS_PER_PAGE) + 1) { | ||||
|         // In this case we have to load more data and then append them. | ||||
|         await loadRedemptions(activePage - 1); | ||||
|       } | ||||
|       setActivePage(activePage); | ||||
|     })(); | ||||
|   }; | ||||
|  | ||||
|   useEffect(() => { | ||||
|     loadRedemptions(0) | ||||
|       .then() | ||||
|       .catch((reason) => { | ||||
|         showError(reason); | ||||
|       }); | ||||
|   }, []); | ||||
|  | ||||
|   const manageRedemption = async (id, action, idx) => { | ||||
|     let data = { id }; | ||||
|     let res; | ||||
|     switch (action) { | ||||
|       case 'delete': | ||||
|         res = await API.delete(`/api/redemption/${id}/`); | ||||
|         break; | ||||
|       case 'enable': | ||||
|         data.status = 1; | ||||
|         res = await API.put('/api/redemption/?status_only=true', data); | ||||
|         break; | ||||
|       case 'disable': | ||||
|         data.status = 2; | ||||
|         res = await API.put('/api/redemption/?status_only=true', data); | ||||
|         break; | ||||
|     } | ||||
|     const { success, message } = res.data; | ||||
|     if (success) { | ||||
|       showSuccess('操作成功完成!'); | ||||
|       let redemption = res.data.data; | ||||
|       let newRedemptions = [...redemptions]; | ||||
|       let realIdx = (activePage - 1) * ITEMS_PER_PAGE + idx; | ||||
|       if (action === 'delete') { | ||||
|         newRedemptions[realIdx].deleted = true; | ||||
|       } else { | ||||
|         newRedemptions[realIdx].status = redemption.status; | ||||
|       } | ||||
|       setRedemptions(newRedemptions); | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   const searchRedemptions = async () => { | ||||
|     if (searchKeyword === '') { | ||||
|       // if keyword is blank, load files instead. | ||||
|       await loadRedemptions(0); | ||||
|       setActivePage(1); | ||||
|       return; | ||||
|     } | ||||
|     setSearching(true); | ||||
|     const res = await API.get(`/api/redemption/search?keyword=${searchKeyword}`); | ||||
|     const { success, message, data } = res.data; | ||||
|     if (success) { | ||||
|       setRedemptions(data); | ||||
|       setActivePage(1); | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
|     setSearching(false); | ||||
|   }; | ||||
|  | ||||
|   const handleKeywordChange = async (e, { value }) => { | ||||
|     setSearchKeyword(value.trim()); | ||||
|   }; | ||||
|  | ||||
|   const sortRedemption = (key) => { | ||||
|     if (redemptions.length === 0) return; | ||||
|     setLoading(true); | ||||
|     let sortedRedemptions = [...redemptions]; | ||||
|     sortedRedemptions.sort((a, b) => { | ||||
|       return ('' + a[key]).localeCompare(b[key]); | ||||
|     }); | ||||
|     if (sortedRedemptions[0].id === redemptions[0].id) { | ||||
|       sortedRedemptions.reverse(); | ||||
|     } | ||||
|     setRedemptions(sortedRedemptions); | ||||
|     setLoading(false); | ||||
|   }; | ||||
|  | ||||
|   return ( | ||||
|     <> | ||||
|       <Form onSubmit={searchRedemptions}> | ||||
|         <Form.Input | ||||
|           icon='search' | ||||
|           fluid | ||||
|           iconPosition='left' | ||||
|           placeholder='搜索兑换码的 ID 和名称 ...' | ||||
|           value={searchKeyword} | ||||
|           loading={searching} | ||||
|           onChange={handleKeywordChange} | ||||
|         /> | ||||
|       </Form> | ||||
|  | ||||
|       <Table basic> | ||||
|         <Table.Header> | ||||
|           <Table.Row> | ||||
|             <Table.HeaderCell | ||||
|               style={{ cursor: 'pointer' }} | ||||
|               onClick={() => { | ||||
|                 sortRedemption('id'); | ||||
|               }} | ||||
|             > | ||||
|               ID | ||||
|             </Table.HeaderCell> | ||||
|             <Table.HeaderCell | ||||
|               style={{ cursor: 'pointer' }} | ||||
|               onClick={() => { | ||||
|                 sortRedemption('name'); | ||||
|               }} | ||||
|             > | ||||
|               名称 | ||||
|             </Table.HeaderCell> | ||||
|             <Table.HeaderCell | ||||
|               style={{ cursor: 'pointer' }} | ||||
|               onClick={() => { | ||||
|                 sortRedemption('status'); | ||||
|               }} | ||||
|             > | ||||
|               状态 | ||||
|             </Table.HeaderCell> | ||||
|             <Table.HeaderCell | ||||
|               style={{ cursor: 'pointer' }} | ||||
|               onClick={() => { | ||||
|                 sortRedemption('quota'); | ||||
|               }} | ||||
|             > | ||||
|               额度 | ||||
|             </Table.HeaderCell> | ||||
|             <Table.HeaderCell | ||||
|               style={{ cursor: 'pointer' }} | ||||
|               onClick={() => { | ||||
|                 sortRedemption('created_time'); | ||||
|               }} | ||||
|             > | ||||
|               创建时间 | ||||
|             </Table.HeaderCell> | ||||
|             <Table.HeaderCell | ||||
|               style={{ cursor: 'pointer' }} | ||||
|               onClick={() => { | ||||
|                 sortRedemption('redeemed_time'); | ||||
|               }} | ||||
|             > | ||||
|               兑换时间 | ||||
|             </Table.HeaderCell> | ||||
|             <Table.HeaderCell>操作</Table.HeaderCell> | ||||
|           </Table.Row> | ||||
|         </Table.Header> | ||||
|  | ||||
|         <Table.Body> | ||||
|           {redemptions | ||||
|             .slice( | ||||
|               (activePage - 1) * ITEMS_PER_PAGE, | ||||
|               activePage * ITEMS_PER_PAGE | ||||
|             ) | ||||
|             .map((redemption, idx) => { | ||||
|               if (redemption.deleted) return <></>; | ||||
|               return ( | ||||
|                 <Table.Row key={redemption.id}> | ||||
|                   <Table.Cell>{redemption.id}</Table.Cell> | ||||
|                   <Table.Cell>{redemption.name ? redemption.name : '无'}</Table.Cell> | ||||
|                   <Table.Cell>{renderStatus(redemption.status)}</Table.Cell> | ||||
|                   <Table.Cell>{redemption.quota}</Table.Cell> | ||||
|                   <Table.Cell>{renderTimestamp(redemption.created_time)}</Table.Cell> | ||||
|                   <Table.Cell>{redemption.redeemed_time ? renderTimestamp(redemption.redeemed_time) : "尚未兑换"} </Table.Cell> | ||||
|                   <Table.Cell> | ||||
|                     <div> | ||||
|                       <Button | ||||
|                         size={'small'} | ||||
|                         positive | ||||
|                         onClick={async () => { | ||||
|                           if (await copy(redemption.key)) { | ||||
|                             showSuccess('已复制到剪贴板!'); | ||||
|                           } else { | ||||
|                             showWarning('无法复制到剪贴板,请手动复制,已将兑换码填入搜索框。') | ||||
|                             setSearchKeyword(redemption.key); | ||||
|                           } | ||||
|                         }} | ||||
|                       > | ||||
|                         复制 | ||||
|                       </Button> | ||||
|                       <Button | ||||
|                         size={'small'} | ||||
|                         negative | ||||
|                         onClick={() => { | ||||
|                           manageRedemption(redemption.id, 'delete', idx); | ||||
|                         }} | ||||
|                       > | ||||
|                         删除 | ||||
|                       </Button> | ||||
|                       <Button | ||||
|                         size={'small'} | ||||
|                         disabled={redemption.status === 3}  // used | ||||
|                         onClick={() => { | ||||
|                           manageRedemption( | ||||
|                             redemption.id, | ||||
|                             redemption.status === 1 ? 'disable' : 'enable', | ||||
|                             idx | ||||
|                           ); | ||||
|                         }} | ||||
|                       > | ||||
|                         {redemption.status === 1 ? '禁用' : '启用'} | ||||
|                       </Button> | ||||
|                       <Button | ||||
|                         size={'small'} | ||||
|                         as={Link} | ||||
|                         to={'/redemption/edit/' + redemption.id} | ||||
|                       > | ||||
|                         编辑 | ||||
|                       </Button> | ||||
|                     </div> | ||||
|                   </Table.Cell> | ||||
|                 </Table.Row> | ||||
|               ); | ||||
|             })} | ||||
|         </Table.Body> | ||||
|  | ||||
|         <Table.Footer> | ||||
|           <Table.Row> | ||||
|             <Table.HeaderCell colSpan='8'> | ||||
|               <Button size='small' as={Link} to='/redemption/add' loading={loading}> | ||||
|                 添加新的兑换码 | ||||
|               </Button> | ||||
|               <Pagination | ||||
|                 floated='right' | ||||
|                 activePage={activePage} | ||||
|                 onPageChange={onPaginationChange} | ||||
|                 size='small' | ||||
|                 siblingRange={1} | ||||
|                 totalPages={ | ||||
|                   Math.ceil(redemptions.length / ITEMS_PER_PAGE) + | ||||
|                   (redemptions.length % ITEMS_PER_PAGE === 0 ? 1 : 0) | ||||
|                 } | ||||
|               /> | ||||
|             </Table.HeaderCell> | ||||
|           </Table.Row> | ||||
|         </Table.Footer> | ||||
|       </Table> | ||||
|     </> | ||||
|   ); | ||||
| }; | ||||
|  | ||||
| export default RedemptionsTable; | ||||
| @@ -1,6 +1,6 @@ | ||||
| import React, { useEffect, useState } from 'react'; | ||||
| import { Divider, Form, Grid, Header, Message } from 'semantic-ui-react'; | ||||
| import { API, removeTrailingSlash, showError } from '../helpers'; | ||||
| import { API, removeTrailingSlash, showError, verifyJSON } from '../helpers'; | ||||
|  | ||||
| const SystemSetting = () => { | ||||
|   let [inputs, setInputs] = useState({ | ||||
| @@ -12,7 +12,9 @@ const SystemSetting = () => { | ||||
|     GitHubClientSecret: '', | ||||
|     Notice: '', | ||||
|     SMTPServer: '', | ||||
|     SMTPPort: '', | ||||
|     SMTPAccount: '', | ||||
|     SMTPFrom: '', | ||||
|     SMTPToken: '', | ||||
|     ServerAddress: '', | ||||
|     Footer: '', | ||||
| @@ -24,6 +26,9 @@ const SystemSetting = () => { | ||||
|     TurnstileSiteKey: '', | ||||
|     TurnstileSecretKey: '', | ||||
|     RegisterEnabled: '', | ||||
|     QuotaForNewUser: 0, | ||||
|     ModelRatio: '', | ||||
|     TopUpLink: '' | ||||
|   }); | ||||
|   let originInputs = {}; | ||||
|   let [loading, setLoading] = useState(false); | ||||
| @@ -64,7 +69,7 @@ const SystemSetting = () => { | ||||
|     } | ||||
|     const res = await API.put('/api/option', { | ||||
|       key, | ||||
|       value, | ||||
|       value | ||||
|     }); | ||||
|     const { success, message } = res.data; | ||||
|     if (success) { | ||||
| @@ -86,7 +91,10 @@ const SystemSetting = () => { | ||||
|       name === 'WeChatServerToken' || | ||||
|       name === 'WeChatAccountQRCodeImageURL' || | ||||
|       name === 'TurnstileSiteKey' || | ||||
|       name === 'TurnstileSecretKey' | ||||
|       name === 'TurnstileSecretKey' || | ||||
|       name === 'QuotaForNewUser' || | ||||
|       name === 'ModelRatio' || | ||||
|       name === 'TopUpLink' | ||||
|     ) { | ||||
|       setInputs((inputs) => ({ ...inputs, [name]: value })); | ||||
|     } else { | ||||
| @@ -99,6 +107,22 @@ const SystemSetting = () => { | ||||
|     await updateOption('ServerAddress', ServerAddress); | ||||
|   }; | ||||
|  | ||||
|   const submitOperationConfig = async () => { | ||||
|     if (originInputs['QuotaForNewUser'] !== inputs.QuotaForNewUser) { | ||||
|       await updateOption('QuotaForNewUser', inputs.QuotaForNewUser); | ||||
|     } | ||||
|     if (originInputs['ModelRatio'] !== inputs.ModelRatio) { | ||||
|       if (!verifyJSON(inputs.ModelRatio)) { | ||||
|         showError('模型倍率不是合法的 JSON 字符串'); | ||||
|         return; | ||||
|       } | ||||
|       await updateOption('ModelRatio', inputs.ModelRatio); | ||||
|     } | ||||
|     if (originInputs['TopUpLink'] !== inputs.TopUpLink) { | ||||
|       await updateOption('TopUpLink', inputs.TopUpLink); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   const submitSMTP = async () => { | ||||
|     if (originInputs['SMTPServer'] !== inputs.SMTPServer) { | ||||
|       await updateOption('SMTPServer', inputs.SMTPServer); | ||||
| @@ -106,6 +130,15 @@ const SystemSetting = () => { | ||||
|     if (originInputs['SMTPAccount'] !== inputs.SMTPAccount) { | ||||
|       await updateOption('SMTPAccount', inputs.SMTPAccount); | ||||
|     } | ||||
|     if (originInputs['SMTPFrom'] !== inputs.SMTPFrom) { | ||||
|       await updateOption('SMTPFrom', inputs.SMTPFrom); | ||||
|     } | ||||
|     if ( | ||||
|       originInputs['SMTPPort'] !== inputs.SMTPPort && | ||||
|       inputs.SMTPPort !== '' | ||||
|     ) { | ||||
|       await updateOption('SMTPPort', inputs.SMTPPort); | ||||
|     } | ||||
|     if ( | ||||
|       originInputs['SMTPToken'] !== inputs.SMTPToken && | ||||
|       inputs.SMTPToken !== '' | ||||
| @@ -228,6 +261,43 @@ const SystemSetting = () => { | ||||
|             /> | ||||
|           </Form.Group> | ||||
|           <Divider /> | ||||
|           <Header as='h3'> | ||||
|             运营设置 | ||||
|           </Header> | ||||
|           <Form.Group widths={3}> | ||||
|             <Form.Input | ||||
|               label='新用户初始配额' | ||||
|               name='QuotaForNewUser' | ||||
|               onChange={handleInputChange} | ||||
|               autoComplete='new-password' | ||||
|               value={inputs.QuotaForNewUser} | ||||
|               type='number' | ||||
|               min='0' | ||||
|               placeholder='例如:100' | ||||
|             /> | ||||
|             <Form.Input | ||||
|               label='充值链接' | ||||
|               name='TopUpLink' | ||||
|               onChange={handleInputChange} | ||||
|               autoComplete='new-password' | ||||
|               value={inputs.TopUpLink} | ||||
|               type='link' | ||||
|               placeholder='例如发卡网站的购买链接' | ||||
|             /> | ||||
|           </Form.Group> | ||||
|           <Form.Group widths='equal'> | ||||
|             <Form.TextArea | ||||
|               label='模型倍率' | ||||
|               name='ModelRatio' | ||||
|               onChange={handleInputChange} | ||||
|               style={{ minHeight: 250, fontFamily: 'JetBrains Mono, Consolas' }} | ||||
|               autoComplete='new-password' | ||||
|               value={inputs.ModelRatio} | ||||
|               placeholder='为一个 JSON 文本,键为模型名称,值为倍率' | ||||
|             /> | ||||
|           </Form.Group> | ||||
|           <Form.Button onClick={submitOperationConfig}>保存运营设置</Form.Button> | ||||
|           <Divider /> | ||||
|           <Header as='h3'> | ||||
|             配置 SMTP | ||||
|             <Header.Subheader>用以支持系统的邮件发送</Header.Subheader> | ||||
| @@ -237,24 +307,42 @@ const SystemSetting = () => { | ||||
|               label='SMTP 服务器地址' | ||||
|               name='SMTPServer' | ||||
|               onChange={handleInputChange} | ||||
|               autoComplete='off' | ||||
|               autoComplete='new-password' | ||||
|               value={inputs.SMTPServer} | ||||
|               placeholder='例如:smtp.qq.com' | ||||
|             /> | ||||
|             <Form.Input | ||||
|               label='SMTP 端口' | ||||
|               name='SMTPPort' | ||||
|               onChange={handleInputChange} | ||||
|               autoComplete='new-password' | ||||
|               value={inputs.SMTPPort} | ||||
|               placeholder='默认: 587' | ||||
|             /> | ||||
|             <Form.Input | ||||
|               label='SMTP 账户' | ||||
|               name='SMTPAccount' | ||||
|               onChange={handleInputChange} | ||||
|               autoComplete='off' | ||||
|               autoComplete='new-password' | ||||
|               value={inputs.SMTPAccount} | ||||
|               placeholder='通常是邮箱地址' | ||||
|             /> | ||||
|           </Form.Group> | ||||
|           <Form.Group widths={3}> | ||||
|             <Form.Input | ||||
|               label='SMTP 发送者邮箱' | ||||
|               name='SMTPFrom' | ||||
|               onChange={handleInputChange} | ||||
|               autoComplete='new-password' | ||||
|               value={inputs.SMTPFrom} | ||||
|               placeholder='通常和邮箱地址保持一致' | ||||
|             /> | ||||
|             <Form.Input | ||||
|               label='SMTP 访问凭证' | ||||
|               name='SMTPToken' | ||||
|               onChange={handleInputChange} | ||||
|               type='password' | ||||
|               autoComplete='off' | ||||
|               autoComplete='new-password' | ||||
|               value={inputs.SMTPToken} | ||||
|               placeholder='敏感信息不会发送到前端显示' | ||||
|             /> | ||||
| @@ -281,7 +369,7 @@ const SystemSetting = () => { | ||||
|               label='GitHub Client ID' | ||||
|               name='GitHubClientId' | ||||
|               onChange={handleInputChange} | ||||
|               autoComplete='off' | ||||
|               autoComplete='new-password' | ||||
|               value={inputs.GitHubClientId} | ||||
|               placeholder='输入你注册的 GitHub OAuth APP 的 ID' | ||||
|             /> | ||||
| @@ -290,7 +378,7 @@ const SystemSetting = () => { | ||||
|               name='GitHubClientSecret' | ||||
|               onChange={handleInputChange} | ||||
|               type='password' | ||||
|               autoComplete='off' | ||||
|               autoComplete='new-password' | ||||
|               value={inputs.GitHubClientSecret} | ||||
|               placeholder='敏感信息不会发送到前端显示' | ||||
|             /> | ||||
| @@ -318,7 +406,7 @@ const SystemSetting = () => { | ||||
|               name='WeChatServerAddress' | ||||
|               placeholder='例如:https://yourdomain.com' | ||||
|               onChange={handleInputChange} | ||||
|               autoComplete='off' | ||||
|               autoComplete='new-password' | ||||
|               value={inputs.WeChatServerAddress} | ||||
|             /> | ||||
|             <Form.Input | ||||
| @@ -326,7 +414,7 @@ const SystemSetting = () => { | ||||
|               name='WeChatServerToken' | ||||
|               type='password' | ||||
|               onChange={handleInputChange} | ||||
|               autoComplete='off' | ||||
|               autoComplete='new-password' | ||||
|               value={inputs.WeChatServerToken} | ||||
|               placeholder='敏感信息不会发送到前端显示' | ||||
|             /> | ||||
| @@ -334,7 +422,7 @@ const SystemSetting = () => { | ||||
|               label='微信公众号二维码图片链接' | ||||
|               name='WeChatAccountQRCodeImageURL' | ||||
|               onChange={handleInputChange} | ||||
|               autoComplete='off' | ||||
|               autoComplete='new-password' | ||||
|               value={inputs.WeChatAccountQRCodeImageURL} | ||||
|               placeholder='输入一个图片链接' | ||||
|             /> | ||||
| @@ -358,7 +446,7 @@ const SystemSetting = () => { | ||||
|               label='Turnstile Site Key' | ||||
|               name='TurnstileSiteKey' | ||||
|               onChange={handleInputChange} | ||||
|               autoComplete='off' | ||||
|               autoComplete='new-password' | ||||
|               value={inputs.TurnstileSiteKey} | ||||
|               placeholder='输入你注册的 Turnstile Site Key' | ||||
|             /> | ||||
| @@ -367,7 +455,7 @@ const SystemSetting = () => { | ||||
|               name='TurnstileSecretKey' | ||||
|               onChange={handleInputChange} | ||||
|               type='password' | ||||
|               autoComplete='off' | ||||
|               autoComplete='new-password' | ||||
|               value={inputs.TurnstileSecretKey} | ||||
|               placeholder='敏感信息不会发送到前端显示' | ||||
|             /> | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| import React, { useEffect, useState } from 'react'; | ||||
| import { Button, Form, Label, Message, Pagination, Table } from 'semantic-ui-react'; | ||||
| import { Button, Form, Label, Modal, Pagination, Table } from 'semantic-ui-react'; | ||||
| import { Link } from 'react-router-dom'; | ||||
| import { API, copy, showError, showInfo, showSuccess, showWarning, timestamp2string } from '../helpers'; | ||||
| import { API, copy, showError, showSuccess, showWarning, timestamp2string } from '../helpers'; | ||||
|  | ||||
| import { ITEMS_PER_PAGE } from '../constants'; | ||||
|  | ||||
| @@ -34,6 +34,10 @@ const TokensTable = () => { | ||||
|   const [activePage, setActivePage] = useState(1); | ||||
|   const [searchKeyword, setSearchKeyword] = useState(''); | ||||
|   const [searching, setSearching] = useState(false); | ||||
|   const [showTopUpModal, setShowTopUpModal] = useState(false); | ||||
|   const [targetTokenIdx, setTargetTokenIdx] = useState(0); | ||||
|   const [redemptionCode, setRedemptionCode] = useState(''); | ||||
|   const [topUpLink, setTopUpLink] = useState(''); | ||||
|  | ||||
|   const loadTokens = async (startIdx) => { | ||||
|     const res = await API.get(`/api/token/?p=${startIdx}`); | ||||
| @@ -68,6 +72,13 @@ const TokensTable = () => { | ||||
|       .catch((reason) => { | ||||
|         showError(reason); | ||||
|       }); | ||||
|     let status = localStorage.getItem('status'); | ||||
|     if (status) { | ||||
|       status = JSON.parse(status); | ||||
|       if (status.top_up_link) { | ||||
|         setTopUpLink(status.top_up_link); | ||||
|       } | ||||
|     } | ||||
|   }, []); | ||||
|  | ||||
|   const manageToken = async (id, action, idx) => { | ||||
| @@ -79,11 +90,11 @@ const TokensTable = () => { | ||||
|         break; | ||||
|       case 'enable': | ||||
|         data.status = 1; | ||||
|         res = await API.put('/api/token/', data); | ||||
|         res = await API.put('/api/token/?status_only=true', data); | ||||
|         break; | ||||
|       case 'disable': | ||||
|         data.status = 2; | ||||
|         res = await API.put('/api/token/', data); | ||||
|         res = await API.put('/api/token/?status_only=true', data); | ||||
|         break; | ||||
|     } | ||||
|     const { success, message } = res.data; | ||||
| @@ -140,6 +151,28 @@ const TokensTable = () => { | ||||
|     setLoading(false); | ||||
|   }; | ||||
|  | ||||
|   const topUp = async () => { | ||||
|     if (redemptionCode === '') { | ||||
|       return; | ||||
|     } | ||||
|     const res = await API.post('/api/token/topup/', { | ||||
|       id: tokens[targetTokenIdx].id, | ||||
|       key: redemptionCode | ||||
|     }); | ||||
|     const { success, message, data } = res.data; | ||||
|     if (success) { | ||||
|       showSuccess('充值成功!'); | ||||
|       let newTokens = [...tokens]; | ||||
|       let realIdx = (activePage - 1) * ITEMS_PER_PAGE + targetTokenIdx; | ||||
|       newTokens[realIdx].remain_quota += data; | ||||
|       setTokens(newTokens); | ||||
|       setRedemptionCode(''); | ||||
|       setShowTopUpModal(false); | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   return ( | ||||
|     <> | ||||
|       <Form onSubmit={searchTokens}> | ||||
| @@ -184,10 +217,10 @@ const TokensTable = () => { | ||||
|             <Table.HeaderCell | ||||
|               style={{ cursor: 'pointer' }} | ||||
|               onClick={() => { | ||||
|                 sortToken('remain_times'); | ||||
|                 sortToken('remain_quota'); | ||||
|               }} | ||||
|             > | ||||
|               剩余次数 | ||||
|               额度 | ||||
|             </Table.HeaderCell> | ||||
|             <Table.HeaderCell | ||||
|               style={{ cursor: 'pointer' }} | ||||
| @@ -197,14 +230,6 @@ const TokensTable = () => { | ||||
|             > | ||||
|               创建时间 | ||||
|             </Table.HeaderCell> | ||||
|             <Table.HeaderCell | ||||
|               style={{ cursor: 'pointer' }} | ||||
|               onClick={() => { | ||||
|                 sortToken('accessed_time'); | ||||
|               }} | ||||
|             > | ||||
|               访问时间 | ||||
|             </Table.HeaderCell> | ||||
|             <Table.HeaderCell | ||||
|               style={{ cursor: 'pointer' }} | ||||
|               onClick={() => { | ||||
| @@ -230,10 +255,9 @@ const TokensTable = () => { | ||||
|                   <Table.Cell>{token.id}</Table.Cell> | ||||
|                   <Table.Cell>{token.name ? token.name : '无'}</Table.Cell> | ||||
|                   <Table.Cell>{renderStatus(token.status)}</Table.Cell> | ||||
|                   <Table.Cell>{token.remain_times === -1 ? "无限制" : token.remain_times}</Table.Cell> | ||||
|                   <Table.Cell>{token.unlimited_quota ? '无限制' : token.remain_quota}</Table.Cell> | ||||
|                   <Table.Cell>{renderTimestamp(token.created_time)}</Table.Cell> | ||||
|                   <Table.Cell>{renderTimestamp(token.accessed_time)}</Table.Cell> | ||||
|                   <Table.Cell>{token.expired_time === -1 ? "永不过期" : renderTimestamp(token.expired_time)}</Table.Cell> | ||||
|                   <Table.Cell>{token.expired_time === -1 ? '永不过期' : renderTimestamp(token.expired_time)}</Table.Cell> | ||||
|                   <Table.Cell> | ||||
|                     <div> | ||||
|                       <Button | ||||
| @@ -243,13 +267,22 @@ const TokensTable = () => { | ||||
|                           if (await copy(token.key)) { | ||||
|                             showSuccess('已复制到剪贴板!'); | ||||
|                           } else { | ||||
|                             showWarning('无法复制到剪贴板,请手动复制,已将令牌填入搜索框。') | ||||
|                             showWarning('无法复制到剪贴板,请手动复制,已将令牌填入搜索框。'); | ||||
|                             setSearchKeyword(token.key); | ||||
|                           } | ||||
|                         }} | ||||
|                       > | ||||
|                         复制 | ||||
|                       </Button> | ||||
|                       <Button | ||||
|                         size={'small'} | ||||
|                         color={'yellow'} | ||||
|                         onClick={() => { | ||||
|                           setTargetTokenIdx(idx); | ||||
|                           setShowTopUpModal(true); | ||||
|                         }}> | ||||
|                         充值 | ||||
|                       </Button> | ||||
|                       <Button | ||||
|                         size={'small'} | ||||
|                         negative | ||||
| @@ -306,6 +339,39 @@ const TokensTable = () => { | ||||
|           </Table.Row> | ||||
|         </Table.Footer> | ||||
|       </Table> | ||||
|  | ||||
|       <Modal | ||||
|         onClose={() => setShowTopUpModal(false)} | ||||
|         onOpen={() => setShowTopUpModal(true)} | ||||
|         open={showTopUpModal} | ||||
|         size={'mini'} | ||||
|       > | ||||
|         <Modal.Header>通过兑换码为令牌「{tokens[targetTokenIdx]?.name}」充值</Modal.Header> | ||||
|         <Modal.Content> | ||||
|           <Modal.Description> | ||||
|             {/*<Image src={status.wechat_qrcode} fluid />*/} | ||||
|             { | ||||
|               topUpLink && <p> | ||||
|                   <a target='_blank' href={topUpLink}>点击此处获取兑换码</a> | ||||
|               </p> | ||||
|             } | ||||
|             <Form size='large'> | ||||
|               <Form.Input | ||||
|                 fluid | ||||
|                 placeholder='兑换码' | ||||
|                 name='redemptionCode' | ||||
|                 value={redemptionCode} | ||||
|                 onChange={(e) => { | ||||
|                   setRedemptionCode(e.target.value); | ||||
|                 }} | ||||
|               /> | ||||
|               <Button color='' fluid size='large' onClick={topUp}> | ||||
|                 充值 | ||||
|               </Button> | ||||
|             </Form> | ||||
|           </Modal.Description> | ||||
|         </Modal.Content> | ||||
|       </Modal> | ||||
|     </> | ||||
|   ); | ||||
| }; | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| import React, { useEffect, useState } from 'react'; | ||||
| import { Button, Form, Label, Pagination, Table } from 'semantic-ui-react'; | ||||
| import { Button, Form, Label, Pagination, Popup, Table } from 'semantic-ui-react'; | ||||
| import { Link } from 'react-router-dom'; | ||||
| import { API, showError, showSuccess } from '../helpers'; | ||||
|  | ||||
| @@ -237,15 +237,25 @@ const UsersTable = () => { | ||||
|                       > | ||||
|                         降级 | ||||
|                       </Button> | ||||
|                       <Button | ||||
|                         size={'small'} | ||||
|                         negative | ||||
|                         onClick={() => { | ||||
|                           manageUser(user.username, 'delete', idx); | ||||
|                         }} | ||||
|                       <Popup | ||||
|                         trigger={ | ||||
|                           <Button size='small' negative> | ||||
|                             删除 | ||||
|                           </Button> | ||||
|                         } | ||||
|                         on='click' | ||||
|                         flowing | ||||
|                         hoverable | ||||
|                       > | ||||
|                         删除 | ||||
|                       </Button> | ||||
|                         <Button | ||||
|                           negative | ||||
|                           onClick={() => { | ||||
|                             manageUser(user.username, 'delete', idx); | ||||
|                           }} | ||||
|                         > | ||||
|                           删除用户 {user.username} | ||||
|                         </Button> | ||||
|                       </Popup> | ||||
|                       <Button | ||||
|                         size={'small'} | ||||
|                         onClick={() => { | ||||
|   | ||||
| @@ -54,14 +54,18 @@ export function showError(error) { | ||||
|   console.error(error); | ||||
|   if (error.message) { | ||||
|     if (error.name === 'AxiosError') { | ||||
|       switch (error.message) { | ||||
|         case 'Request failed with status code 429': | ||||
|       switch (error.response.status) { | ||||
|         case 401: | ||||
|           // toast.error('错误:未登录或登录已过期,请重新登录!', showErrorOptions); | ||||
|           window.location.href = '/login?expired=true'; | ||||
|           break; | ||||
|         case 429: | ||||
|           toast.error('错误:请求次数过多,请稍后再试!', showErrorOptions); | ||||
|           break; | ||||
|         case 'Request failed with status code 500': | ||||
|         case 500: | ||||
|           toast.error('错误:服务器内部错误,请联系管理员!', showErrorOptions); | ||||
|           break; | ||||
|         case 'Request failed with status code 405': | ||||
|         case 405: | ||||
|           toast.info('本站仅作演示之用,无服务端!'); | ||||
|           break; | ||||
|         default: | ||||
| @@ -139,4 +143,22 @@ export function timestamp2string(timestamp) { | ||||
|     ':' + | ||||
|     second | ||||
|   ); | ||||
| } | ||||
| } | ||||
|  | ||||
| export function downloadTextAsFile(text, filename) { | ||||
|   let blob = new Blob([text], { type: 'text/plain;charset=utf-8' }); | ||||
|   let url = URL.createObjectURL(blob); | ||||
|   let a = document.createElement('a'); | ||||
|   a.href = url; | ||||
|   a.download = filename; | ||||
|   a.click(); | ||||
| } | ||||
|  | ||||
| export const verifyJSON = (str) => { | ||||
|   try { | ||||
|     JSON.parse(str); | ||||
|   } catch (e) { | ||||
|     return false; | ||||
|   } | ||||
|   return true; | ||||
| }; | ||||
| @@ -1,95 +0,0 @@ | ||||
| import React, { useState } from 'react'; | ||||
| import { Button, Form, Header, Segment } from 'semantic-ui-react'; | ||||
| import { API, showError, showSuccess } from '../../helpers'; | ||||
| import { CHANNEL_OPTIONS } from '../../constants'; | ||||
|  | ||||
| const AddChannel = () => { | ||||
|   const originInputs = { | ||||
|     name: '', | ||||
|     type: 1, | ||||
|     key: '', | ||||
|     base_url: '', | ||||
|   }; | ||||
|   const [inputs, setInputs] = useState(originInputs); | ||||
|   const { name, type, key } = inputs; | ||||
|  | ||||
|   const handleInputChange = (e, { name, value }) => { | ||||
|     setInputs((inputs) => ({ ...inputs, [name]: value })); | ||||
|   }; | ||||
|  | ||||
|   const submit = async () => { | ||||
|     if (inputs.name === '' || inputs.key === '') return; | ||||
|     if (inputs.base_url.endsWith('/')) { | ||||
|       inputs.base_url = inputs.base_url.slice(0, inputs.base_url.length - 1); | ||||
|     } | ||||
|     const res = await API.post(`/api/channel/`, inputs); | ||||
|     const { success, message } = res.data; | ||||
|     if (success) { | ||||
|       showSuccess('渠道创建成功!'); | ||||
|       setInputs(originInputs); | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   return ( | ||||
|     <> | ||||
|       <Segment> | ||||
|         <Header as='h3'>创建新的渠道</Header> | ||||
|         <Form autoComplete='off'> | ||||
|           <Form.Field> | ||||
|             <Form.Select | ||||
|               label='类型' | ||||
|               name='type' | ||||
|               options={CHANNEL_OPTIONS} | ||||
|               value={inputs.type} | ||||
|               onChange={handleInputChange} | ||||
|             /> | ||||
|           </Form.Field> | ||||
|           { | ||||
|             type === 8 && ( | ||||
|               <Form.Field> | ||||
|                 <Form.Input | ||||
|                   label='Base URL' | ||||
|                   name='base_url' | ||||
|                   placeholder={'请输入自定义渠道的 Base URL'} | ||||
|                   onChange={handleInputChange} | ||||
|                   value={inputs.base_url} | ||||
|                   autoComplete='off' | ||||
|                 /> | ||||
|               </Form.Field> | ||||
|             ) | ||||
|           } | ||||
|           <Form.Field> | ||||
|             <Form.Input | ||||
|               label='名称' | ||||
|               name='name' | ||||
|               placeholder={'请输入名称'} | ||||
|               onChange={handleInputChange} | ||||
|               value={name} | ||||
|               autoComplete='off' | ||||
|               required | ||||
|             /> | ||||
|           </Form.Field> | ||||
|           <Form.Field> | ||||
|             <Form.Input | ||||
|               label='密钥' | ||||
|               name='key' | ||||
|               placeholder={'请输入密钥'} | ||||
|               onChange={handleInputChange} | ||||
|               value={key} | ||||
|               // type='password' | ||||
|               autoComplete='off' | ||||
|               required | ||||
|             /> | ||||
|           </Form.Field> | ||||
|           <Button type={'submit'} onClick={submit}> | ||||
|             提交 | ||||
|           </Button> | ||||
|         </Form> | ||||
|       </Segment> | ||||
|     </> | ||||
|   ); | ||||
| }; | ||||
|  | ||||
| export default AddChannel; | ||||
| @@ -1,5 +1,5 @@ | ||||
| import React, { useEffect, useState } from 'react'; | ||||
| import { Button, Form, Header, Segment } from 'semantic-ui-react'; | ||||
| import { Button, Form, Header, Message, Segment } from 'semantic-ui-react'; | ||||
| import { useParams } from 'react-router-dom'; | ||||
| import { API, showError, showSuccess } from '../../helpers'; | ||||
| import { CHANNEL_OPTIONS } from '../../constants'; | ||||
| @@ -7,14 +7,19 @@ import { CHANNEL_OPTIONS } from '../../constants'; | ||||
| const EditChannel = () => { | ||||
|   const params = useParams(); | ||||
|   const channelId = params.id; | ||||
|   const [loading, setLoading] = useState(true); | ||||
|   const [inputs, setInputs] = useState({ | ||||
|   const isEdit = channelId !== undefined; | ||||
|   const [loading, setLoading] = useState(isEdit); | ||||
|   const originInputs = { | ||||
|     name: '', | ||||
|     key: '', | ||||
|     type: 1, | ||||
|     key: '', | ||||
|     base_url: '', | ||||
|   }); | ||||
|     other: '' | ||||
|   }; | ||||
|   const [batch, setBatch] = useState(false); | ||||
|   const [inputs, setInputs] = useState(originInputs); | ||||
|   const handleInputChange = (e, { name, value }) => { | ||||
|     console.log(name, value); | ||||
|     setInputs((inputs) => ({ ...inputs, [name]: value })); | ||||
|   }; | ||||
|  | ||||
| @@ -30,17 +35,31 @@ const EditChannel = () => { | ||||
|     setLoading(false); | ||||
|   }; | ||||
|   useEffect(() => { | ||||
|     loadChannel().then(); | ||||
|     if (isEdit) { | ||||
|       loadChannel().then(); | ||||
|     } | ||||
|   }, []); | ||||
|  | ||||
|   const submit = async () => { | ||||
|     if (inputs.base_url.endsWith('/')) { | ||||
|       inputs.base_url = inputs.base_url.slice(0, inputs.base_url.length - 1); | ||||
|     if (!isEdit && (inputs.name === '' || inputs.key === '')) return; | ||||
|     let localInputs = inputs; | ||||
|     if (localInputs.base_url.endsWith('/')) { | ||||
|       localInputs.base_url = localInputs.base_url.slice(0, localInputs.base_url.length - 1); | ||||
|     } | ||||
|     let res; | ||||
|     if (isEdit) { | ||||
|       res = await API.put(`/api/channel/`, { ...localInputs, id: parseInt(channelId) }); | ||||
|     } else { | ||||
|       res = await API.post(`/api/channel/`, localInputs); | ||||
|     } | ||||
|     let res = await API.put(`/api/channel/`, { ...inputs, id: parseInt(channelId) }); | ||||
|     const { success, message } = res.data; | ||||
|     if (success) { | ||||
|       showSuccess('渠道更新成功!'); | ||||
|       if (isEdit) { | ||||
|         showSuccess('渠道更新成功!'); | ||||
|       } else { | ||||
|         showSuccess('渠道创建成功!'); | ||||
|         setInputs(originInputs); | ||||
|       } | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
| @@ -49,8 +68,8 @@ const EditChannel = () => { | ||||
|   return ( | ||||
|     <> | ||||
|       <Segment loading={loading}> | ||||
|         <Header as='h3'>更新渠道信息</Header> | ||||
|         <Form autoComplete='off'> | ||||
|         <Header as='h3'>{isEdit ? '更新渠道信息' : '创建新的渠道'}</Header> | ||||
|         <Form autoComplete='new-password'> | ||||
|           <Form.Field> | ||||
|             <Form.Select | ||||
|               label='类型' | ||||
| @@ -60,16 +79,45 @@ const EditChannel = () => { | ||||
|               onChange={handleInputChange} | ||||
|             /> | ||||
|           </Form.Field> | ||||
|           { | ||||
|             inputs.type === 3 && ( | ||||
|               <> | ||||
|                 <Message> | ||||
|                   注意,<strong>模型部署名称必须和模型名称保持一致</strong>,因为 One API 会把请求体中的 model 参数替换为你的部署名称(模型名称中的点会被剔除)。 | ||||
|                 </Message> | ||||
|                 <Form.Field> | ||||
|                   <Form.Input | ||||
|                     label='AZURE_OPENAI_ENDPOINT' | ||||
|                     name='base_url' | ||||
|                     placeholder={'请输入 AZURE_OPENAI_ENDPOINT,例如:https://docs-test-001.openai.azure.com'} | ||||
|                     onChange={handleInputChange} | ||||
|                     value={inputs.base_url} | ||||
|                     autoComplete='new-password' | ||||
|                   /> | ||||
|                 </Form.Field> | ||||
|                 <Form.Field> | ||||
|                   <Form.Input | ||||
|                     label='默认 API 版本' | ||||
|                     name='other' | ||||
|                     placeholder={'请输入默认 API 版本,例如:2023-03-15-preview,该配置可以被实际的请求查询参数所覆盖'} | ||||
|                     onChange={handleInputChange} | ||||
|                     value={inputs.other} | ||||
|                     autoComplete='new-password' | ||||
|                   /> | ||||
|                 </Form.Field> | ||||
|               </> | ||||
|             ) | ||||
|           } | ||||
|           { | ||||
|             inputs.type === 8 && ( | ||||
|               <Form.Field> | ||||
|                 <Form.Input | ||||
|                   label='Base URL' | ||||
|                   name='base_url' | ||||
|                   placeholder={'请输入新的自定义渠道的 Base URL'} | ||||
|                   placeholder={'请输入自定义渠道的 Base URL,例如:https://openai.justsong.cn'} | ||||
|                   onChange={handleInputChange} | ||||
|                   value={inputs.base_url} | ||||
|                   autoComplete='off' | ||||
|                   autoComplete='new-password' | ||||
|                 /> | ||||
|               </Form.Field> | ||||
|             ) | ||||
| @@ -78,23 +126,44 @@ const EditChannel = () => { | ||||
|             <Form.Input | ||||
|               label='名称' | ||||
|               name='name' | ||||
|               placeholder={'请输入新的名称'} | ||||
|               placeholder={'请输入名称'} | ||||
|               onChange={handleInputChange} | ||||
|               value={inputs.name} | ||||
|               autoComplete='off' | ||||
|             /> | ||||
|           </Form.Field> | ||||
|           <Form.Field> | ||||
|             <Form.Input | ||||
|               label='密钥' | ||||
|               name='key' | ||||
|               placeholder={'请输入新的密钥'} | ||||
|               onChange={handleInputChange} | ||||
|               value={inputs.key} | ||||
|               // type='password' | ||||
|               autoComplete='off' | ||||
|               autoComplete='new-password' | ||||
|             /> | ||||
|           </Form.Field> | ||||
|           { | ||||
|             batch ? <Form.Field> | ||||
|               <Form.TextArea | ||||
|                 label='密钥' | ||||
|                 name='key' | ||||
|                 placeholder={'请输入密钥,一行一个'} | ||||
|                 onChange={handleInputChange} | ||||
|                 value={inputs.key} | ||||
|                 style={{ minHeight: 150, fontFamily: 'JetBrains Mono, Consolas' }} | ||||
|                 autoComplete='new-password' | ||||
|               /> | ||||
|             </Form.Field> : <Form.Field> | ||||
|               <Form.Input | ||||
|                 label='密钥' | ||||
|                 name='key' | ||||
|                 placeholder={'请输入密钥'} | ||||
|                 onChange={handleInputChange} | ||||
|                 value={inputs.key} | ||||
|                 autoComplete='new-password' | ||||
|                 /> | ||||
|             </Form.Field> | ||||
|           } | ||||
|           { | ||||
|             !isEdit && ( | ||||
|               <Form.Checkbox | ||||
|                 checked={batch} | ||||
|                 label='批量创建' | ||||
|                 name='batch' | ||||
|                 onChange={() => setBatch(!batch)} | ||||
|               /> | ||||
|             ) | ||||
|           } | ||||
|           <Button onClick={submit}>提交</Button> | ||||
|         </Form> | ||||
|       </Segment> | ||||
|   | ||||
| @@ -1,10 +1,12 @@ | ||||
| import React, { useContext, useEffect } from 'react'; | ||||
| import React, { useContext, useEffect, useState } from 'react'; | ||||
| import { Card, Grid, Header, Segment } from 'semantic-ui-react'; | ||||
| import { API, showError, showNotice, timestamp2string } from '../../helpers'; | ||||
| import { StatusContext } from '../../context/Status'; | ||||
| import { marked } from 'marked'; | ||||
|  | ||||
| const Home = () => { | ||||
|   const [statusState, statusDispatch] = useContext(StatusContext); | ||||
|   const [homePageContent, setHomePageContent] = useState(''); | ||||
|  | ||||
|   const displayNotice = async () => { | ||||
|     const res = await API.get('/api/notice'); | ||||
| @@ -20,6 +22,19 @@ const Home = () => { | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   const displayHomePageContent = async () => { | ||||
|     const res = await API.get('/api/home_page_content'); | ||||
|     const { success, message, data } = res.data; | ||||
|     if (success) { | ||||
|       let HTMLContent = marked.parse(data); | ||||
|       localStorage.setItem('home_page_content', HTMLContent); | ||||
|       setHomePageContent(HTMLContent); | ||||
|     } else { | ||||
|       showError(message); | ||||
|       setHomePageContent('加载首页内容失败...'); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   const getStartTimeString = () => { | ||||
|     const timestamp = statusState?.status?.start_time; | ||||
|     return timestamp2string(timestamp); | ||||
| @@ -27,69 +42,76 @@ const Home = () => { | ||||
|  | ||||
|   useEffect(() => { | ||||
|     displayNotice().then(); | ||||
|     displayHomePageContent().then(); | ||||
|   }, []); | ||||
|   return ( | ||||
|     <> | ||||
|       <Segment> | ||||
|         <Header as='h3'>系统状况</Header> | ||||
|         <Grid columns={2} stackable> | ||||
|           <Grid.Column> | ||||
|             <Card fluid> | ||||
|               <Card.Content> | ||||
|                 <Card.Header>系统信息</Card.Header> | ||||
|                 <Card.Meta>系统信息总览</Card.Meta> | ||||
|                 <Card.Description> | ||||
|                   <p>名称:{statusState?.status?.system_name}</p> | ||||
|                   <p>版本:{statusState?.status?.version}</p> | ||||
|                   <p> | ||||
|                     源码: | ||||
|                     <a | ||||
|                       href='https://github.com/songquanpeng/one-api' | ||||
|                       target='_blank' | ||||
|                     > | ||||
|                       https://github.com/songquanpeng/one-api | ||||
|                     </a> | ||||
|                   </p> | ||||
|                   <p>启动时间:{getStartTimeString()}</p> | ||||
|                 </Card.Description> | ||||
|               </Card.Content> | ||||
|             </Card> | ||||
|           </Grid.Column> | ||||
|           <Grid.Column> | ||||
|             <Card fluid> | ||||
|               <Card.Content> | ||||
|                 <Card.Header>系统配置</Card.Header> | ||||
|                 <Card.Meta>系统配置总览</Card.Meta> | ||||
|                 <Card.Description> | ||||
|                   <p> | ||||
|                     邮箱验证: | ||||
|                     {statusState?.status?.email_verification === true | ||||
|                       ? '已启用' | ||||
|                       : '未启用'} | ||||
|                   </p> | ||||
|                   <p> | ||||
|                     GitHub 身份验证: | ||||
|                     {statusState?.status?.github_oauth === true | ||||
|                       ? '已启用' | ||||
|                       : '未启用'} | ||||
|                   </p> | ||||
|                   <p> | ||||
|                     微信身份验证: | ||||
|                     {statusState?.status?.wechat_login === true | ||||
|                       ? '已启用' | ||||
|                       : '未启用'} | ||||
|                   </p> | ||||
|                   <p> | ||||
|                     Turnstile 用户校验: | ||||
|                     {statusState?.status?.turnstile_check === true | ||||
|                       ? '已启用' | ||||
|                       : '未启用'} | ||||
|                   </p> | ||||
|                 </Card.Description> | ||||
|               </Card.Content> | ||||
|             </Card> | ||||
|           </Grid.Column> | ||||
|         </Grid> | ||||
|         { | ||||
|           homePageContent  === '' ? <> | ||||
|           <Header as='h3'>系统状况</Header> | ||||
|           <Grid columns={2} stackable> | ||||
|             <Grid.Column> | ||||
|               <Card fluid> | ||||
|                 <Card.Content> | ||||
|                   <Card.Header>系统信息</Card.Header> | ||||
|                   <Card.Meta>系统信息总览</Card.Meta> | ||||
|                   <Card.Description> | ||||
|                     <p>名称:{statusState?.status?.system_name}</p> | ||||
|                     <p>版本:{statusState?.status?.version}</p> | ||||
|                     <p> | ||||
|                       源码: | ||||
|                       <a | ||||
|                         href='https://github.com/songquanpeng/one-api' | ||||
|                         target='_blank' | ||||
|                       > | ||||
|                         https://github.com/songquanpeng/one-api | ||||
|                       </a> | ||||
|                     </p> | ||||
|                     <p>启动时间:{getStartTimeString()}</p> | ||||
|                   </Card.Description> | ||||
|                 </Card.Content> | ||||
|               </Card> | ||||
|             </Grid.Column> | ||||
|             <Grid.Column> | ||||
|               <Card fluid> | ||||
|                 <Card.Content> | ||||
|                   <Card.Header>系统配置</Card.Header> | ||||
|                   <Card.Meta>系统配置总览</Card.Meta> | ||||
|                   <Card.Description> | ||||
|                     <p> | ||||
|                       邮箱验证: | ||||
|                       {statusState?.status?.email_verification === true | ||||
|                         ? '已启用' | ||||
|                         : '未启用'} | ||||
|                     </p> | ||||
|                     <p> | ||||
|                       GitHub 身份验证: | ||||
|                       {statusState?.status?.github_oauth === true | ||||
|                         ? '已启用' | ||||
|                         : '未启用'} | ||||
|                     </p> | ||||
|                     <p> | ||||
|                       微信身份验证: | ||||
|                       {statusState?.status?.wechat_login === true | ||||
|                         ? '已启用' | ||||
|                         : '未启用'} | ||||
|                     </p> | ||||
|                     <p> | ||||
|                       Turnstile 用户校验: | ||||
|                       {statusState?.status?.turnstile_check === true | ||||
|                         ? '已启用' | ||||
|                         : '未启用'} | ||||
|                     </p> | ||||
|                   </Card.Description> | ||||
|                 </Card.Content> | ||||
|               </Card> | ||||
|             </Grid.Column> | ||||
|           </Grid> | ||||
|           </> : <> | ||||
|           <div dangerouslySetInnerHTML={{ __html: homePageContent}}></div> | ||||
|           </> | ||||
|         } | ||||
|       </Segment> | ||||
|     </> | ||||
|   ); | ||||
|   | ||||
							
								
								
									
										121
									
								
								web/src/pages/Redemption/EditRedemption.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										121
									
								
								web/src/pages/Redemption/EditRedemption.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,121 @@ | ||||
| import React, { useEffect, useState } from 'react'; | ||||
| import { Button, Form, Header, Segment } from 'semantic-ui-react'; | ||||
| import { useParams } from 'react-router-dom'; | ||||
| import { API, downloadTextAsFile, showError, showSuccess } from '../../helpers'; | ||||
|  | ||||
| const EditRedemption = () => { | ||||
|   const params = useParams(); | ||||
|   const redemptionId = params.id; | ||||
|   const isEdit = redemptionId !== undefined; | ||||
|   const [loading, setLoading] = useState(isEdit); | ||||
|   const originInputs = { | ||||
|     name: '', | ||||
|     quota: 100, | ||||
|     count: 1 | ||||
|   }; | ||||
|   const [inputs, setInputs] = useState(originInputs); | ||||
|   const { name, quota, count } = inputs; | ||||
|  | ||||
|   const handleInputChange = (e, { name, value }) => { | ||||
|     setInputs((inputs) => ({ ...inputs, [name]: value })); | ||||
|   }; | ||||
|  | ||||
|   const loadRedemption = async () => { | ||||
|     let res = await API.get(`/api/redemption/${redemptionId}`); | ||||
|     const { success, message, data } = res.data; | ||||
|     if (success) { | ||||
|       setInputs(data); | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
|     setLoading(false); | ||||
|   }; | ||||
|   useEffect(() => { | ||||
|     if (isEdit) { | ||||
|       loadRedemption().then(); | ||||
|     } | ||||
|   }, []); | ||||
|  | ||||
|   const submit = async () => { | ||||
|     if (!isEdit && inputs.name === '') return; | ||||
|     let localInputs = inputs; | ||||
|     localInputs.count = parseInt(localInputs.count); | ||||
|     localInputs.quota = parseInt(localInputs.quota); | ||||
|     let res; | ||||
|     if (isEdit) { | ||||
|       res = await API.put(`/api/redemption/`, { ...localInputs, id: parseInt(redemptionId) }); | ||||
|     } else { | ||||
|       res = await API.post(`/api/redemption/`, { | ||||
|         ...localInputs | ||||
|       }); | ||||
|     } | ||||
|     const { success, message, data } = res.data; | ||||
|     if (success) { | ||||
|       if (isEdit) { | ||||
|         showSuccess('兑换码更新成功!'); | ||||
|       } else { | ||||
|         showSuccess('兑换码创建成功!'); | ||||
|         setInputs(originInputs); | ||||
|       } | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
|     if (!isEdit && data) { | ||||
|       let text = ""; | ||||
|       for (let i = 0; i < data.length; i++) { | ||||
|         text += data[i] + "\n"; | ||||
|       } | ||||
|       downloadTextAsFile(text, `${inputs.name}.txt`); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   return ( | ||||
|     <> | ||||
|       <Segment loading={loading}> | ||||
|         <Header as='h3'>{isEdit ? '更新兑换码信息' : '创建新的兑换码'}</Header> | ||||
|         <Form autoComplete='new-password'> | ||||
|           <Form.Field> | ||||
|             <Form.Input | ||||
|               label='名称' | ||||
|               name='name' | ||||
|               placeholder={'请输入名称'} | ||||
|               onChange={handleInputChange} | ||||
|               value={name} | ||||
|               autoComplete='new-password' | ||||
|               required={!isEdit} | ||||
|             /> | ||||
|           </Form.Field> | ||||
|           <Form.Field> | ||||
|             <Form.Input | ||||
|               label='额度' | ||||
|               name='quota' | ||||
|               placeholder={'请输入单个兑换码中包含的额度'} | ||||
|               onChange={handleInputChange} | ||||
|               value={quota} | ||||
|               autoComplete='new-password' | ||||
|               type='number' | ||||
|             /> | ||||
|           </Form.Field> | ||||
|           { | ||||
|             !isEdit && <> | ||||
|               <Form.Field> | ||||
|                 <Form.Input | ||||
|                   label='生成数量' | ||||
|                   name='count' | ||||
|                   placeholder={'请输入生成数量'} | ||||
|                   onChange={handleInputChange} | ||||
|                   value={count} | ||||
|                   autoComplete='new-password' | ||||
|                   type='number' | ||||
|                 /> | ||||
|               </Form.Field> | ||||
|             </> | ||||
|           } | ||||
|           <Button onClick={submit}>提交</Button> | ||||
|         </Form> | ||||
|       </Segment> | ||||
|     </> | ||||
|   ); | ||||
| }; | ||||
|  | ||||
| export default EditRedemption; | ||||
							
								
								
									
										14
									
								
								web/src/pages/Redemption/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								web/src/pages/Redemption/index.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | ||||
| import React from 'react'; | ||||
| import { Segment, Header } from 'semantic-ui-react'; | ||||
| import RedemptionsTable from '../../components/RedemptionsTable'; | ||||
|  | ||||
| const Redemption = () => ( | ||||
|   <> | ||||
|     <Segment> | ||||
|       <Header as='h3'>管理兑换码</Header> | ||||
|       <RedemptionsTable/> | ||||
|     </Segment> | ||||
|   </> | ||||
| ); | ||||
|  | ||||
| export default Redemption; | ||||
| @@ -1,7 +1,7 @@ | ||||
| import React, { useEffect, useState } from 'react'; | ||||
| import { Button, Form, Header, Segment } from 'semantic-ui-react'; | ||||
| import { useParams } from 'react-router-dom'; | ||||
| import { API, showError, showSuccess, timestamp2string } from '../../helpers'; | ||||
| import { API, isAdmin, showError, showSuccess, timestamp2string } from '../../helpers'; | ||||
|  | ||||
| const EditToken = () => { | ||||
|   const params = useParams(); | ||||
| @@ -10,11 +10,13 @@ const EditToken = () => { | ||||
|   const [loading, setLoading] = useState(isEdit); | ||||
|   const originInputs = { | ||||
|     name: '', | ||||
|     remain_times: -1, | ||||
|     expired_time: -1 | ||||
|     remain_quota: 0, | ||||
|     expired_time: -1, | ||||
|     unlimited_quota: false | ||||
|   }; | ||||
|   const isAdminUser = isAdmin(); | ||||
|   const [inputs, setInputs] = useState(originInputs); | ||||
|   const { name, remain_times, expired_time } = inputs; | ||||
|   const { name, remain_quota, expired_time, unlimited_quota } = inputs; | ||||
|  | ||||
|   const handleInputChange = (e, { name, value }) => { | ||||
|     setInputs((inputs) => ({ ...inputs, [name]: value })); | ||||
| @@ -35,6 +37,10 @@ const EditToken = () => { | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   const setUnlimitedQuota = () => { | ||||
|     setInputs({ ...inputs, unlimited_quota: !unlimited_quota }); | ||||
|   }; | ||||
|  | ||||
|   const loadToken = async () => { | ||||
|     let res = await API.get(`/api/token/${tokenId}`); | ||||
|     const { success, message, data } = res.data; | ||||
| @@ -57,7 +63,7 @@ const EditToken = () => { | ||||
|   const submit = async () => { | ||||
|     if (!isEdit && inputs.name === '') return; | ||||
|     let localInputs = inputs; | ||||
|     localInputs.remain_times = parseInt(localInputs.remain_times); | ||||
|     localInputs.remain_quota = parseInt(localInputs.remain_quota); | ||||
|     if (localInputs.expired_time !== -1) { | ||||
|       let time = Date.parse(localInputs.expired_time); | ||||
|       if (isNaN(time)) { | ||||
| @@ -88,8 +94,8 @@ const EditToken = () => { | ||||
|   return ( | ||||
|     <> | ||||
|       <Segment loading={loading}> | ||||
|         <Header as='h3'>{isEdit ? "更新令牌信息" : "创建新的令牌"}</Header> | ||||
|         <Form autoComplete='off'> | ||||
|         <Header as='h3'>{isEdit ? '更新令牌信息' : '创建新的令牌'}</Header> | ||||
|         <Form autoComplete='new-password'> | ||||
|           <Form.Field> | ||||
|             <Form.Input | ||||
|               label='名称' | ||||
| @@ -97,21 +103,29 @@ const EditToken = () => { | ||||
|               placeholder={'请输入名称'} | ||||
|               onChange={handleInputChange} | ||||
|               value={name} | ||||
|               autoComplete='off' | ||||
|               autoComplete='new-password' | ||||
|               required={!isEdit} | ||||
|             /> | ||||
|           </Form.Field> | ||||
|           <Form.Field> | ||||
|             <Form.Input | ||||
|               label='剩余次数' | ||||
|               name='remain_times' | ||||
|               placeholder={'请输入剩余次数,-1 表示无限制'} | ||||
|               onChange={handleInputChange} | ||||
|               value={remain_times} | ||||
|               autoComplete='off' | ||||
|               type='number' | ||||
|             /> | ||||
|           </Form.Field> | ||||
|           { | ||||
|             isAdminUser && <> | ||||
|               <Form.Field> | ||||
|                 <Form.Input | ||||
|                   label='额度' | ||||
|                   name='remain_quota' | ||||
|                   placeholder={'请输入额度'} | ||||
|                   onChange={handleInputChange} | ||||
|                   value={remain_quota} | ||||
|                   autoComplete='new-password' | ||||
|                   type='number' | ||||
|                   disabled={unlimited_quota} | ||||
|                 /> | ||||
|               </Form.Field> | ||||
|               <Button type={'button'} style={{marginBottom: '14px'}} onClick={() => { | ||||
|                 setUnlimitedQuota(); | ||||
|               }}>{unlimited_quota ? '取消无限额度' : '设置为无限额度'}</Button> | ||||
|             </> | ||||
|           } | ||||
|           <Form.Field> | ||||
|             <Form.Input | ||||
|               label='过期时间' | ||||
| @@ -119,7 +133,7 @@ const EditToken = () => { | ||||
|               placeholder={'请输入过期时间,格式为 yyyy-MM-dd HH:mm:ss,-1 表示无限制'} | ||||
|               onChange={handleInputChange} | ||||
|               value={expired_time} | ||||
|               autoComplete='off' | ||||
|               autoComplete='new-password' | ||||
|               type='datetime-local' | ||||
|             /> | ||||
|           </Form.Field> | ||||
|   | ||||
| @@ -60,7 +60,7 @@ const EditUser = () => { | ||||
|     <> | ||||
|       <Segment loading={loading}> | ||||
|         <Header as='h3'>更新用户信息</Header> | ||||
|         <Form autoComplete='off'> | ||||
|         <Form autoComplete='new-password'> | ||||
|           <Form.Field> | ||||
|             <Form.Input | ||||
|               label='用户名' | ||||
| @@ -68,7 +68,7 @@ const EditUser = () => { | ||||
|               placeholder={'请输入新的用户名'} | ||||
|               onChange={handleInputChange} | ||||
|               value={username} | ||||
|               autoComplete='off' | ||||
|               autoComplete='new-password' | ||||
|             /> | ||||
|           </Form.Field> | ||||
|           <Form.Field> | ||||
| @@ -79,7 +79,7 @@ const EditUser = () => { | ||||
|               placeholder={'请输入新的密码'} | ||||
|               onChange={handleInputChange} | ||||
|               value={password} | ||||
|               autoComplete='off' | ||||
|               autoComplete='new-password' | ||||
|             /> | ||||
|           </Form.Field> | ||||
|           <Form.Field> | ||||
| @@ -89,7 +89,7 @@ const EditUser = () => { | ||||
|               placeholder={'请输入新的显示名称'} | ||||
|               onChange={handleInputChange} | ||||
|               value={display_name} | ||||
|               autoComplete='off' | ||||
|               autoComplete='new-password' | ||||
|             /> | ||||
|           </Form.Field> | ||||
|           <Form.Field> | ||||
| @@ -97,7 +97,7 @@ const EditUser = () => { | ||||
|               label='已绑定的 GitHub 账户' | ||||
|               name='github_id' | ||||
|               value={github_id} | ||||
|               autoComplete='off' | ||||
|               autoComplete='new-password' | ||||
|               placeholder='此项只读,需要用户通过个人设置页面的相关绑定按钮进行绑定,不可直接修改' | ||||
|               readOnly | ||||
|             /> | ||||
| @@ -107,7 +107,7 @@ const EditUser = () => { | ||||
|               label='已绑定的微信账户' | ||||
|               name='wechat_id' | ||||
|               value={wechat_id} | ||||
|               autoComplete='off' | ||||
|               autoComplete='new-password' | ||||
|               placeholder='此项只读,需要用户通过个人设置页面的相关绑定按钮进行绑定,不可直接修改' | ||||
|               readOnly | ||||
|             /> | ||||
| @@ -117,7 +117,7 @@ const EditUser = () => { | ||||
|               label='已绑定的邮箱账户' | ||||
|               name='email' | ||||
|               value={email} | ||||
|               autoComplete='off' | ||||
|               autoComplete='new-password' | ||||
|               placeholder='此项只读,需要用户通过个人设置页面的相关绑定按钮进行绑定,不可直接修改' | ||||
|               readOnly | ||||
|             /> | ||||
|   | ||||
		Reference in New Issue
	
	Block a user