mirror of
				https://github.com/songquanpeng/one-api.git
				synced 2025-10-26 11:23:43 +08:00 
			
		
		
		
	Compare commits
	
		
			45 Commits
		
	
	
		
			v0.1.1
			...
			v0.1.6-alp
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | f97c2b4c22 | ||
|  | 54b1e4adef | ||
|  | 9272884381 | ||
|  | 195e94a75d | ||
|  | 5bfc224669 | ||
|  | fd149c242f | ||
|  | b9cc5dfa3f | ||
|  | 8c305dc1bc | ||
|  | f62a671fbe | ||
|  | 9e2f2383b9 | ||
|  | e7a809b082 | ||
|  | 4f8cbd643d | ||
|  | 1dd92a3f92 | ||
|  | 34a3329f5f | ||
|  | 4fb07b6d6d | ||
|  | 8be7c9ae80 | ||
|  | 4e8dc8d0cf | ||
|  | 1e46b9d135 | ||
|  | f16a2a5645 | ||
|  | 03491029f2 | ||
|  | faf84d833d | ||
|  | 109736cc05 | ||
|  | eb8f43acb5 | ||
|  | 05dd7dfd2a | ||
|  | b874784058 | ||
|  | 284beed8dc | ||
|  | 69ee87c57f | ||
|  | b74a17c963 | ||
|  | 5d602e9b57 | ||
|  | f067f64a3a | ||
|  | 01c1b906b5 | ||
|  | f6194fa86c | ||
|  | abb2449b35 | ||
|  | cc5ef9871a | ||
|  | 9e30524e2a | ||
|  | 16271e7813 | ||
|  | 46dfc6dcdd | ||
|  | 0198df5962 | ||
|  | 4c0dc50af0 | ||
|  | 423978baf4 | ||
|  | 5ed4a3d405 | ||
|  | 336c03a125 | ||
|  | 918ba60802 | ||
|  | c0bb2338de | ||
|  | 963a52521c | 
							
								
								
									
										29
									
								
								.github/workflows/github-pages.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										29
									
								
								.github/workflows/github-pages.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,29 +0,0 @@ | ||||
| name: Build GitHub Pages | ||||
| on: | ||||
|   workflow_dispatch: | ||||
|     inputs: | ||||
|       name: | ||||
|         description: 'Reason' | ||||
|         required: false | ||||
| jobs: | ||||
|   build-and-deploy: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Checkout 🛎️ | ||||
|         uses: actions/checkout@v2 # If you're using actions/checkout@v2 you must set persist-credentials to false in most cases for the deployment to work correctly. | ||||
|         with: | ||||
|           persist-credentials: false | ||||
|       - name: Install and Build 🔧 # This example project is built using npm and outputs the result to the 'build' folder. Replace with the commands required to build your project, or remove this step entirely if your site is pre-built. | ||||
|         env: | ||||
|           CI: "" | ||||
|         run: | | ||||
|           cd web | ||||
|           npm install | ||||
|           npm run build | ||||
|  | ||||
|       - name: Deploy 🚀 | ||||
|         uses: JamesIves/github-pages-deploy-action@releases/v3 | ||||
|         with: | ||||
|           ACCESS_TOKEN: ${{ secrets.ACCESS_TOKEN }} | ||||
|           BRANCH: gh-pages # The branch the action should deploy to. | ||||
|           FOLDER: web/build # The folder the action should deploy. | ||||
							
								
								
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -4,3 +4,4 @@ upload | ||||
| *.exe | ||||
| *.db | ||||
| 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 | ||||
|   | ||||
							
								
								
									
										69
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										69
									
								
								README.md
									
									
									
									
									
								
							| @@ -17,6 +17,9 @@ _✨ All in one 的 OpenAI 接口,整合各种 API 访问方式,开箱即用 | ||||
|   <a href="https://github.com/songquanpeng/one-api/releases/latest"> | ||||
|     <img src="https://img.shields.io/github/v/release/songquanpeng/one-api?color=brightgreen&include_prereleases" alt="release"> | ||||
|   </a> | ||||
|   <a href="https://hub.docker.com/repository/docker/justsong/one-api"> | ||||
|     <img src="https://img.shields.io/docker/pulls/justsong/one-api?color=brightgreen" alt="docker pull"> | ||||
|   </a> | ||||
|   <a href="https://github.com/songquanpeng/one-api/releases/latest"> | ||||
|     <img src="https://img.shields.io/github/downloads/songquanpeng/one-api/total?color=brightgreen&include_prereleases" alt="release"> | ||||
|   </a> | ||||
| @@ -49,23 +52,71 @@ _✨ All in one 的 OpenAI 接口,整合各种 API 访问方式,开箱即用 | ||||
|    + [x] 自定义渠道 | ||||
| 2. 支持通过负载均衡的方式访问多个渠道。 | ||||
| 3. 支持单个访问渠道设置多个 API Key,利用起来你的多个 API Key。 | ||||
| 4. 支持 HTTP SSE。 | ||||
| 5. 多种用户登录注册方式: | ||||
|    + 邮箱登录注册以及通过邮箱进行密码重置。 | ||||
|    + [GitHub 开放授权](https://github.com/settings/applications/new)。 | ||||
|    + 微信公众号授权(需要额外部署 [WeChat Server](https://github.com/songquanpeng/wechat-server))。 | ||||
| 6. 支持用户管理。 | ||||
| 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 进行部署 | ||||
| 执行:`docker run -d --restart always -p 3000:3000 -v /home/ubuntu/data/one-api:/data -v /etc/ssl/certs:/etc/ssl/certs:ro justsong/one-api` | ||||
| 执行:`docker run -d --restart always -p 3000:3000 -v /home/ubuntu/data/one-api:/data justsong/one-api` | ||||
|  | ||||
| 数据将会保存在宿主机的 `/home/ubuntu/data/one-api` 目录。 | ||||
| `-p 3000:3000` 中的第一个 `3000` 是宿主机的端口,可以根据需要进行修改。 | ||||
|  | ||||
| 数据将会保存在宿主机的 `/home/ubuntu/data/one-api` 目录,请确保该目录存在且具有写入权限,或者更改为合适的目录。 | ||||
|  | ||||
| Nginx 的参考配置: | ||||
| ``` | ||||
| server{ | ||||
|    server_name openai.justsong.cn;  # 请根据实际情况修改你的域名 | ||||
|     | ||||
|    location / { | ||||
|           client_max_body_size  64m; | ||||
|           proxy_http_version 1.1; | ||||
|           proxy_pass http://localhost:3000;  # 请根据实际情况修改你的端口 | ||||
|           proxy_set_header Host $host; | ||||
|           proxy_set_header X-Forwarded-For $remote_addr; | ||||
|           proxy_cache_bypass $http_upgrade; | ||||
|           proxy_set_header Accept-Encoding gzip; | ||||
|           proxy_buffering off;  # 重要:关闭代理缓冲 | ||||
|    } | ||||
| } | ||||
| ``` | ||||
|  | ||||
| 注意,为了 SSE 正常工作,需要关闭 Nginx 的代理缓冲。 | ||||
|  | ||||
| 之后使用 Let's Encrypt 的 certbot 配置 HTTPS: | ||||
| ```bash | ||||
| # Ubuntu 安装 certbot: | ||||
| sudo snap install --classic certbot | ||||
| sudo ln -s /snap/bin/certbot /usr/bin/certbot | ||||
| # 生成证书 & 修改 Nginx 配置 | ||||
| sudo certbot --nginx | ||||
| # 根据指示进行操作 | ||||
| # 重启 Nginx | ||||
| sudo service nginx restart | ||||
| ``` | ||||
|  | ||||
| ### 手动部署 | ||||
| 1. 从 [GitHub Releases](https://github.com/songquanpeng/one-api/releases/latest) 下载可执行文件或者从源码编译: | ||||
|    ```shell | ||||
|    git clone https://github.com/songquanpeng/one-api.git | ||||
|     | ||||
|    # 构建前端 | ||||
|    cd one-api/web | ||||
|    npm install | ||||
|    npm run build | ||||
|  | ||||
|    # 构建后端 | ||||
|    cd .. | ||||
|    go mod download | ||||
|    go build -ldflags "-s -w" -o one-api | ||||
|    ```` | ||||
| @@ -91,6 +142,7 @@ _✨ All in one 的 OpenAI 接口,整合各种 API 访问方式,开箱即用 | ||||
| 之后就可以使用你的令牌访问 One API 了,使用方式与 [OpenAI API](https://platform.openai.com/docs/api-reference/introduction) 一致。 | ||||
|  | ||||
| 可以通过在令牌后面添加渠道 ID 的方式指定使用哪一个渠道处理本次请求,例如:`Authorization: Bearer ONE_API_KEY-CHANNEL_ID`。 | ||||
| 注意,需要是管理员用户创建的令牌才能指定渠道 ID。 | ||||
|  | ||||
| 不加的话将会使用负载均衡的方式使用多个渠道。 | ||||
|  | ||||
| @@ -108,6 +160,7 @@ _✨ All in one 的 OpenAI 接口,整合各种 API 访问方式,开箱即用 | ||||
| 2. `--log-dir <log_dir>`: 指定日志文件夹,如果没有设置,日志将不会被保存。 | ||||
|    + 例子:`--log-dir ./logs` | ||||
| 3. `--version`: 打印系统版本号并退出。 | ||||
| 4. `--help`: 查看命令的使用帮助和参数说明。 | ||||
|  | ||||
| ## 演示 | ||||
| ### 在线演示 | ||||
|   | ||||
| @@ -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 | ||||
|  | ||||
| @@ -46,6 +47,8 @@ var WeChatAccountQRCodeImageURL = "" | ||||
| var TurnstileSiteKey = "" | ||||
| var TurnstileSecretKey = "" | ||||
|  | ||||
| var QuotaForNewUser = 100 | ||||
|  | ||||
| const ( | ||||
| 	RoleGuestUser  = 0 | ||||
| 	RoleCommonUser = 1 | ||||
| @@ -63,7 +66,7 @@ var ( | ||||
| // All duration's unit is seconds | ||||
| // Shouldn't larger then RateLimitKeyExpirationDuration | ||||
| var ( | ||||
| 	GlobalApiRateLimitNum            = 60 | ||||
| 	GlobalApiRateLimitNum            = 180 | ||||
| 	GlobalApiRateLimitDuration int64 = 3 * 60 | ||||
|  | ||||
| 	GlobalWebRateLimitNum            = 60 | ||||
| @@ -87,8 +90,16 @@ const ( | ||||
| ) | ||||
|  | ||||
| const ( | ||||
| 	TokenStatusEnabled  = 1 // don't use 0, 0 is the default value! | ||||
| 	TokenStatusDisabled = 2 // also don't use 0 | ||||
| 	TokenStatusEnabled   = 1 // don't use 0, 0 is the default value! | ||||
| 	TokenStatusDisabled  = 2 // also don't use 0 | ||||
| 	TokenStatusExpired   = 3 | ||||
| 	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 ( | ||||
|   | ||||
							
								
								
									
										82
									
								
								common/custom_event.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										82
									
								
								common/custom_event.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,82 @@ | ||||
| // Copyright 2014 Manu Martinez-Almeida.  All rights reserved. | ||||
| // Use of this source code is governed by a MIT style | ||||
| // license that can be found in the LICENSE file. | ||||
|  | ||||
| package common | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"io" | ||||
| 	"net/http" | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| type stringWriter interface { | ||||
| 	io.Writer | ||||
| 	writeString(string) (int, error) | ||||
| } | ||||
|  | ||||
| type stringWrapper struct { | ||||
| 	io.Writer | ||||
| } | ||||
|  | ||||
| func (w stringWrapper) writeString(str string) (int, error) { | ||||
| 	return w.Writer.Write([]byte(str)) | ||||
| } | ||||
|  | ||||
| func checkWriter(writer io.Writer) stringWriter { | ||||
| 	if w, ok := writer.(stringWriter); ok { | ||||
| 		return w | ||||
| 	} else { | ||||
| 		return stringWrapper{writer} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // Server-Sent Events | ||||
| // W3C Working Draft 29 October 2009 | ||||
| // http://www.w3.org/TR/2009/WD-eventsource-20091029/ | ||||
|  | ||||
| var contentType = []string{"text/event-stream"} | ||||
| var noCache = []string{"no-cache"} | ||||
|  | ||||
| var fieldReplacer = strings.NewReplacer( | ||||
| 	"\n", "\\n", | ||||
| 	"\r", "\\r") | ||||
|  | ||||
| var dataReplacer = strings.NewReplacer( | ||||
| 	"\n", "\ndata:", | ||||
| 	"\r", "\\r") | ||||
|  | ||||
| type CustomEvent struct { | ||||
| 	Event string | ||||
| 	Id    string | ||||
| 	Retry uint | ||||
| 	Data  interface{} | ||||
| } | ||||
|  | ||||
| func encode(writer io.Writer, event CustomEvent) error { | ||||
| 	w := checkWriter(writer) | ||||
| 	return writeData(w, event.Data) | ||||
| } | ||||
|  | ||||
| func writeData(w stringWriter, data interface{}) error { | ||||
| 	dataReplacer.WriteString(w, fmt.Sprint(data)) | ||||
| 	if strings.HasPrefix(data.(string), "data") { | ||||
| 		w.writeString("\n\n") | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (r CustomEvent) Render(w http.ResponseWriter) error { | ||||
| 	r.WriteContentType(w) | ||||
| 	return encode(w, r) | ||||
| } | ||||
|  | ||||
| func (r CustomEvent) WriteContentType(w http.ResponseWriter) { | ||||
| 	header := w.Header() | ||||
| 	header["Content-Type"] = contentType | ||||
|  | ||||
| 	if _, exist := header["Cache-Control"]; !exist { | ||||
| 		header["Cache-Control"] = noCache | ||||
| 	} | ||||
| } | ||||
| @@ -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 | ||||
|   | ||||
							
								
								
									
										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 | ||||
| } | ||||
| @@ -1,20 +1,26 @@ | ||||
| package controller | ||||
|  | ||||
| import ( | ||||
| 	"bufio" | ||||
| 	"fmt" | ||||
| 	"github.com/gin-gonic/gin" | ||||
| 	"io" | ||||
| 	"net/http" | ||||
| 	"one-api/common" | ||||
| 	"one-api/model" | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| func Relay(c *gin.Context) { | ||||
| 	channelType := c.GetInt("channel") | ||||
| 	tokenId := c.GetInt("token_id") | ||||
| 	isUnlimitedTimes := c.GetBool("unlimited_times") | ||||
| 	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) | ||||
| 	requestURL := c.Request.URL.String() | ||||
| 	req, err := http.NewRequest(c.Request.Method, fmt.Sprintf("%s%s", baseURL, requestURL), c.Request.Body) | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"error": gin.H{ | ||||
| @@ -24,10 +30,14 @@ func Relay(c *gin.Context) { | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	req.Header = c.Request.Header.Clone() | ||||
| 	//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.Del("Accept-Encoding") | ||||
| 	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) | ||||
| @@ -40,17 +50,73 @@ func Relay(c *gin.Context) { | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	for k, v := range resp.Header { | ||||
| 		c.Writer.Header().Set(k, v[0]) | ||||
| 	} | ||||
| 	_, 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", | ||||
| 			}, | ||||
|  | ||||
| 	defer func() { | ||||
| 		err := req.Body.Close() | ||||
| 		if err != nil { | ||||
| 			common.SysError("Error closing request body: " + err.Error()) | ||||
| 		} | ||||
| 		if !isUnlimitedTimes && requestURL == "/v1/chat/completions" { | ||||
| 			err := model.DecreaseTokenRemainTimesById(tokenId) | ||||
| 			if err != nil { | ||||
| 				common.SysError("Error decreasing token remain times: " + err.Error()) | ||||
| 			} | ||||
| 		} | ||||
| 	}() | ||||
| 	isStream := resp.Header.Get("Content-Type") == "text/event-stream" | ||||
| 	if isStream { | ||||
| 		scanner := bufio.NewScanner(resp.Body) | ||||
| 		scanner.Split(func(data []byte, atEOF bool) (advance int, token []byte, err error) { | ||||
| 			if atEOF && len(data) == 0 { | ||||
| 				return 0, nil, nil | ||||
| 			} | ||||
|  | ||||
| 			if i := strings.Index(string(data), "\n\n"); i >= 0 { | ||||
| 				return i + 2, data[0:i], nil | ||||
| 			} | ||||
|  | ||||
| 			if atEOF { | ||||
| 				return len(data), data, nil | ||||
| 			} | ||||
|  | ||||
| 			return 0, nil, nil | ||||
| 		}) | ||||
| 		dataChan := make(chan string) | ||||
| 		stopChan := make(chan bool) | ||||
| 		go func() { | ||||
| 			for scanner.Scan() { | ||||
| 				data := scanner.Text() | ||||
| 				dataChan <- data | ||||
| 			} | ||||
| 			stopChan <- true | ||||
| 		}() | ||||
| 		c.Writer.Header().Set("Content-Type", "text/event-stream") | ||||
| 		c.Writer.Header().Set("Cache-Control", "no-cache") | ||||
| 		c.Writer.Header().Set("Connection", "keep-alive") | ||||
| 		c.Writer.Header().Set("Transfer-Encoding", "chunked") | ||||
| 		c.Stream(func(w io.Writer) bool { | ||||
| 			select { | ||||
| 			case data := <-dataChan: | ||||
| 				c.Render(-1, common.CustomEvent{Data: data}) | ||||
| 				return true | ||||
| 			case <-stopChan: | ||||
| 				return false | ||||
| 			} | ||||
| 		}) | ||||
| 		return | ||||
| 	} else { | ||||
| 		for k, v := range resp.Header { | ||||
| 			c.Writer.Header().Set(k, v[0]) | ||||
| 		} | ||||
| 		_, 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 | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|   | ||||
| @@ -76,6 +76,7 @@ func GetToken(c *gin.Context) { | ||||
| } | ||||
|  | ||||
| func AddToken(c *gin.Context) { | ||||
| 	isAdmin := c.GetInt("role") >= common.RoleAdminUser | ||||
| 	token := model.Token{} | ||||
| 	err := c.ShouldBindJSON(&token) | ||||
| 	if err != nil { | ||||
| @@ -98,6 +99,24 @@ func AddToken(c *gin.Context) { | ||||
| 		Key:          common.GetUUID(), | ||||
| 		CreatedTime:  common.GetTimestamp(), | ||||
| 		AccessedTime: common.GetTimestamp(), | ||||
| 		ExpiredTime:  token.ExpiredTime, | ||||
| 	} | ||||
| 	if isAdmin { | ||||
| 		cleanToken.RemainTimes = token.RemainTimes | ||||
| 		cleanToken.UnlimitedTimes = token.UnlimitedTimes | ||||
| 	} 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 | ||||
| 		} | ||||
| 		if quota > 0 { | ||||
| 			cleanToken.RemainTimes = quota | ||||
| 		} | ||||
| 	} | ||||
| 	err = cleanToken.Insert() | ||||
| 	if err != nil { | ||||
| @@ -107,6 +126,10 @@ func AddToken(c *gin.Context) { | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	if !isAdmin { | ||||
| 		// update user quota | ||||
| 		err = model.DecreaseUserQuota(c.GetInt("id"), cleanToken.RemainTimes) | ||||
| 	} | ||||
| 	c.JSON(http.StatusOK, gin.H{ | ||||
| 		"success": true, | ||||
| 		"message": "", | ||||
| @@ -133,7 +156,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 { | ||||
| @@ -151,8 +176,33 @@ func UpdateToken(c *gin.Context) { | ||||
| 		}) | ||||
| 		return | ||||
| 	} | ||||
| 	cleanToken.Name = token.Name | ||||
| 	cleanToken.Status = token.Status | ||||
| 	if token.Status == common.TokenStatusEnabled { | ||||
| 		if cleanToken.Status == common.TokenStatusExpired && cleanToken.ExpiredTime <= common.GetTimestamp() { | ||||
| 			c.JSON(http.StatusOK, gin.H{ | ||||
| 				"success": false, | ||||
| 				"message": "令牌已过期,无法启用,请先修改令牌过期时间", | ||||
| 			}) | ||||
| 			return | ||||
| 		} | ||||
| 		if cleanToken.Status == common.TokenStatusExhausted && cleanToken.RemainTimes <= 0 && !cleanToken.UnlimitedTimes { | ||||
| 			c.JSON(http.StatusOK, gin.H{ | ||||
| 				"success": false, | ||||
| 				"message": "令牌可用次数已用尽,无法启用,请先修改令牌剩余次数,或者设置为无限次数", | ||||
| 			}) | ||||
| 			return | ||||
| 		} | ||||
| 	} | ||||
| 	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.RemainTimes = token.RemainTimes | ||||
| 			cleanToken.UnlimitedTimes = token.UnlimitedTimes | ||||
| 		} | ||||
| 	} | ||||
| 	err = cleanToken.Update() | ||||
| 	if err != nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| @@ -168,3 +218,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) | ||||
| @@ -523,6 +559,13 @@ func ManageUser(c *gin.Context) { | ||||
| 		} | ||||
| 		user.Role = common.RoleAdminUser | ||||
| 	case "demote": | ||||
| 		if user.Role == common.RoleRootUser { | ||||
| 			c.JSON(http.StatusOK, gin.H{ | ||||
| 				"success": false, | ||||
| 				"message": "无法降级超级管理员用户", | ||||
| 			}) | ||||
| 			return | ||||
| 		} | ||||
| 		user.Role = common.RoleCommonUser | ||||
| 	} | ||||
|  | ||||
|   | ||||
							
								
								
									
										36
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										36
									
								
								go.mod
									
									
									
									
									
								
							| @@ -8,11 +8,11 @@ require ( | ||||
| 	github.com/gin-contrib/gzip v0.0.6 | ||||
| 	github.com/gin-contrib/sessions v0.0.5 | ||||
| 	github.com/gin-contrib/static v0.0.1 | ||||
| 	github.com/gin-gonic/gin v1.8.1 | ||||
| 	github.com/go-playground/validator/v10 v10.11.1 | ||||
| 	github.com/gin-gonic/gin v1.9.0 | ||||
| 	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 | ||||
| 	golang.org/x/crypto v0.1.0 | ||||
| 	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 | ||||
| @@ -21,13 +21,15 @@ require ( | ||||
|  | ||||
| require ( | ||||
| 	github.com/boj/redistore v0.0.0-20180917114910-cd5dcc76aeff // indirect | ||||
| 	github.com/bytedance/sonic v1.8.8 // indirect | ||||
| 	github.com/cespare/xxhash/v2 v2.1.2 // indirect | ||||
| 	github.com/chenzhuoyu/base64x v0.0.0-20221115062448-fe3a3abad311 // indirect | ||||
| 	github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f // indirect | ||||
| 	github.com/gin-contrib/sse v0.1.0 // indirect | ||||
| 	github.com/go-playground/locales v0.14.0 // indirect | ||||
| 	github.com/go-playground/universal-translator v0.18.0 // indirect | ||||
| 	github.com/go-playground/locales v0.14.1 // indirect | ||||
| 	github.com/go-playground/universal-translator v0.18.1 // indirect | ||||
| 	github.com/go-sql-driver/mysql v1.6.0 // indirect | ||||
| 	github.com/goccy/go-json v0.9.7 // indirect | ||||
| 	github.com/goccy/go-json v0.10.2 // indirect | ||||
| 	github.com/gomodule/redigo v2.0.0+incompatible // indirect | ||||
| 	github.com/gorilla/context v1.1.1 // indirect | ||||
| 	github.com/gorilla/securecookie v1.1.1 // indirect | ||||
| @@ -35,17 +37,21 @@ require ( | ||||
| 	github.com/jinzhu/inflection v1.0.0 // indirect | ||||
| 	github.com/jinzhu/now v1.1.5 // indirect | ||||
| 	github.com/json-iterator/go v1.1.12 // indirect | ||||
| 	github.com/leodido/go-urn v1.2.1 // indirect | ||||
| 	github.com/mattn/go-isatty v0.0.14 // indirect | ||||
| 	github.com/klauspost/cpuid/v2 v2.2.4 // indirect | ||||
| 	github.com/leodido/go-urn v1.2.3 // indirect | ||||
| 	github.com/mattn/go-isatty v0.0.18 // indirect | ||||
| 	github.com/mattn/go-sqlite3 v2.0.3+incompatible // indirect | ||||
| 	github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421 // indirect | ||||
| 	github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd // indirect | ||||
| 	github.com/modern-go/reflect2 v1.0.2 // indirect | ||||
| 	github.com/pelletier/go-toml/v2 v2.0.1 // indirect | ||||
| 	github.com/ugorji/go/codec v1.2.7 // indirect | ||||
| 	golang.org/x/net v0.7.0 // indirect | ||||
| 	golang.org/x/sys v0.5.0 // indirect | ||||
| 	golang.org/x/text v0.7.0 // indirect | ||||
| 	google.golang.org/protobuf v1.28.0 // indirect | ||||
| 	github.com/pelletier/go-toml/v2 v2.0.7 // indirect | ||||
| 	github.com/twitchyliquid64/golang-asm v0.15.1 // indirect | ||||
| 	github.com/ugorji/go/codec v1.2.11 // indirect | ||||
| 	golang.org/x/arch v0.3.0 // indirect | ||||
| 	golang.org/x/net v0.9.0 // indirect | ||||
| 	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 | ||||
| ) | ||||
|   | ||||
							
								
								
									
										51
									
								
								go.sum
									
									
									
									
									
								
							
							
						
						
									
										51
									
								
								go.sum
									
									
									
									
									
								
							| @@ -1,7 +1,13 @@ | ||||
| github.com/boj/redistore v0.0.0-20180917114910-cd5dcc76aeff h1:RmdPFa+slIr4SCBg4st/l/vZWVe9QJKMXGO60Bxbe04= | ||||
| github.com/boj/redistore v0.0.0-20180917114910-cd5dcc76aeff/go.mod h1:+RTT1BOk5P97fT2CiHkbFQwkK3mjsFAP6zCYV2aXtjw= | ||||
| github.com/bytedance/sonic v1.5.0/go.mod h1:ED5hyg4y6t3/9Ku1R6dU/4KyJ48DZ4jPhfY1O2AihPM= | ||||
| github.com/bytedance/sonic v1.8.8 h1:Kj4AYbZSeENfyXicsYppYKO0K2YWab+i2UTSY7Ukz9Q= | ||||
| github.com/bytedance/sonic v1.8.8/go.mod h1:i736AoUSYt75HyZLoJW9ERYxcy6eaN6h4BZXU064P/U= | ||||
| github.com/cespare/xxhash/v2 v2.1.2 h1:YRXhKfTDauu4ajMg1TPgFO5jnlC2HCbmLXMcTG5cbYE= | ||||
| github.com/cespare/xxhash/v2 v2.1.2/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= | ||||
| github.com/chenzhuoyu/base64x v0.0.0-20211019084208-fb5309c8db06/go.mod h1:DH46F32mSOjUmXrMHnKwZdA8wcEefY7UVqBKYGjpdQY= | ||||
| github.com/chenzhuoyu/base64x v0.0.0-20221115062448-fe3a3abad311 h1:qSGYFH7+jGhDF8vLC+iwCD4WpbV1EBDSzWkJODFLams= | ||||
| github.com/chenzhuoyu/base64x v0.0.0-20221115062448-fe3a3abad311/go.mod h1:b583jCggY9gE99b6G5LEC39OIiVsWj+R97kbl5odCEk= | ||||
| github.com/creack/pty v1.1.9/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ33E= | ||||
| github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | ||||
| github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= | ||||
| @@ -22,24 +28,34 @@ github.com/gin-contrib/static v0.0.1/go.mod h1:CSxeF+wep05e0kCOsqWdAWbSszmc31zTI | ||||
| 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/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= | ||||
| github.com/golang/protobuf v1.3.3/go.mod h1:vzj43D7+SQXF/4pzW/hwtAqwc6iTitCiVSaWz5lYuqw= | ||||
| github.com/golang/protobuf v1.5.0/go.mod h1:FsONVRAS9T7sI+LIUmWTfcYkHO4aIWwzhcaSAoJOfIk= | ||||
| github.com/gomodule/redigo v2.0.0+incompatible h1:K/R+8tc58AaqLkqG2Ol3Qk+DR/TlNuhuh457pBFPtt0= | ||||
| @@ -64,6 +80,9 @@ github.com/jinzhu/now v1.1.5/go.mod h1:d3SSVoowX0Lcu0IBviAWJpolVfI5UJVZZ7cO71lE/ | ||||
| github.com/json-iterator/go v1.1.9/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4= | ||||
| github.com/json-iterator/go v1.1.12 h1:PV8peI4a0ysnczrg+LtxykD8LfKY9ML6u2jnxaEnrnM= | ||||
| github.com/json-iterator/go v1.1.12/go.mod h1:e30LSqwooZae/UwlEbR2852Gd8hjQvJoHmT4TnhNGBo= | ||||
| github.com/klauspost/cpuid/v2 v2.0.9/go.mod h1:FInQzS24/EEf25PyTYn52gqo7WaD8xa0213Md/qVLRg= | ||||
| github.com/klauspost/cpuid/v2 v2.2.4 h1:acbojRNwl3o09bUq+yDCtZFc1aiwaAAxtcn8YkZXnvk= | ||||
| github.com/klauspost/cpuid/v2 v2.2.4/go.mod h1:RVVoqg1df56z8g3pUjL/3lE5UfnlrJX8tyFgg4nqhuY= | ||||
| github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= | ||||
| github.com/kr/pretty v0.2.1/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI= | ||||
| github.com/kr/pretty v0.3.0 h1:WgNl7dwNpEZ6jJ9k1snq4pZsg7DOEN8hP9Xw0Tsjwk0= | ||||
| @@ -75,14 +94,20 @@ 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= | ||||
| github.com/modern-go/reflect2 v0.0.0-20180701023420-4b7aa43c6742/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0= | ||||
| github.com/modern-go/reflect2 v1.0.2 h1:xBagoLtFs94CBntxluKeaWgTMpvLxC4ur3nMaC9Gz0M= | ||||
| github.com/modern-go/reflect2 v1.0.2/go.mod h1:yWuevngMOJpCy52FWWMvUC8ws7m/LJsjYzDa0/r8luk= | ||||
| @@ -91,6 +116,8 @@ 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/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= | ||||
| github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= | ||||
| @@ -99,6 +126,7 @@ github.com/rogpeppe/go-internal v1.8.0 h1:FCbCCtXNOY3UtUuHUYaghJg4y7Fd14rXifAYUA | ||||
| github.com/rogpeppe/go-internal v1.8.0/go.mod h1:WmiCO8CzOY8rg0OYDC4/i/2WRWAB6poM+XZ2dLUbcbE= | ||||
| github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= | ||||
| github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw= | ||||
| github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo= | ||||
| github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= | ||||
| github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4= | ||||
| github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= | ||||
| @@ -106,27 +134,45 @@ github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/ | ||||
| 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= | ||||
| github.com/stretchr/testify v1.8.2/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4= | ||||
| github.com/twitchyliquid64/golang-asm v0.15.1 h1:SU5vSMR7hnwNxj24w34ZyCi/FmDZTkS4MhqMhdFk5YI= | ||||
| github.com/twitchyliquid64/golang-asm v0.15.1/go.mod h1:a1lVb/DtPvCB8fslRZhAngC2+aY1QWCk3Cedj/Gdt08= | ||||
| github.com/ugorji/go v1.1.7/go.mod h1:kZn38zHttfInRq0xu/PH0az30d+z6vm202qpg1oXVMw= | ||||
| github.com/ugorji/go v1.2.7/go.mod h1:nF9osbDWLy6bDVv/Rtoh6QgnvNDpmCalQV5urGCCS6M= | ||||
| github.com/ugorji/go/codec v1.1.7/go.mod h1:Ax+UKWsSmolVDwsd+7N3ZtXu+yMGCf907BLYF3GoBXY= | ||||
| github.com/ugorji/go/codec v1.2.7 h1:YPXUKf7fYbp/y8xloBqZOw2qaVggbfwMlI8WM3wZUJ0= | ||||
| github.com/ugorji/go/codec v1.2.7/go.mod h1:WGN1fab3R1fzQlVQTkfxVtIBhWDRqOviHU95kRgeqEY= | ||||
| github.com/ugorji/go/codec v1.2.11 h1:BMaWp1Bb6fHwEtbplGBGJ498wD+LKlNSl25MjdZY4dU= | ||||
| github.com/ugorji/go/codec v1.2.11/go.mod h1:UNopzCgEMSXjBc6AOMqYvWC1ktqTAfzJZUZgYf6w6lg= | ||||
| golang.org/x/arch v0.0.0-20210923205945-b76863e36670/go.mod h1:5om86z9Hs0C8fWVUuoMHwpExlXzs5Tkyp9hOrfG7pp8= | ||||
| golang.org/x/arch v0.3.0 h1:02VY4/ZcO/gBOH6PUaoiptASxtXU10jazRCP865E97k= | ||||
| golang.org/x/arch v0.3.0/go.mod h1:5om86z9Hs0C8fWVUuoMHwpExlXzs5Tkyp9hOrfG7pp8= | ||||
| golang.org/x/crypto v0.0.0-20210711020723-a769d52b0f97/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= | ||||
| golang.org/x/crypto v0.0.0-20211215153901-e495a2d5b3d3/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4= | ||||
| golang.org/x/crypto v0.1.0 h1:MDRAIl0xIo9Io2xV565hzXHw3zVseKrJKodhohM5CjU= | ||||
| golang.org/x/crypto v0.1.0/go.mod h1:RecgLatLF4+eUMCP1PoPZQb+cVrJcOPbHkTkbkB9sbw= | ||||
| golang.org/x/crypto v0.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= | ||||
| golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= | ||||
| golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk= | ||||
| golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||
| @@ -134,12 +180,16 @@ 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= | ||||
| @@ -165,3 +215,4 @@ gorm.io/driver/sqlite v1.4.3/go.mod h1:0Aq3iPO+v9ZKbcdiz8gLWRw5VOPcBOPUQJFLq5e2e | ||||
| gorm.io/gorm v1.23.8/go.mod h1:l2lP/RyAtc1ynaTjFksBde/O8v9oOGIApu2/xRitmZk= | ||||
| gorm.io/gorm v1.24.0 h1:j/CoiSm6xpRpmzbFJsQHYj+I8bGYWLXVHeYEyyKlF74= | ||||
| gorm.io/gorm v1.24.0/go.mod h1:DVrVomtaYTbqs7gB/x2uVvqnXzv0nqjB396B8cG4dBA= | ||||
| rsc.io/pdf v0.1.1/go.mod h1:n8OzWcQ6Sp37PL01nO98y4iUCRdTGarVfzxY20ICaU4= | ||||
|   | ||||
							
								
								
									
										4
									
								
								main.go
									
									
									
									
									
								
							
							
						
						
									
										4
									
								
								main.go
									
									
									
									
									
								
							| @@ -2,7 +2,6 @@ package main | ||||
|  | ||||
| import ( | ||||
| 	"embed" | ||||
| 	"github.com/gin-contrib/gzip" | ||||
| 	"github.com/gin-contrib/sessions" | ||||
| 	"github.com/gin-contrib/sessions/cookie" | ||||
| 	"github.com/gin-contrib/sessions/redis" | ||||
| @@ -51,7 +50,8 @@ func main() { | ||||
|  | ||||
| 	// Initialize HTTP server | ||||
| 	server := gin.Default() | ||||
| 	server.Use(gzip.Gzip(gzip.DefaultCompression)) | ||||
| 	// This will cause SSE not to work!!! | ||||
| 	//server.Use(gzip.Gzip(gzip.DefaultCompression)) | ||||
| 	server.Use(middleware.CORS()) | ||||
|  | ||||
| 	// Initialize session store | ||||
|   | ||||
| @@ -16,12 +16,31 @@ func authHelper(c *gin.Context, minRole int) { | ||||
| 	id := session.Get("id") | ||||
| 	status := session.Get("status") | ||||
| 	if username == nil { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| 			"success": false, | ||||
| 			"message": "无权进行此操作,未登录", | ||||
| 		}) | ||||
| 		c.Abort() | ||||
| 		return | ||||
| 		// Check access token | ||||
| 		accessToken := c.Request.Header.Get("Authorization") | ||||
| 		if accessToken == "" { | ||||
| 			c.JSON(http.StatusUnauthorized, gin.H{ | ||||
| 				"success": false, | ||||
| 				"message": "无权进行此操作,未登录且未提供 access token", | ||||
| 			}) | ||||
| 			c.Abort() | ||||
| 			return | ||||
| 		} | ||||
| 		user := model.ValidateAccessToken(accessToken) | ||||
| 		if user != nil && user.Username != "" { | ||||
| 			// Token is valid | ||||
| 			username = user.Username | ||||
| 			role = user.Role | ||||
| 			id = user.Id | ||||
| 			status = user.Status | ||||
| 		} else { | ||||
| 			c.JSON(http.StatusOK, gin.H{ | ||||
| 				"success": false, | ||||
| 				"message": "无权进行此操作,access token 无效", | ||||
| 			}) | ||||
| 			c.Abort() | ||||
| 			return | ||||
| 		} | ||||
| 	} | ||||
| 	if status.(int) == common.UserStatusDisabled { | ||||
| 		c.JSON(http.StatusOK, gin.H{ | ||||
| @@ -79,9 +98,32 @@ func TokenAuth() func(c *gin.Context) { | ||||
| 			c.Abort() | ||||
| 			return | ||||
| 		} | ||||
| 		if !model.IsUserEnabled(token.UserId) { | ||||
| 			c.JSON(http.StatusOK, gin.H{ | ||||
| 				"error": gin.H{ | ||||
| 					"message": "用户已被封禁", | ||||
| 					"type":    "one_api_error", | ||||
| 				}, | ||||
| 			}) | ||||
| 			c.Abort() | ||||
| 			return | ||||
| 		} | ||||
| 		c.Set("id", token.UserId) | ||||
| 		c.Set("token_id", token.Id) | ||||
| 		c.Set("unlimited_times", token.UnlimitedTimes) | ||||
| 		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) | ||||
| } | ||||
|   | ||||
| @@ -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 { | ||||
|   | ||||
| @@ -46,6 +46,8 @@ func InitOptionMap() { | ||||
| 	common.OptionMap["WeChatAccountQRCodeImageURL"] = "" | ||||
| 	common.OptionMap["TurnstileSiteKey"] = "" | ||||
| 	common.OptionMap["TurnstileSecretKey"] = "" | ||||
| 	common.OptionMap["QuotaForNewUser"] = strconv.Itoa(common.QuotaForNewUser) | ||||
| 	common.OptionMap["TopUpLink"] = common.TopUpLink | ||||
| 	common.OptionMapRWMutex.Unlock() | ||||
| 	options, _ := AllOption() | ||||
| 	for _, option := range options { | ||||
| @@ -131,5 +133,9 @@ func updateOptionMap(key string, value string) { | ||||
| 		common.TurnstileSiteKey = value | ||||
| 	case "TurnstileSecretKey": | ||||
| 		common.TurnstileSecretKey = value | ||||
| 	case "QuotaForNewUser": | ||||
| 		common.QuotaForNewUser, _ = strconv.Atoi(value) | ||||
| 	case "TopUpLink": | ||||
| 		common.TopUpLink = value | ||||
| 	} | ||||
| } | ||||
|   | ||||
							
								
								
									
										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 = TopUpToken(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,18 +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"` | ||||
| 	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 | ||||
| 	RemainTimes    int    `json:"remain_times" gorm:"default:0"` | ||||
| 	UnlimitedTimes bool   `json:"unlimited_times" gorm:"default:false"` | ||||
| } | ||||
|  | ||||
| func GetAllUserTokens(userId int, startIdx int, num int) ([]*Token, error) { | ||||
| @@ -35,21 +39,37 @@ 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 已被禁用") | ||||
| 			return nil, errors.New("该 token 状态不可用") | ||||
| 		} | ||||
| 		if token.ExpiredTime != -1 && token.ExpiredTime < common.GetTimestamp() { | ||||
| 			token.Status = common.TokenStatusExpired | ||||
| 			err := token.SelectUpdate() | ||||
| 			if err != nil { | ||||
| 				common.SysError("更新 token 状态失败:" + err.Error()) | ||||
| 			} | ||||
| 			return nil, errors.New("该 token 已过期") | ||||
| 		} | ||||
| 		if !token.UnlimitedTimes && token.RemainTimes <= 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() | ||||
| 			err := token.Update() | ||||
| 			err := token.SelectUpdate() | ||||
| 			if err != nil { | ||||
| 				common.SysError("更新 token 访问时间失败:" + err.Error()) | ||||
| 				common.SysError("更新 token 失败:" + err.Error()) | ||||
| 			} | ||||
| 		}() | ||||
| 		return token, nil | ||||
| 	} | ||||
| 	return nil, err | ||||
| 	return nil, errors.New("无效的 token") | ||||
| } | ||||
|  | ||||
| func GetTokenByIds(id int, userId int) (*Token, error) { | ||||
| @@ -68,12 +88,18 @@ 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_times", "unlimited_times").Updates(token).Error | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func (token *Token) SelectUpdate() error { | ||||
| 	// This can update zero values | ||||
| 	return DB.Model(token).Select("accessed_time", "status").Updates(token).Error | ||||
| } | ||||
|  | ||||
| func (token *Token) Delete() error { | ||||
| 	var err error | ||||
| 	err = DB.Delete(token).Error | ||||
| @@ -92,3 +118,13 @@ func DeleteTokenById(id int, userId int) (err error) { | ||||
| 	} | ||||
| 	return token.Delete() | ||||
| } | ||||
|  | ||||
| func DecreaseTokenRemainTimesById(id int) (err error) { | ||||
| 	err = DB.Model(&Token{}).Where("id = ?", id).Update("remain_times", gorm.Expr("remain_times - ?", 1)).Error | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func TopUpToken(id int, times int) (err error) { | ||||
| 	err = DB.Model(&Token{}).Where("id = ?", id).Update("remain_times", gorm.Expr("remain_times + ?", times)).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,51 @@ 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 DecreaseUserQuota(id int, quota int) (err error) { | ||||
| 	err = DB.Model(&User{}).Where("id = ?", id).Update("quota", gorm.Expr("quota - ?", quota)).Error | ||||
| 	return err | ||||
| } | ||||
|   | ||||
| @@ -1,6 +1,7 @@ | ||||
| package router | ||||
|  | ||||
| import ( | ||||
| 	"github.com/gin-contrib/gzip" | ||||
| 	"github.com/gin-gonic/gin" | ||||
| 	"one-api/controller" | ||||
| 	"one-api/middleware" | ||||
| @@ -8,6 +9,7 @@ import ( | ||||
|  | ||||
| func SetApiRouter(router *gin.Engine) { | ||||
| 	apiRouter := router.Group("/api") | ||||
| 	apiRouter.Use(gzip.Gzip(gzip.DefaultCompression)) | ||||
| 	apiRouter.Use(middleware.GlobalAPIRateLimit()) | ||||
| 	{ | ||||
| 		apiRouter.GET("/status", controller.GetStatus) | ||||
| @@ -33,6 +35,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("/") | ||||
| @@ -68,10 +71,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) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|   | ||||
| @@ -7,9 +7,14 @@ import ( | ||||
| ) | ||||
|  | ||||
| func SetRelayRouter(router *gin.Engine) { | ||||
| 	relayRouter := router.Group("/v1") | ||||
| 	relayRouter.Use(middleware.GlobalAPIRateLimit(), middleware.TokenAuth(), middleware.Distribute()) | ||||
| 	relayV1Router := router.Group("/v1") | ||||
| 	relayV1Router.Use(middleware.TokenAuth(), middleware.Distribute()) | ||||
| 	{ | ||||
| 		relayRouter.POST("/chat/completions", controller.Relay) | ||||
| 		relayV1Router.Any("/*path", controller.Relay) | ||||
| 	} | ||||
| 	relayDashboardRouter := router.Group("/dashboard") | ||||
| 	relayDashboardRouter.Use(middleware.TokenAuth(), middleware.Distribute()) | ||||
| 	{ | ||||
| 		relayDashboardRouter.Any("/*path", controller.Relay) | ||||
| 	} | ||||
| } | ||||
|   | ||||
| @@ -2,6 +2,7 @@ package router | ||||
|  | ||||
| import ( | ||||
| 	"embed" | ||||
| 	"github.com/gin-contrib/gzip" | ||||
| 	"github.com/gin-contrib/static" | ||||
| 	"github.com/gin-gonic/gin" | ||||
| 	"net/http" | ||||
| @@ -10,6 +11,7 @@ import ( | ||||
| ) | ||||
|  | ||||
| func setWebRouter(router *gin.Engine, buildFS embed.FS, indexPage []byte) { | ||||
| 	router.Use(gzip.Gzip(gzip.DefaultCompression)) | ||||
| 	router.Use(middleware.GlobalWebRateLimit()) | ||||
| 	router.Use(middleware.Cache()) | ||||
| 	router.Use(static.Serve("/", common.EmbedFolder(buildFS, "web/build"))) | ||||
|   | ||||
| @@ -18,9 +18,10 @@ import { StatusContext } from './context/Status'; | ||||
| import Channel from './pages/Channel'; | ||||
| import Token from './pages/Token'; | ||||
| import EditToken from './pages/Token/EditToken'; | ||||
| import AddToken from './pages/Token/AddToken'; | ||||
| 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')); | ||||
| @@ -116,7 +117,31 @@ function App() { | ||||
|         path='/token/add' | ||||
|         element={ | ||||
|           <Suspense fallback={<Loading></Loading>}> | ||||
|             <AddToken /> | ||||
|             <EditToken /> | ||||
|           </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> | ||||
|         } | ||||
|       /> | ||||
|   | ||||
| @@ -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); | ||||
|   | ||||
| @@ -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; | ||||
| @@ -24,6 +24,8 @@ const SystemSetting = () => { | ||||
|     TurnstileSiteKey: '', | ||||
|     TurnstileSecretKey: '', | ||||
|     RegisterEnabled: '', | ||||
|     QuotaForNewUser: 0, | ||||
|     TopUpLink: '' | ||||
|   }); | ||||
|   let originInputs = {}; | ||||
|   let [loading, setLoading] = useState(false); | ||||
| @@ -64,7 +66,7 @@ const SystemSetting = () => { | ||||
|     } | ||||
|     const res = await API.put('/api/option', { | ||||
|       key, | ||||
|       value, | ||||
|       value | ||||
|     }); | ||||
|     const { success, message } = res.data; | ||||
|     if (success) { | ||||
| @@ -86,7 +88,9 @@ const SystemSetting = () => { | ||||
|       name === 'WeChatServerToken' || | ||||
|       name === 'WeChatAccountQRCodeImageURL' || | ||||
|       name === 'TurnstileSiteKey' || | ||||
|       name === 'TurnstileSecretKey' | ||||
|       name === 'TurnstileSecretKey' || | ||||
|       name === 'QuotaForNewUser' || | ||||
|       name === 'TopUpLink' | ||||
|     ) { | ||||
|       setInputs((inputs) => ({ ...inputs, [name]: value })); | ||||
|     } else { | ||||
| @@ -99,6 +103,15 @@ const SystemSetting = () => { | ||||
|     await updateOption('ServerAddress', ServerAddress); | ||||
|   }; | ||||
|  | ||||
|   const submitOperationConfig = async () => { | ||||
|     if (originInputs['QuotaForNewUser'] !== inputs.QuotaForNewUser) { | ||||
|       await updateOption('QuotaForNewUser', inputs.QuotaForNewUser); | ||||
|     } | ||||
|     if (originInputs['TopUpLink'] !== inputs.TopUpLink) { | ||||
|       await updateOption('TopUpLink', inputs.TopUpLink); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   const submitSMTP = async () => { | ||||
|     if (originInputs['SMTPServer'] !== inputs.SMTPServer) { | ||||
|       await updateOption('SMTPServer', inputs.SMTPServer); | ||||
| @@ -228,6 +241,32 @@ const SystemSetting = () => { | ||||
|             /> | ||||
|           </Form.Group> | ||||
|           <Divider /> | ||||
|           <Header as='h3'> | ||||
|             运营设置 | ||||
|           </Header> | ||||
|           <Form.Group widths={3}> | ||||
|             <Form.Input | ||||
|               label='新用户初始配额' | ||||
|               name='QuotaForNewUser' | ||||
|               onChange={handleInputChange} | ||||
|               autoComplete='off' | ||||
|               value={inputs.QuotaForNewUser} | ||||
|               type='number' | ||||
|               min='0' | ||||
|               placeholder='例如:100' | ||||
|             /> | ||||
|             <Form.Input | ||||
|               label='充值链接' | ||||
|               name='TopUpLink' | ||||
|               onChange={handleInputChange} | ||||
|               autoComplete='off' | ||||
|               value={inputs.TopUpLink} | ||||
|               type='link' | ||||
|               placeholder='例如发卡网站的购买链接' | ||||
|             /> | ||||
|           </Form.Group> | ||||
|           <Form.Button onClick={submitOperationConfig}>保存运营设置</Form.Button> | ||||
|           <Divider /> | ||||
|           <Header as='h3'> | ||||
|             配置 SMTP | ||||
|             <Header.Subheader>用以支持系统的邮件发送</Header.Subheader> | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| import React, { useEffect, useState } from 'react'; | ||||
| import { Button, Form, Label, 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, showSuccess, timestamp2string } from '../helpers'; | ||||
| import { API, copy, showError, showSuccess, showWarning, timestamp2string } from '../helpers'; | ||||
|  | ||||
| import { ITEMS_PER_PAGE } from '../constants'; | ||||
|  | ||||
| @@ -13,12 +13,31 @@ function renderTimestamp(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='yellow'> 已过期 </Label>; | ||||
|     case 4: | ||||
|       return <Label basic color='grey'> 已耗尽 </Label>; | ||||
|     default: | ||||
|       return <Label basic color='black'> 未知状态 </Label>; | ||||
|   } | ||||
| } | ||||
|  | ||||
| const TokensTable = () => { | ||||
|   const [tokens, setTokens] = useState([]); | ||||
|   const [loading, setLoading] = useState(true); | ||||
|   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}`); | ||||
| @@ -53,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) => { | ||||
| @@ -64,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; | ||||
| @@ -88,25 +114,6 @@ const TokensTable = () => { | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   const renderStatus = (status) => { | ||||
|     switch (status) { | ||||
|       case 1: | ||||
|         return <Label basic color='green'>已启用</Label>; | ||||
|       case 2: | ||||
|         return ( | ||||
|           <Label basic color='red'> | ||||
|             已禁用 | ||||
|           </Label> | ||||
|         ); | ||||
|       default: | ||||
|         return ( | ||||
|           <Label basic color='grey'> | ||||
|             未知状态 | ||||
|           </Label> | ||||
|         ); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   const searchTokens = async () => { | ||||
|     if (searchKeyword === '') { | ||||
|       // if keyword is blank, load files instead. | ||||
| @@ -144,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_times += data; | ||||
|       setTokens(newTokens); | ||||
|       setRedemptionCode(''); | ||||
|       setShowTopUpModal(false); | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   return ( | ||||
|     <> | ||||
|       <Form onSubmit={searchTokens}> | ||||
| @@ -185,6 +214,14 @@ const TokensTable = () => { | ||||
|             > | ||||
|               状态 | ||||
|             </Table.HeaderCell> | ||||
|             <Table.HeaderCell | ||||
|               style={{ cursor: 'pointer' }} | ||||
|               onClick={() => { | ||||
|                 sortToken('remain_times'); | ||||
|               }} | ||||
|             > | ||||
|               剩余次数 | ||||
|             </Table.HeaderCell> | ||||
|             <Table.HeaderCell | ||||
|               style={{ cursor: 'pointer' }} | ||||
|               onClick={() => { | ||||
| @@ -196,10 +233,10 @@ const TokensTable = () => { | ||||
|             <Table.HeaderCell | ||||
|               style={{ cursor: 'pointer' }} | ||||
|               onClick={() => { | ||||
|                 sortToken('accessed_time'); | ||||
|                 sortToken('expired_time'); | ||||
|               }} | ||||
|             > | ||||
|               访问时间 | ||||
|               过期时间 | ||||
|             </Table.HeaderCell> | ||||
|             <Table.HeaderCell>操作</Table.HeaderCell> | ||||
|           </Table.Row> | ||||
| @@ -218,8 +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.unlimited_times ? '无限制' : token.remain_times}</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> | ||||
|                     <div> | ||||
|                       <Button | ||||
| @@ -229,12 +267,22 @@ const TokensTable = () => { | ||||
|                           if (await copy(token.key)) { | ||||
|                             showSuccess('已复制到剪贴板!'); | ||||
|                           } else { | ||||
|                             showError('复制失败!'); | ||||
|                             showWarning('无法复制到剪贴板,请手动复制,已将令牌填入搜索框。'); | ||||
|                             setSearchKeyword(token.key); | ||||
|                           } | ||||
|                         }} | ||||
|                       > | ||||
|                         复制 | ||||
|                       </Button> | ||||
|                       <Button | ||||
|                         size={'small'} | ||||
|                         color={'yellow'} | ||||
|                         onClick={() => { | ||||
|                           setTargetTokenIdx(idx); | ||||
|                           setShowTopUpModal(true); | ||||
|                         }}> | ||||
|                         充值 | ||||
|                       </Button> | ||||
|                       <Button | ||||
|                         size={'small'} | ||||
|                         negative | ||||
| @@ -272,7 +320,7 @@ const TokensTable = () => { | ||||
|  | ||||
|         <Table.Footer> | ||||
|           <Table.Row> | ||||
|             <Table.HeaderCell colSpan='6'> | ||||
|             <Table.HeaderCell colSpan='8'> | ||||
|               <Button size='small' as={Link} to='/token/add' loading={loading}> | ||||
|                 添加新的令牌 | ||||
|               </Button> | ||||
| @@ -291,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> | ||||
|     </> | ||||
|   ); | ||||
| }; | ||||
|   | ||||
| @@ -2,5 +2,6 @@ export const toastConstants = { | ||||
|   SUCCESS_TIMEOUT: 500, | ||||
|   INFO_TIMEOUT: 3000, | ||||
|   ERROR_TIMEOUT: 5000, | ||||
|   WARNING_TIMEOUT: 10000, | ||||
|   NOTICE_TIMEOUT: 20000 | ||||
| }; | ||||
|   | ||||
| @@ -31,6 +31,7 @@ export function isMobile() { | ||||
| } | ||||
|  | ||||
| let showErrorOptions = { autoClose: toastConstants.ERROR_TIMEOUT }; | ||||
| let showWarningOptions = { autoClose: toastConstants.WARNING_TIMEOUT }; | ||||
| let showSuccessOptions = { autoClose: toastConstants.SUCCESS_TIMEOUT }; | ||||
| let showInfoOptions = { autoClose: toastConstants.INFO_TIMEOUT }; | ||||
| let showNoticeOptions = { autoClose: false }; | ||||
| @@ -53,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: | ||||
| @@ -74,6 +79,10 @@ export function showError(error) { | ||||
|   } | ||||
| } | ||||
|  | ||||
| export function showWarning(message) { | ||||
|   toast.warn(message, showWarningOptions); | ||||
| } | ||||
|  | ||||
| export function showSuccess(message) { | ||||
|   toast.success(message, showSuccessOptions); | ||||
| } | ||||
| @@ -135,3 +144,12 @@ 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(); | ||||
| } | ||||
							
								
								
									
										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='off'> | ||||
|           <Form.Field> | ||||
|             <Form.Input | ||||
|               label='名称' | ||||
|               name='name' | ||||
|               placeholder={'请输入名称'} | ||||
|               onChange={handleInputChange} | ||||
|               value={name} | ||||
|               autoComplete='off' | ||||
|               required={!isEdit} | ||||
|             /> | ||||
|           </Form.Field> | ||||
|           <Form.Field> | ||||
|             <Form.Input | ||||
|               label='额度' | ||||
|               name='quota' | ||||
|               placeholder={'请输入单个兑换码中包含的额度'} | ||||
|               onChange={handleInputChange} | ||||
|               value={quota} | ||||
|               autoComplete='off' | ||||
|               type='number' | ||||
|             /> | ||||
|           </Form.Field> | ||||
|           { | ||||
|             !isEdit && <> | ||||
|               <Form.Field> | ||||
|                 <Form.Input | ||||
|                   label='生成数量' | ||||
|                   name='count' | ||||
|                   placeholder={'请输入生成数量'} | ||||
|                   onChange={handleInputChange} | ||||
|                   value={count} | ||||
|                   autoComplete='off' | ||||
|                   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,53 +0,0 @@ | ||||
| import React, { useState } from 'react'; | ||||
| import { Button, Form, Header, Segment } from 'semantic-ui-react'; | ||||
| import { API, showError, showSuccess } from '../../helpers'; | ||||
|  | ||||
| const AddToken = () => { | ||||
|   const originInputs = { | ||||
|     name: '', | ||||
|   }; | ||||
|   const [inputs, setInputs] = useState(originInputs); | ||||
|   const { name, display_name, password } = inputs; | ||||
|  | ||||
|   const handleInputChange = (e, { name, value }) => { | ||||
|     setInputs((inputs) => ({ ...inputs, [name]: value })); | ||||
|   }; | ||||
|  | ||||
|   const submit = async () => { | ||||
|     if (inputs.name === '') return; | ||||
|     const res = await API.post(`/api/token/`, 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.Input | ||||
|               label="名称" | ||||
|               name="name" | ||||
|               placeholder={'请输入名称'} | ||||
|               onChange={handleInputChange} | ||||
|               value={name} | ||||
|               autoComplete="off" | ||||
|               required | ||||
|             /> | ||||
|           </Form.Field> | ||||
|           <Button type={'submit'} onClick={submit}> | ||||
|             提交 | ||||
|           </Button> | ||||
|         </Form> | ||||
|       </Segment> | ||||
|     </> | ||||
|   ); | ||||
| }; | ||||
|  | ||||
| export default AddToken; | ||||
| @@ -1,25 +1,53 @@ | ||||
| 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 } from '../../helpers'; | ||||
| import { API, isAdmin, showError, showSuccess, timestamp2string } from '../../helpers'; | ||||
|  | ||||
| const EditToken = () => { | ||||
|   const params = useParams(); | ||||
|   const tokenId = params.id; | ||||
|   const [loading, setLoading] = useState(true); | ||||
|   const [inputs, setInputs] = useState({ | ||||
|     name: '' | ||||
|   }); | ||||
|   const { name } = inputs; | ||||
|   const isEdit = tokenId !== undefined; | ||||
|   const [loading, setLoading] = useState(isEdit); | ||||
|   const originInputs = { | ||||
|     name: '', | ||||
|     remain_times: 0, | ||||
|     expired_time: -1, | ||||
|     unlimited_times: false | ||||
|   }; | ||||
|   const isAdminUser = isAdmin(); | ||||
|   const [inputs, setInputs] = useState(originInputs); | ||||
|   const { name, remain_times, expired_time, unlimited_times } = inputs; | ||||
|  | ||||
|   const handleInputChange = (e, { name, value }) => { | ||||
|     setInputs((inputs) => ({ ...inputs, [name]: value })); | ||||
|   }; | ||||
|  | ||||
|   const setExpiredTime = (month, day, hour, minute) => { | ||||
|     let now = new Date(); | ||||
|     let timestamp = now.getTime() / 1000; | ||||
|     let seconds = month * 30 * 24 * 60 * 60; | ||||
|     seconds += day * 24 * 60 * 60; | ||||
|     seconds += hour * 60 * 60; | ||||
|     seconds += minute * 60; | ||||
|     if (seconds !== 0) { | ||||
|       timestamp += seconds; | ||||
|       setInputs({ ...inputs, expired_time: timestamp2string(timestamp) }); | ||||
|     } else { | ||||
|       setInputs({ ...inputs, expired_time: -1 }); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   const setUnlimitedTimes = () => { | ||||
|     setInputs({ ...inputs, unlimited_times: !unlimited_times }); | ||||
|   }; | ||||
|  | ||||
|   const loadToken = async () => { | ||||
|     let res = await API.get(`/api/token/${tokenId}`); | ||||
|     const { success, message, data } = res.data; | ||||
|     if (success) { | ||||
|       data.password = ''; | ||||
|       if (data.expired_time !== -1) { | ||||
|         data.expired_time = timestamp2string(data.expired_time); | ||||
|       } | ||||
|       setInputs(data); | ||||
|     } else { | ||||
|       showError(message); | ||||
| @@ -27,14 +55,37 @@ const EditToken = () => { | ||||
|     setLoading(false); | ||||
|   }; | ||||
|   useEffect(() => { | ||||
|     loadToken().then(); | ||||
|     if (isEdit) { | ||||
|       loadToken().then(); | ||||
|     } | ||||
|   }, []); | ||||
|  | ||||
|   const submit = async () => { | ||||
|     let res = await API.put(`/api/token/`, { ...inputs, id: parseInt(tokenId) }); | ||||
|     if (!isEdit && inputs.name === '') return; | ||||
|     let localInputs = inputs; | ||||
|     localInputs.remain_times = parseInt(localInputs.remain_times); | ||||
|     if (localInputs.expired_time !== -1) { | ||||
|       let time = Date.parse(localInputs.expired_time); | ||||
|       if (isNaN(time)) { | ||||
|         showError('过期时间格式错误!'); | ||||
|         return; | ||||
|       } | ||||
|       localInputs.expired_time = Math.ceil(time / 1000); | ||||
|     } | ||||
|     let res; | ||||
|     if (isEdit) { | ||||
|       res = await API.put(`/api/token/`, { ...localInputs, id: parseInt(tokenId) }); | ||||
|     } else { | ||||
|       res = await API.post(`/api/token/`, localInputs); | ||||
|     } | ||||
|     const { success, message } = res.data; | ||||
|     if (success) { | ||||
|       showSuccess('令牌更新成功!'); | ||||
|       if (isEdit) { | ||||
|         showSuccess('令牌更新成功!'); | ||||
|       } else { | ||||
|         showSuccess('令牌创建成功!'); | ||||
|         setInputs(originInputs); | ||||
|       } | ||||
|     } else { | ||||
|       showError(message); | ||||
|     } | ||||
| @@ -43,18 +94,64 @@ const EditToken = () => { | ||||
|   return ( | ||||
|     <> | ||||
|       <Segment loading={loading}> | ||||
|         <Header as='h3'>更新令牌信息</Header> | ||||
|         <Header as='h3'>{isEdit ? '更新令牌信息' : '创建新的令牌'}</Header> | ||||
|         <Form autoComplete='off'> | ||||
|           <Form.Field> | ||||
|             <Form.Input | ||||
|               label='名称' | ||||
|               name='name' | ||||
|               placeholder={'请输入新的名称'} | ||||
|               placeholder={'请输入名称'} | ||||
|               onChange={handleInputChange} | ||||
|               value={name} | ||||
|               autoComplete='off' | ||||
|               required={!isEdit} | ||||
|             /> | ||||
|           </Form.Field> | ||||
|           { | ||||
|             isAdminUser && <> | ||||
|               <Form.Field> | ||||
|                 <Form.Input | ||||
|                   label='剩余次数' | ||||
|                   name='remain_times' | ||||
|                   placeholder={'请输入剩余次数'} | ||||
|                   onChange={handleInputChange} | ||||
|                   value={remain_times} | ||||
|                   autoComplete='off' | ||||
|                   type='number' | ||||
|                   disabled={unlimited_times} | ||||
|                 /> | ||||
|               </Form.Field> | ||||
|               <Button type={'button'} onClick={() => { | ||||
|                 setUnlimitedTimes(); | ||||
|               }}>{unlimited_times ? '取消无限次' : '设置为无限次'}</Button> | ||||
|             </> | ||||
|           } | ||||
|           <Form.Field> | ||||
|             <Form.Input | ||||
|               label='过期时间' | ||||
|               name='expired_time' | ||||
|               placeholder={'请输入过期时间,格式为 yyyy-MM-dd HH:mm:ss,-1 表示无限制'} | ||||
|               onChange={handleInputChange} | ||||
|               value={expired_time} | ||||
|               autoComplete='off' | ||||
|               type='datetime-local' | ||||
|             /> | ||||
|           </Form.Field> | ||||
|           <Button type={'button'} onClick={() => { | ||||
|             setExpiredTime(0, 0, 0, 0); | ||||
|           }}>永不过期</Button> | ||||
|           <Button type={'button'} onClick={() => { | ||||
|             setExpiredTime(1, 0, 0, 0); | ||||
|           }}>一个月后过期</Button> | ||||
|           <Button type={'button'} onClick={() => { | ||||
|             setExpiredTime(0, 1, 0, 0); | ||||
|           }}>一天后过期</Button> | ||||
|           <Button type={'button'} onClick={() => { | ||||
|             setExpiredTime(0, 0, 1, 0); | ||||
|           }}>一小时后过期</Button> | ||||
|           <Button type={'button'} onClick={() => { | ||||
|             setExpiredTime(0, 0, 0, 1); | ||||
|           }}>一分钟后过期</Button> | ||||
|           <Button onClick={submit}>提交</Button> | ||||
|         </Form> | ||||
|       </Segment> | ||||
|   | ||||
		Reference in New Issue
	
	Block a user