mirror of
https://github.com/linux-do/new-api.git
synced 2025-09-18 00:16:37 +08:00
synced with upstream
Signed-off-by: wozulong <>
This commit is contained in:
commit
e4753e7411
@ -2,9 +2,7 @@
|
|||||||
|
|
||||||
**简介**:Midjourney Proxy API文档
|
**简介**:Midjourney Proxy API文档
|
||||||
|
|
||||||
## 模型价格设置(在设置-运营设置-模型固定价格设置中设置)
|
## 模型列表
|
||||||
|
|
||||||
### 模型列表
|
|
||||||
|
|
||||||
### midjourney-proxy支持
|
### midjourney-proxy支持
|
||||||
|
|
||||||
@ -27,6 +25,7 @@
|
|||||||
- mj_pan (平移)
|
- mj_pan (平移)
|
||||||
- swap_face (换脸)
|
- swap_face (换脸)
|
||||||
|
|
||||||
|
## 模型价格设置(在设置-运营设置-模型固定价格设置中设置)
|
||||||
```json
|
```json
|
||||||
{
|
{
|
||||||
"mj_imagine": 0.1,
|
"mj_imagine": 0.1,
|
||||||
@ -46,6 +45,7 @@
|
|||||||
"swap_face": 0.05
|
"swap_face": 0.05
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
其中mj_inpaint和mj_custom_zoom的价格设置为0,是因为这两个模型需要搭配mj_modal使用,所以价格由mj_modal决定。
|
||||||
|
|
||||||
## 渠道设置
|
## 渠道设置
|
||||||
|
|
||||||
@ -56,12 +56,12 @@
|
|||||||
部署Midjourney-Proxy,并配置好midjourney账号等(强烈建议设置密钥),[项目地址](https://github.com/novicezk/midjourney-proxy)
|
部署Midjourney-Proxy,并配置好midjourney账号等(强烈建议设置密钥),[项目地址](https://github.com/novicezk/midjourney-proxy)
|
||||||
|
|
||||||
2. 在渠道管理中添加渠道,渠道类型选择**Midjourney Proxy**,如果是plus版本选择**Midjourney Proxy Plus**
|
2. 在渠道管理中添加渠道,渠道类型选择**Midjourney Proxy**,如果是plus版本选择**Midjourney Proxy Plus**
|
||||||
,模型选择midjourney,如果有换脸模型,可以选择swap_face
|
,模型请参考上方模型列表
|
||||||
3. 地址填写midjourney-proxy部署的地址,例如:http://localhost:8080
|
3. 地址填写midjourney-proxy部署的地址,例如:http://localhost:8080
|
||||||
4. 密钥填写midjourney-proxy的密钥,如果没有设置密钥,可以随便填
|
4. 密钥填写midjourney-proxy的密钥,如果没有设置密钥,可以随便填
|
||||||
|
|
||||||
### 对接上游new api
|
### 对接上游new api
|
||||||
|
|
||||||
1. 在渠道管理中添加渠道,渠道类型选择**Midjourney Proxy Plus**,模型选择midjourney,如果有换脸模型,可以选择swap_face
|
1. 在渠道管理中添加渠道,渠道类型选择**Midjourney Proxy Plus**,模型请参考上方模型列表
|
||||||
2. 地址填写上游new api的地址,例如:http://localhost:3000
|
2. 地址填写上游new api的地址,例如:http://localhost:3000
|
||||||
3. 密钥填写上游new api的密钥
|
3. 密钥填写上游new api的密钥
|
10
README.md
10
README.md
@ -18,7 +18,7 @@
|
|||||||
此分叉版本的主要变更如下:
|
此分叉版本的主要变更如下:
|
||||||
|
|
||||||
1. 全新的UI界面(部分界面还待更新)
|
1. 全新的UI界面(部分界面还待更新)
|
||||||
2. 添加[Midjourney-Proxy(Plus)](https://github.com/novicezk/midjourney-proxy)接口的支持
|
2. 添加[Midjourney-Proxy(Plus)](https://github.com/novicezk/midjourney-proxy)接口的支持,[对接文档](Midjourney.md),支持的接口如下:
|
||||||
+ [x] /mj/submit/imagine
|
+ [x] /mj/submit/imagine
|
||||||
+ [x] /mj/submit/change
|
+ [x] /mj/submit/change
|
||||||
+ [x] /mj/submit/blend
|
+ [x] /mj/submit/blend
|
||||||
@ -54,7 +54,7 @@
|
|||||||
2. 智谱glm-4v,glm-4v识图
|
2. 智谱glm-4v,glm-4v识图
|
||||||
3. Anthropic Claude 3 (claude-3-opus-20240229, claude-3-sonnet-20240229)
|
3. Anthropic Claude 3 (claude-3-opus-20240229, claude-3-sonnet-20240229)
|
||||||
4. [Ollama](https://github.com/ollama/ollama?tab=readme-ov-file),添加渠道时,密钥可以随便填写,默认的请求地址是[http://localhost:11434](http://localhost:11434),如果需要修改请在渠道中修改
|
4. [Ollama](https://github.com/ollama/ollama?tab=readme-ov-file),添加渠道时,密钥可以随便填写,默认的请求地址是[http://localhost:11434](http://localhost:11434),如果需要修改请在渠道中修改
|
||||||
5. [Midjourney-Proxy(Plus)](https://github.com/novicezk/midjourney-proxy)接口
|
5. [Midjourney-Proxy(Plus)](https://github.com/novicezk/midjourney-proxy)接口,[对接文档](Midjourney.md)
|
||||||
|
|
||||||
您可以在渠道中添加自定义模型gpt-4-gizmo-*,此模型并非OpenAI官方模型,而是第三方模型,使用官方key无法调用。
|
您可以在渠道中添加自定义模型gpt-4-gizmo-*,此模型并非OpenAI官方模型,而是第三方模型,使用官方key无法调用。
|
||||||
|
|
||||||
@ -98,6 +98,12 @@ docker run --name new-api -d --restart always -p 3000:3000 -e SQL_DSN="root:1234
|
|||||||

|

|
||||||

|

|
||||||
|
|
||||||
|
## 相关项目
|
||||||
|
- [One API](https://github.com/songquanpeng/one-api):原版项目
|
||||||
|
- [Midjourney-Proxy](https://github.com/novicezk/midjourney-proxy):Midjourney接口支持
|
||||||
|
- [chatnio](https://github.com/Deeptrain-Community/chatnio):下一代 AI 一站式 B/C 端解决方案
|
||||||
|
- [neko-api-key-tool](https://github.com/Calcium-Ion/neko-api-key-tool):用key查询使用额度
|
||||||
|
|
||||||
## Star History
|
## Star History
|
||||||
|
|
||||||
[](https://star-history.com/#Calcium-Ion/new-api&Date)
|
[](https://star-history.com/#Calcium-Ion/new-api&Date)
|
||||||
|
@ -215,6 +215,7 @@ const (
|
|||||||
ChannelTypeGemini = 24
|
ChannelTypeGemini = 24
|
||||||
ChannelTypeMoonshot = 25
|
ChannelTypeMoonshot = 25
|
||||||
ChannelTypeZhipu_v4 = 26
|
ChannelTypeZhipu_v4 = 26
|
||||||
|
ChannelTypePerplexity = 27
|
||||||
)
|
)
|
||||||
|
|
||||||
var ChannelBaseURLs = []string{
|
var ChannelBaseURLs = []string{
|
||||||
@ -245,4 +246,5 @@ var ChannelBaseURLs = []string{
|
|||||||
"https://generativelanguage.googleapis.com", //24
|
"https://generativelanguage.googleapis.com", //24
|
||||||
"https://api.moonshot.cn", //25
|
"https://api.moonshot.cn", //25
|
||||||
"https://open.bigmodel.cn", //26
|
"https://open.bigmodel.cn", //26
|
||||||
|
"https://api.perplexity.ai", //27
|
||||||
}
|
}
|
||||||
|
@ -5,7 +5,7 @@ import (
|
|||||||
"encoding/base64"
|
"encoding/base64"
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"github.com/chai2010/webp"
|
"golang.org/x/image/webp"
|
||||||
"image"
|
"image"
|
||||||
"io"
|
"io"
|
||||||
"net/http"
|
"net/http"
|
||||||
|
@ -13,7 +13,7 @@ import (
|
|||||||
// TODO: when a new api is enabled, check the pricing here
|
// TODO: when a new api is enabled, check the pricing here
|
||||||
// 1 === $0.002 / 1K tokens
|
// 1 === $0.002 / 1K tokens
|
||||||
// 1 === ¥0.014 / 1k tokens
|
// 1 === ¥0.014 / 1k tokens
|
||||||
var ModelRatio = map[string]float64{
|
var DefaultModelRatio = map[string]float64{
|
||||||
//"midjourney": 50,
|
//"midjourney": 50,
|
||||||
"gpt-4-gizmo-*": 15,
|
"gpt-4-gizmo-*": 15,
|
||||||
"gpt-4": 15,
|
"gpt-4": 15,
|
||||||
@ -115,6 +115,7 @@ var DefaultModelPrice = map[string]float64{
|
|||||||
}
|
}
|
||||||
|
|
||||||
var ModelPrice = map[string]float64{}
|
var ModelPrice = map[string]float64{}
|
||||||
|
var ModelRatio = map[string]float64{}
|
||||||
|
|
||||||
func ModelPrice2JSONString() string {
|
func ModelPrice2JSONString() string {
|
||||||
if len(ModelPrice) == 0 {
|
if len(ModelPrice) == 0 {
|
||||||
@ -150,6 +151,9 @@ func GetModelPrice(name string, printErr bool) float64 {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func ModelRatio2JSONString() string {
|
func ModelRatio2JSONString() string {
|
||||||
|
if len(ModelRatio) == 0 {
|
||||||
|
ModelRatio = DefaultModelRatio
|
||||||
|
}
|
||||||
jsonBytes, err := json.Marshal(ModelRatio)
|
jsonBytes, err := json.Marshal(ModelRatio)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
SysError("error marshalling model ratio: " + err.Error())
|
SysError("error marshalling model ratio: " + err.Error())
|
||||||
@ -163,6 +167,9 @@ func UpdateModelRatioByJSONString(jsonStr string) error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func GetModelRatio(name string) float64 {
|
func GetModelRatio(name string) float64 {
|
||||||
|
if len(ModelRatio) == 0 {
|
||||||
|
ModelRatio = DefaultModelRatio
|
||||||
|
}
|
||||||
if strings.HasPrefix(name, "gpt-4-gizmo") {
|
if strings.HasPrefix(name, "gpt-4-gizmo") {
|
||||||
name = "gpt-4-gizmo-*"
|
name = "gpt-4-gizmo-*"
|
||||||
}
|
}
|
||||||
|
@ -1,5 +1,7 @@
|
|||||||
package constant
|
package constant
|
||||||
|
|
||||||
|
var MjNotifyEnabled = false
|
||||||
|
|
||||||
const (
|
const (
|
||||||
MjErrorUnknown = 5
|
MjErrorUnknown = 5
|
||||||
MjRequestError = 4
|
MjRequestError = 4
|
||||||
|
@ -10,9 +10,13 @@ import (
|
|||||||
|
|
||||||
func GetAllLogs(c *gin.Context) {
|
func GetAllLogs(c *gin.Context) {
|
||||||
p, _ := strconv.Atoi(c.Query("p"))
|
p, _ := strconv.Atoi(c.Query("p"))
|
||||||
|
pageSize, _ := strconv.Atoi(c.Query("page_size"))
|
||||||
if p < 0 {
|
if p < 0 {
|
||||||
p = 0
|
p = 0
|
||||||
}
|
}
|
||||||
|
if pageSize < 0 {
|
||||||
|
pageSize = common.ItemsPerPage
|
||||||
|
}
|
||||||
logType, _ := strconv.Atoi(c.Query("type"))
|
logType, _ := strconv.Atoi(c.Query("type"))
|
||||||
startTimestamp, _ := strconv.ParseInt(c.Query("start_timestamp"), 10, 64)
|
startTimestamp, _ := strconv.ParseInt(c.Query("start_timestamp"), 10, 64)
|
||||||
endTimestamp, _ := strconv.ParseInt(c.Query("end_timestamp"), 10, 64)
|
endTimestamp, _ := strconv.ParseInt(c.Query("end_timestamp"), 10, 64)
|
||||||
@ -20,7 +24,7 @@ func GetAllLogs(c *gin.Context) {
|
|||||||
tokenName := c.Query("token_name")
|
tokenName := c.Query("token_name")
|
||||||
modelName := c.Query("model_name")
|
modelName := c.Query("model_name")
|
||||||
channel, _ := strconv.Atoi(c.Query("channel"))
|
channel, _ := strconv.Atoi(c.Query("channel"))
|
||||||
logs, err := model.GetAllLogs(logType, startTimestamp, endTimestamp, modelName, username, tokenName, p*common.ItemsPerPage, common.ItemsPerPage, channel)
|
logs, err := model.GetAllLogs(logType, startTimestamp, endTimestamp, modelName, username, tokenName, p*pageSize, pageSize, channel)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
c.JSON(http.StatusOK, gin.H{
|
c.JSON(http.StatusOK, gin.H{
|
||||||
"success": false,
|
"success": false,
|
||||||
@ -38,16 +42,23 @@ func GetAllLogs(c *gin.Context) {
|
|||||||
|
|
||||||
func GetUserLogs(c *gin.Context) {
|
func GetUserLogs(c *gin.Context) {
|
||||||
p, _ := strconv.Atoi(c.Query("p"))
|
p, _ := strconv.Atoi(c.Query("p"))
|
||||||
|
pageSize, _ := strconv.Atoi(c.Query("page_size"))
|
||||||
if p < 0 {
|
if p < 0 {
|
||||||
p = 0
|
p = 0
|
||||||
}
|
}
|
||||||
|
if pageSize < 0 {
|
||||||
|
pageSize = common.ItemsPerPage
|
||||||
|
}
|
||||||
|
if pageSize > 100 {
|
||||||
|
pageSize = 100
|
||||||
|
}
|
||||||
userId := c.GetInt("id")
|
userId := c.GetInt("id")
|
||||||
logType, _ := strconv.Atoi(c.Query("type"))
|
logType, _ := strconv.Atoi(c.Query("type"))
|
||||||
startTimestamp, _ := strconv.ParseInt(c.Query("start_timestamp"), 10, 64)
|
startTimestamp, _ := strconv.ParseInt(c.Query("start_timestamp"), 10, 64)
|
||||||
endTimestamp, _ := strconv.ParseInt(c.Query("end_timestamp"), 10, 64)
|
endTimestamp, _ := strconv.ParseInt(c.Query("end_timestamp"), 10, 64)
|
||||||
tokenName := c.Query("token_name")
|
tokenName := c.Query("token_name")
|
||||||
modelName := c.Query("model_name")
|
modelName := c.Query("model_name")
|
||||||
logs, err := model.GetUserLogs(userId, logType, startTimestamp, endTimestamp, modelName, tokenName, p*common.ItemsPerPage, common.ItemsPerPage)
|
logs, err := model.GetUserLogs(userId, logType, startTimestamp, endTimestamp, modelName, tokenName, p*pageSize, pageSize)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
c.JSON(http.StatusOK, gin.H{
|
c.JSON(http.StatusOK, gin.H{
|
||||||
"success": false,
|
"success": false,
|
||||||
|
@ -5,6 +5,7 @@ import (
|
|||||||
"fmt"
|
"fmt"
|
||||||
"net/http"
|
"net/http"
|
||||||
"one-api/common"
|
"one-api/common"
|
||||||
|
"one-api/constant"
|
||||||
"one-api/model"
|
"one-api/model"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
@ -61,6 +62,7 @@ func GetStatus(c *gin.Context) {
|
|||||||
"data_export_default_time": common.DataExportDefaultTime,
|
"data_export_default_time": common.DataExportDefaultTime,
|
||||||
"default_collapse_sidebar": common.DefaultCollapseSidebar,
|
"default_collapse_sidebar": common.DefaultCollapseSidebar,
|
||||||
"enable_online_topup": common.PayAddress != "" && common.EpayId != "" && common.EpayKey != "",
|
"enable_online_topup": common.PayAddress != "" && common.EpayId != "" && common.EpayKey != "",
|
||||||
|
"mj_notify_enabled": constant.MjNotifyEnabled,
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
return
|
return
|
||||||
|
14
go.mod
14
go.mod
@ -4,13 +4,12 @@ module one-api
|
|||||||
go 1.18
|
go 1.18
|
||||||
|
|
||||||
require (
|
require (
|
||||||
github.com/chai2010/webp v1.1.1
|
|
||||||
github.com/gin-contrib/cors v1.4.0
|
github.com/gin-contrib/cors v1.4.0
|
||||||
github.com/gin-contrib/gzip v0.0.6
|
github.com/gin-contrib/gzip v0.0.6
|
||||||
github.com/gin-contrib/sessions v0.0.5
|
github.com/gin-contrib/sessions v0.0.5
|
||||||
github.com/gin-contrib/static v0.0.1
|
github.com/gin-contrib/static v0.0.1
|
||||||
github.com/gin-gonic/gin v1.9.1
|
github.com/gin-gonic/gin v1.9.1
|
||||||
github.com/go-playground/validator/v10 v10.16.0
|
github.com/go-playground/validator/v10 v10.19.0
|
||||||
github.com/go-redis/redis/v8 v8.11.5
|
github.com/go-redis/redis/v8 v8.11.5
|
||||||
github.com/golang-jwt/jwt v3.2.2+incompatible
|
github.com/golang-jwt/jwt v3.2.2+incompatible
|
||||||
github.com/google/uuid v1.3.0
|
github.com/google/uuid v1.3.0
|
||||||
@ -19,7 +18,8 @@ require (
|
|||||||
github.com/samber/lo v1.38.1
|
github.com/samber/lo v1.38.1
|
||||||
github.com/shirou/gopsutil v3.21.11+incompatible
|
github.com/shirou/gopsutil v3.21.11+incompatible
|
||||||
github.com/star-horizon/go-epay v0.0.0-20230204124159-fa2e2293fdc2
|
github.com/star-horizon/go-epay v0.0.0-20230204124159-fa2e2293fdc2
|
||||||
golang.org/x/crypto v0.17.0
|
golang.org/x/crypto v0.21.0
|
||||||
|
golang.org/x/image v0.15.0
|
||||||
gorm.io/driver/mysql v1.4.3
|
gorm.io/driver/mysql v1.4.3
|
||||||
gorm.io/driver/postgres v1.5.2
|
gorm.io/driver/postgres v1.5.2
|
||||||
gorm.io/driver/sqlite v1.4.3
|
gorm.io/driver/sqlite v1.4.3
|
||||||
@ -32,7 +32,7 @@ require (
|
|||||||
github.com/chenzhuoyu/base64x v0.0.0-20221115062448-fe3a3abad311 // indirect
|
github.com/chenzhuoyu/base64x v0.0.0-20221115062448-fe3a3abad311 // indirect
|
||||||
github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f // indirect
|
github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f // indirect
|
||||||
github.com/dlclark/regexp2 v1.10.0 // indirect
|
github.com/dlclark/regexp2 v1.10.0 // indirect
|
||||||
github.com/gabriel-vasile/mimetype v1.4.2 // indirect
|
github.com/gabriel-vasile/mimetype v1.4.3 // indirect
|
||||||
github.com/gin-contrib/sse v0.1.0 // indirect
|
github.com/gin-contrib/sse v0.1.0 // indirect
|
||||||
github.com/go-ole/go-ole v1.2.6 // indirect
|
github.com/go-ole/go-ole v1.2.6 // indirect
|
||||||
github.com/go-playground/locales v0.14.1 // indirect
|
github.com/go-playground/locales v0.14.1 // indirect
|
||||||
@ -50,7 +50,7 @@ require (
|
|||||||
github.com/jinzhu/now v1.1.5 // indirect
|
github.com/jinzhu/now v1.1.5 // indirect
|
||||||
github.com/json-iterator/go v1.1.12 // indirect
|
github.com/json-iterator/go v1.1.12 // indirect
|
||||||
github.com/klauspost/cpuid/v2 v2.2.4 // indirect
|
github.com/klauspost/cpuid/v2 v2.2.4 // indirect
|
||||||
github.com/leodido/go-urn v1.2.4 // indirect
|
github.com/leodido/go-urn v1.4.0 // indirect
|
||||||
github.com/mattn/go-isatty v0.0.20 // indirect
|
github.com/mattn/go-isatty v0.0.20 // indirect
|
||||||
github.com/mattn/go-sqlite3 v2.0.3+incompatible // indirect
|
github.com/mattn/go-sqlite3 v2.0.3+incompatible // indirect
|
||||||
github.com/mitchellh/mapstructure v1.5.0 // indirect
|
github.com/mitchellh/mapstructure v1.5.0 // indirect
|
||||||
@ -64,9 +64,9 @@ require (
|
|||||||
github.com/yusufpapurcu/wmi v1.2.3 // indirect
|
github.com/yusufpapurcu/wmi v1.2.3 // indirect
|
||||||
golang.org/x/arch v0.3.0 // indirect
|
golang.org/x/arch v0.3.0 // indirect
|
||||||
golang.org/x/exp v0.0.0-20220303212507-bbda1eaf7a17 // indirect
|
golang.org/x/exp v0.0.0-20220303212507-bbda1eaf7a17 // indirect
|
||||||
golang.org/x/net v0.17.0 // indirect
|
golang.org/x/net v0.21.0 // indirect
|
||||||
golang.org/x/sync v0.1.0 // indirect
|
golang.org/x/sync v0.1.0 // indirect
|
||||||
golang.org/x/sys v0.15.0 // indirect
|
golang.org/x/sys v0.18.0 // indirect
|
||||||
golang.org/x/text v0.14.0 // indirect
|
golang.org/x/text v0.14.0 // indirect
|
||||||
google.golang.org/protobuf v1.30.0 // indirect
|
google.golang.org/protobuf v1.30.0 // indirect
|
||||||
gopkg.in/yaml.v3 v3.0.1 // indirect
|
gopkg.in/yaml.v3 v3.0.1 // indirect
|
||||||
|
29
go.sum
29
go.sum
@ -3,8 +3,6 @@ github.com/bytedance/sonic v1.9.1 h1:6iJ6NqdoxCDr6mbY8h18oSO+cShGSMRGCEo7F2h0x8s
|
|||||||
github.com/bytedance/sonic v1.9.1/go.mod h1:i736AoUSYt75HyZLoJW9ERYxcy6eaN6h4BZXU064P/U=
|
github.com/bytedance/sonic v1.9.1/go.mod h1:i736AoUSYt75HyZLoJW9ERYxcy6eaN6h4BZXU064P/U=
|
||||||
github.com/cespare/xxhash/v2 v2.1.2 h1:YRXhKfTDauu4ajMg1TPgFO5jnlC2HCbmLXMcTG5cbYE=
|
github.com/cespare/xxhash/v2 v2.1.2 h1:YRXhKfTDauu4ajMg1TPgFO5jnlC2HCbmLXMcTG5cbYE=
|
||||||
github.com/cespare/xxhash/v2 v2.1.2/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs=
|
github.com/cespare/xxhash/v2 v2.1.2/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs=
|
||||||
github.com/chai2010/webp v1.1.1 h1:jTRmEccAJ4MGrhFOrPMpNGIJ/eybIgwKpcACsrTEapk=
|
|
||||||
github.com/chai2010/webp v1.1.1/go.mod h1:0XVwvZWdjjdxpUEIf7b9g9VkHFnInUSYujwqTLEuldU=
|
|
||||||
github.com/chenzhuoyu/base64x v0.0.0-20211019084208-fb5309c8db06/go.mod h1:DH46F32mSOjUmXrMHnKwZdA8wcEefY7UVqBKYGjpdQY=
|
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 h1:qSGYFH7+jGhDF8vLC+iwCD4WpbV1EBDSzWkJODFLams=
|
||||||
github.com/chenzhuoyu/base64x v0.0.0-20221115062448-fe3a3abad311/go.mod h1:b583jCggY9gE99b6G5LEC39OIiVsWj+R97kbl5odCEk=
|
github.com/chenzhuoyu/base64x v0.0.0-20221115062448-fe3a3abad311/go.mod h1:b583jCggY9gE99b6G5LEC39OIiVsWj+R97kbl5odCEk=
|
||||||
@ -19,6 +17,8 @@ github.com/dlclark/regexp2 v1.10.0/go.mod h1:DHkYz0B9wPfa6wondMfaivmHpzrQ3v9q8cn
|
|||||||
github.com/fsnotify/fsnotify v1.4.9 h1:hsms1Qyu0jgnwNXIxa+/V/PDsU6CfLf6CNO8H7IWoS4=
|
github.com/fsnotify/fsnotify v1.4.9 h1:hsms1Qyu0jgnwNXIxa+/V/PDsU6CfLf6CNO8H7IWoS4=
|
||||||
github.com/gabriel-vasile/mimetype v1.4.2 h1:w5qFW6JKBz9Y393Y4q372O9A7cUSequkh1Q7OhCmWKU=
|
github.com/gabriel-vasile/mimetype v1.4.2 h1:w5qFW6JKBz9Y393Y4q372O9A7cUSequkh1Q7OhCmWKU=
|
||||||
github.com/gabriel-vasile/mimetype v1.4.2/go.mod h1:zApsH/mKG4w07erKIaJPFiX0Tsq9BFQgN3qGY5GnNgA=
|
github.com/gabriel-vasile/mimetype v1.4.2/go.mod h1:zApsH/mKG4w07erKIaJPFiX0Tsq9BFQgN3qGY5GnNgA=
|
||||||
|
github.com/gabriel-vasile/mimetype v1.4.3 h1:in2uUcidCuFcDKtdcBxlR0rJ1+fsokWf+uqxgUFjbI0=
|
||||||
|
github.com/gabriel-vasile/mimetype v1.4.3/go.mod h1:d8uq/6HKRL6CGdk+aubisF/M5GcPfT7nKyLpA0lbSSk=
|
||||||
github.com/gin-contrib/cors v1.4.0 h1:oJ6gwtUl3lqV0WEIwM/LxPF1QZ5qe2lGWdY2+bz7y0g=
|
github.com/gin-contrib/cors v1.4.0 h1:oJ6gwtUl3lqV0WEIwM/LxPF1QZ5qe2lGWdY2+bz7y0g=
|
||||||
github.com/gin-contrib/cors v1.4.0/go.mod h1:bs9pNM0x/UsmHPBWT2xZz9ROh8xYjYkiURUfmBoMlcs=
|
github.com/gin-contrib/cors v1.4.0/go.mod h1:bs9pNM0x/UsmHPBWT2xZz9ROh8xYjYkiURUfmBoMlcs=
|
||||||
github.com/gin-contrib/gzip v0.0.6 h1:NjcunTcGAj5CO1gn4N8jHOSIeRFHIbn51z6K+xaN4d4=
|
github.com/gin-contrib/gzip v0.0.6 h1:NjcunTcGAj5CO1gn4N8jHOSIeRFHIbn51z6K+xaN4d4=
|
||||||
@ -37,6 +37,7 @@ github.com/go-ole/go-ole v1.2.6 h1:/Fpf6oFPoeFik9ty7siob0G6Ke8QvQEuVcuChpwXzpY=
|
|||||||
github.com/go-ole/go-ole v1.2.6/go.mod h1:pprOEPIfldk/42T2oK7lQ4v4JSDwmV0As9GaiUsvbm0=
|
github.com/go-ole/go-ole v1.2.6/go.mod h1:pprOEPIfldk/42T2oK7lQ4v4JSDwmV0As9GaiUsvbm0=
|
||||||
github.com/go-playground/assert/v2 v2.0.1/go.mod h1:VDjEfimB/XKnb+ZQfWdccd7VUvScMdVu0Titje2rxJ4=
|
github.com/go-playground/assert/v2 v2.0.1/go.mod h1:VDjEfimB/XKnb+ZQfWdccd7VUvScMdVu0Titje2rxJ4=
|
||||||
github.com/go-playground/assert/v2 v2.2.0 h1:JvknZsQTYeFEAhQwI4qEt9cyV5ONwRHC+lYKSsYSR8s=
|
github.com/go-playground/assert/v2 v2.2.0 h1:JvknZsQTYeFEAhQwI4qEt9cyV5ONwRHC+lYKSsYSR8s=
|
||||||
|
github.com/go-playground/assert/v2 v2.2.0/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.13.0/go.mod h1:taPMhCMXrRLJO55olJkUXHZBHCxTMfnGwq/HNwmWNS8=
|
||||||
github.com/go-playground/locales v0.14.0/go.mod h1:sawfccIbzZTqEDETgFXqTho0QybSa7l++s0DH+LDiLs=
|
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 h1:EWaQ/wswjilfKLTECiXz7Rh+3BjFhfDFKv/oXslEjJA=
|
||||||
@ -47,10 +48,10 @@ github.com/go-playground/universal-translator v0.18.1 h1:Bcnm0ZwsGyWbCzImXv+pAJn
|
|||||||
github.com/go-playground/universal-translator v0.18.1/go.mod h1:xekY+UJKNuX9WP91TpwSH2VMlDf28Uj24BCp08ZFTUY=
|
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.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.10.0/go.mod h1:74x4gJWsvQexRdW8Pn3dXSGrTK4nAUsbPlLADvpJkos=
|
||||||
github.com/go-playground/validator/v10 v10.14.0 h1:vgvQWe3XCz3gIeFDm/HnTIbj6UGmg/+t63MyGU2n5js=
|
|
||||||
github.com/go-playground/validator/v10 v10.14.0/go.mod h1:9iXMNT7sEkjXb0I+enO7QXmzG6QCsPWY4zveKFVRSyU=
|
|
||||||
github.com/go-playground/validator/v10 v10.16.0 h1:x+plE831WK4vaKHO/jpgUGsvLKIqRRkz6M78GuJAfGE=
|
github.com/go-playground/validator/v10 v10.16.0 h1:x+plE831WK4vaKHO/jpgUGsvLKIqRRkz6M78GuJAfGE=
|
||||||
github.com/go-playground/validator/v10 v10.16.0/go.mod h1:9iXMNT7sEkjXb0I+enO7QXmzG6QCsPWY4zveKFVRSyU=
|
github.com/go-playground/validator/v10 v10.16.0/go.mod h1:9iXMNT7sEkjXb0I+enO7QXmzG6QCsPWY4zveKFVRSyU=
|
||||||
|
github.com/go-playground/validator/v10 v10.19.0 h1:ol+5Fu+cSq9JD7SoSqe04GMI92cbn0+wvQ3bZ8b/AU4=
|
||||||
|
github.com/go-playground/validator/v10 v10.19.0/go.mod h1:dbuPbCMFw/DrkbEynArYaCwl3amGuJotoKCe95atGMM=
|
||||||
github.com/go-redis/redis/v8 v8.11.5 h1:AcZZR7igkdvfVmQTPnu9WE37LRrO/YrBH5zWyjDC0oI=
|
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-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 h1:BCTh4TKNUYmOmMUcQ3IipzF5prigylS7XXjEkfCHuOE=
|
||||||
@ -79,8 +80,6 @@ github.com/jackc/pgpassfile v1.0.0 h1:/6Hmqy13Ss2zCq62VdNG8tM1wchn8zjSGOBJ6icpsI
|
|||||||
github.com/jackc/pgpassfile v1.0.0/go.mod h1:CEx0iS5ambNFdcRtxPj5JhEz+xB6uRky5eyVu/W2HEg=
|
github.com/jackc/pgpassfile v1.0.0/go.mod h1:CEx0iS5ambNFdcRtxPj5JhEz+xB6uRky5eyVu/W2HEg=
|
||||||
github.com/jackc/pgservicefile v0.0.0-20221227161230-091c0ba34f0a h1:bbPeKD0xmW/Y25WS6cokEszi5g+S0QxI/d45PkRi7Nk=
|
github.com/jackc/pgservicefile v0.0.0-20221227161230-091c0ba34f0a h1:bbPeKD0xmW/Y25WS6cokEszi5g+S0QxI/d45PkRi7Nk=
|
||||||
github.com/jackc/pgservicefile v0.0.0-20221227161230-091c0ba34f0a/go.mod h1:5TJZWKEWniPve33vlWYSoGYefn3gLQRzjfDlhSJ9ZKM=
|
github.com/jackc/pgservicefile v0.0.0-20221227161230-091c0ba34f0a/go.mod h1:5TJZWKEWniPve33vlWYSoGYefn3gLQRzjfDlhSJ9ZKM=
|
||||||
github.com/jackc/pgx/v5 v5.3.1 h1:Fcr8QJ1ZeLi5zsPZqQeUZhNhxfkkKBOgJuYkJHoBOtU=
|
|
||||||
github.com/jackc/pgx/v5 v5.3.1/go.mod h1:t3JDKnCBlYIc0ewLF0Q7B8MXmoIaBOZj/ic7iHozM/8=
|
|
||||||
github.com/jackc/pgx/v5 v5.5.1 h1:5I9etrGkLrN+2XPCsi6XLlV5DITbSL/xBZdmAxFcXPI=
|
github.com/jackc/pgx/v5 v5.5.1 h1:5I9etrGkLrN+2XPCsi6XLlV5DITbSL/xBZdmAxFcXPI=
|
||||||
github.com/jackc/pgx/v5 v5.5.1/go.mod h1:Ig06C2Vu0t5qXC60W8sqIthScaEnFvojjj9dSljmHRA=
|
github.com/jackc/pgx/v5 v5.5.1/go.mod h1:Ig06C2Vu0t5qXC60W8sqIthScaEnFvojjj9dSljmHRA=
|
||||||
github.com/jackc/puddle/v2 v2.2.1 h1:RhxXJtFG022u4ibrCSMSiu5aOq1i77R3OHKNJj77OAk=
|
github.com/jackc/puddle/v2 v2.2.1 h1:RhxXJtFG022u4ibrCSMSiu5aOq1i77R3OHKNJj77OAk=
|
||||||
@ -108,10 +107,10 @@ github.com/leodido/go-urn v1.2.0/go.mod h1:+8+nEpDfqqsY+g338gtMEUOtuK+4dEMhiQEgx
|
|||||||
github.com/leodido/go-urn v1.2.1/go.mod h1:zt4jvISO2HfUBqxjfIshjdMTYS56ZS/qv49ictyFfxY=
|
github.com/leodido/go-urn v1.2.1/go.mod h1:zt4jvISO2HfUBqxjfIshjdMTYS56ZS/qv49ictyFfxY=
|
||||||
github.com/leodido/go-urn v1.2.4 h1:XlAE/cm/ms7TE/VMVoduSpNBoyc2dOxHs5MZSwAN63Q=
|
github.com/leodido/go-urn v1.2.4 h1:XlAE/cm/ms7TE/VMVoduSpNBoyc2dOxHs5MZSwAN63Q=
|
||||||
github.com/leodido/go-urn v1.2.4/go.mod h1:7ZrI8mTSeBSHl/UaRyKQW1qZeMgak41ANeCNaVckg+4=
|
github.com/leodido/go-urn v1.2.4/go.mod h1:7ZrI8mTSeBSHl/UaRyKQW1qZeMgak41ANeCNaVckg+4=
|
||||||
|
github.com/leodido/go-urn v1.4.0 h1:WT9HwE9SGECu3lg4d/dIA+jxlljEa1/ffXKmRjqdmIQ=
|
||||||
|
github.com/leodido/go-urn v1.4.0/go.mod h1:bvxc+MVxLKB4z00jd1z+Dvzr47oO32F/QSNjSBOlFxI=
|
||||||
github.com/mattn/go-isatty v0.0.12/go.mod h1:cbi8OIDigv2wuxKPP5vlRcQ1OAZbq2CE4Kysco4FUpU=
|
github.com/mattn/go-isatty v0.0.12/go.mod h1:cbi8OIDigv2wuxKPP5vlRcQ1OAZbq2CE4Kysco4FUpU=
|
||||||
github.com/mattn/go-isatty v0.0.14/go.mod h1:7GGIvUiUoEMVVmxf/4nioHXj79iQHKdU27kJ6hsGG94=
|
github.com/mattn/go-isatty v0.0.14/go.mod h1:7GGIvUiUoEMVVmxf/4nioHXj79iQHKdU27kJ6hsGG94=
|
||||||
github.com/mattn/go-isatty v0.0.19 h1:JITubQf0MOLdlGRuRq+jtsDlekdYPia9ZFsB8h/APPA=
|
|
||||||
github.com/mattn/go-isatty v0.0.19/go.mod h1:W+V8PltTTMOvKvAeJH7IuucS94S2C6jfK/D7dTCTo3Y=
|
|
||||||
github.com/mattn/go-isatty v0.0.20 h1:xfD0iDuEKnDkl03q4limB+vH+GxLEtL/jb4xVJSWWEY=
|
github.com/mattn/go-isatty v0.0.20 h1:xfD0iDuEKnDkl03q4limB+vH+GxLEtL/jb4xVJSWWEY=
|
||||||
github.com/mattn/go-isatty v0.0.20/go.mod h1:W+V8PltTTMOvKvAeJH7IuucS94S2C6jfK/D7dTCTo3Y=
|
github.com/mattn/go-isatty v0.0.20/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 v1.14.15/go.mod h1:2eHXhiwb8IkHr+BDWZGa96P6+rkvnG63S2DGjv9HUNg=
|
||||||
@ -176,15 +175,19 @@ golang.org/x/arch v0.0.0-20210923205945-b76863e36670/go.mod h1:5om86z9Hs0C8fWVUu
|
|||||||
golang.org/x/arch v0.3.0 h1:02VY4/ZcO/gBOH6PUaoiptASxtXU10jazRCP865E97k=
|
golang.org/x/arch v0.3.0 h1:02VY4/ZcO/gBOH6PUaoiptASxtXU10jazRCP865E97k=
|
||||||
golang.org/x/arch v0.3.0/go.mod h1:5om86z9Hs0C8fWVUuoMHwpExlXzs5Tkyp9hOrfG7pp8=
|
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-20210711020723-a769d52b0f97/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc=
|
||||||
golang.org/x/crypto v0.14.0 h1:wBqGXzWJW6m1XrIKlAH0Hs1JJ7+9KBwnIO8v66Q9cHc=
|
|
||||||
golang.org/x/crypto v0.14.0/go.mod h1:MVFd36DqK4CsrnJYDkBA3VC4m2GkXAM0PvzMCn4JQf4=
|
|
||||||
golang.org/x/crypto v0.17.0 h1:r8bRNjWL3GshPW3gkd+RpvzWrZAwPS49OmTGZ/uhM4k=
|
golang.org/x/crypto v0.17.0 h1:r8bRNjWL3GshPW3gkd+RpvzWrZAwPS49OmTGZ/uhM4k=
|
||||||
golang.org/x/crypto v0.17.0/go.mod h1:gCAAfMLgwOJRpTjQ2zCCt2OcSfYMTeZVSRtQlPC7Nq4=
|
golang.org/x/crypto v0.17.0/go.mod h1:gCAAfMLgwOJRpTjQ2zCCt2OcSfYMTeZVSRtQlPC7Nq4=
|
||||||
|
golang.org/x/crypto v0.21.0 h1:X31++rzVUdKhX5sWmSOFZxx8UW/ldWx55cbf08iNAMA=
|
||||||
|
golang.org/x/crypto v0.21.0/go.mod h1:0BP7YvVV9gBbVKyeTG0Gyn+gZm94bibOW5BjDEYAOMs=
|
||||||
golang.org/x/exp v0.0.0-20220303212507-bbda1eaf7a17 h1:3MTrJm4PyNL9NBqvYDSj3DHl46qQakyfqfWo4jgfaEM=
|
golang.org/x/exp v0.0.0-20220303212507-bbda1eaf7a17 h1:3MTrJm4PyNL9NBqvYDSj3DHl46qQakyfqfWo4jgfaEM=
|
||||||
golang.org/x/exp v0.0.0-20220303212507-bbda1eaf7a17/go.mod h1:lgLbSvA5ygNOMpwM/9anMpWVlVJ7Z+cHWq/eFuinpGE=
|
golang.org/x/exp v0.0.0-20220303212507-bbda1eaf7a17/go.mod h1:lgLbSvA5ygNOMpwM/9anMpWVlVJ7Z+cHWq/eFuinpGE=
|
||||||
|
golang.org/x/image v0.15.0 h1:kOELfmgrmJlw4Cdb7g/QGuB3CvDrXbqEIww/pNtNBm8=
|
||||||
|
golang.org/x/image v0.15.0/go.mod h1:HUYqC05R2ZcZ3ejNQsIHQDQiwWM4JBqmm6MKANTp4LE=
|
||||||
golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg=
|
golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg=
|
||||||
golang.org/x/net v0.17.0 h1:pVaXccu2ozPjCXewfr1S7xza/zcXTity9cCdXQYSjIM=
|
golang.org/x/net v0.17.0 h1:pVaXccu2ozPjCXewfr1S7xza/zcXTity9cCdXQYSjIM=
|
||||||
golang.org/x/net v0.17.0/go.mod h1:NxSsAGuq816PNPmqtQdLE42eU2Fs7NoRIZrHJAlaCOE=
|
golang.org/x/net v0.17.0/go.mod h1:NxSsAGuq816PNPmqtQdLE42eU2Fs7NoRIZrHJAlaCOE=
|
||||||
|
golang.org/x/net v0.21.0 h1:AQyQV4dYCvJ7vGmJyKki9+PBdyvhkSd8EIx/qb0AYv4=
|
||||||
|
golang.org/x/net v0.21.0/go.mod h1:bIjVDfnllIU7BJ2DNgfnXvpSvtn8VRwhlsaeUTyUS44=
|
||||||
golang.org/x/sync v0.1.0 h1:wsuoTGHzEhffawBOhz5CYhcrV4IdKZbEyZjBMuTp12o=
|
golang.org/x/sync v0.1.0 h1:wsuoTGHzEhffawBOhz5CYhcrV4IdKZbEyZjBMuTp12o=
|
||||||
golang.org/x/sync v0.1.0/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
golang.org/x/sync v0.1.0/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
||||||
golang.org/x/sys v0.0.0-20190916202348-b4ddaad3f8a3/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
golang.org/x/sys v0.0.0-20190916202348-b4ddaad3f8a3/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||||
@ -197,16 +200,14 @@ golang.org/x/sys v0.0.0-20220704084225-05e143d24a9e/go.mod h1:oPkhp1MJrh7nUepCBc
|
|||||||
golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||||
golang.org/x/sys v0.8.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
golang.org/x/sys v0.8.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||||
golang.org/x/sys v0.11.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
golang.org/x/sys v0.11.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||||
golang.org/x/sys v0.13.0 h1:Af8nKPmuFypiUBjVoU9V20FiaFXOcuZI21p0ycVYYGE=
|
|
||||||
golang.org/x/sys v0.13.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
|
||||||
golang.org/x/sys v0.15.0 h1:h48lPFYpsTvQJZF4EKyI4aLHaev3CxivZmv7yZig9pc=
|
golang.org/x/sys v0.15.0 h1:h48lPFYpsTvQJZF4EKyI4aLHaev3CxivZmv7yZig9pc=
|
||||||
golang.org/x/sys v0.15.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA=
|
golang.org/x/sys v0.15.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA=
|
||||||
|
golang.org/x/sys v0.18.0 h1:DBdB3niSjOA/O0blCZBqDefyWNYveAYMNF1Wum0DYQ4=
|
||||||
|
golang.org/x/sys v0.18.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA=
|
||||||
golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo=
|
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.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk=
|
||||||
golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
|
golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
|
||||||
golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
|
golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
|
||||||
golang.org/x/text v0.13.0 h1:ablQoSUd0tRdKxZewP80B+BaqeKJuVhuRxj/dkrun3k=
|
|
||||||
golang.org/x/text v0.13.0/go.mod h1:TvPlkZtksWOMsz7fbANvkp4WM8x/WCo/om8BMLbz+aE=
|
|
||||||
golang.org/x/text v0.14.0 h1:ScX5w1eTa3QqT8oi6+ziP7dTV1S2+ALU0bI+0zXKWiQ=
|
golang.org/x/text v0.14.0 h1:ScX5w1eTa3QqT8oi6+ziP7dTV1S2+ALU0bI+0zXKWiQ=
|
||||||
golang.org/x/text v0.14.0/go.mod h1:18ZOQIKpY8NJVqYksKHtTdi31H5itFRjB5/qKTNYzSU=
|
golang.org/x/text v0.14.0/go.mod h1:18ZOQIKpY8NJVqYksKHtTdi31H5itFRjB5/qKTNYzSU=
|
||||||
golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
|
golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
|
||||||
|
@ -155,6 +155,9 @@ func Distribute() func(c *gin.Context) {
|
|||||||
if channel.AutoBan != nil && *channel.AutoBan == 0 {
|
if channel.AutoBan != nil && *channel.AutoBan == 0 {
|
||||||
ban = false
|
ban = false
|
||||||
}
|
}
|
||||||
|
if nil != channel.OpenAIOrganization {
|
||||||
|
c.Set("channel_organization", *channel.OpenAIOrganization)
|
||||||
|
}
|
||||||
c.Set("auto_ban", ban)
|
c.Set("auto_ban", ban)
|
||||||
c.Set("model_mapping", channel.GetModelMapping())
|
c.Set("model_mapping", channel.GetModelMapping())
|
||||||
c.Request.Header.Set("Authorization", fmt.Sprintf("Bearer %s", channel.Key))
|
c.Request.Header.Set("Authorization", fmt.Sprintf("Bearer %s", channel.Key))
|
||||||
|
@ -4,18 +4,18 @@ type Midjourney struct {
|
|||||||
Id int `json:"id"`
|
Id int `json:"id"`
|
||||||
Code int `json:"code"`
|
Code int `json:"code"`
|
||||||
UserId int `json:"user_id" gorm:"index"`
|
UserId int `json:"user_id" gorm:"index"`
|
||||||
Action string `json:"action"`
|
Action string `json:"action" gorm:"type:varchar(40);index"`
|
||||||
MjId string `json:"mj_id" gorm:"index"`
|
MjId string `json:"mj_id" gorm:"index"`
|
||||||
Prompt string `json:"prompt"`
|
Prompt string `json:"prompt"`
|
||||||
PromptEn string `json:"prompt_en"`
|
PromptEn string `json:"prompt_en"`
|
||||||
Description string `json:"description"`
|
Description string `json:"description"`
|
||||||
State string `json:"state"`
|
State string `json:"state"`
|
||||||
SubmitTime int64 `json:"submit_time"`
|
SubmitTime int64 `json:"submit_time" gorm:"index"`
|
||||||
StartTime int64 `json:"start_time"`
|
StartTime int64 `json:"start_time" gorm:"index"`
|
||||||
FinishTime int64 `json:"finish_time"`
|
FinishTime int64 `json:"finish_time" gorm:"index"`
|
||||||
ImageUrl string `json:"image_url"`
|
ImageUrl string `json:"image_url"`
|
||||||
Status string `json:"status"`
|
Status string `json:"status" gorm:"type:varchar(20);index"`
|
||||||
Progress string `json:"progress"`
|
Progress string `json:"progress" gorm:"type:varchar(30);index"`
|
||||||
FailReason string `json:"fail_reason"`
|
FailReason string `json:"fail_reason"`
|
||||||
ChannelId int `json:"channel_id"`
|
ChannelId int `json:"channel_id"`
|
||||||
Quota int `json:"quota"`
|
Quota int `json:"quota"`
|
||||||
|
@ -2,6 +2,7 @@ package model
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"one-api/common"
|
"one-api/common"
|
||||||
|
"one-api/constant"
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
@ -91,6 +92,7 @@ func InitOptionMap() {
|
|||||||
common.OptionMap["DataExportInterval"] = strconv.Itoa(common.DataExportInterval)
|
common.OptionMap["DataExportInterval"] = strconv.Itoa(common.DataExportInterval)
|
||||||
common.OptionMap["DataExportDefaultTime"] = common.DataExportDefaultTime
|
common.OptionMap["DataExportDefaultTime"] = common.DataExportDefaultTime
|
||||||
common.OptionMap["DefaultCollapseSidebar"] = strconv.FormatBool(common.DefaultCollapseSidebar)
|
common.OptionMap["DefaultCollapseSidebar"] = strconv.FormatBool(common.DefaultCollapseSidebar)
|
||||||
|
common.OptionMap["MjNotifyEnabled"] = strconv.FormatBool(constant.MjNotifyEnabled)
|
||||||
|
|
||||||
common.OptionMapRWMutex.Unlock()
|
common.OptionMapRWMutex.Unlock()
|
||||||
loadOptionsFromDatabase()
|
loadOptionsFromDatabase()
|
||||||
@ -186,6 +188,8 @@ func updateOptionMap(key string, value string) (err error) {
|
|||||||
common.DataExportEnabled = boolValue
|
common.DataExportEnabled = boolValue
|
||||||
case "DefaultCollapseSidebar":
|
case "DefaultCollapseSidebar":
|
||||||
common.DefaultCollapseSidebar = boolValue
|
common.DefaultCollapseSidebar = boolValue
|
||||||
|
case "MjNotifyEnabled":
|
||||||
|
constant.MjNotifyEnabled = boolValue
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
switch key {
|
switch key {
|
||||||
|
@ -49,6 +49,9 @@ func (a *Adaptor) SetupRequestHeader(c *gin.Context, req *http.Request, info *re
|
|||||||
req.Header.Set("api-key", info.ApiKey)
|
req.Header.Set("api-key", info.ApiKey)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
if info.ChannelType == common.ChannelTypeOpenAI && "" != info.Organization {
|
||||||
|
req.Header.Set("OpenAI-Organization", info.Organization)
|
||||||
|
}
|
||||||
req.Header.Set("Authorization", "Bearer "+info.ApiKey)
|
req.Header.Set("Authorization", "Bearer "+info.ApiKey)
|
||||||
//if info.ChannelType == common.ChannelTypeOpenRouter {
|
//if info.ChannelType == common.ChannelTypeOpenRouter {
|
||||||
// req.Header.Set("HTTP-Referer", "https://github.com/songquanpeng/one-api")
|
// req.Header.Set("HTTP-Referer", "https://github.com/songquanpeng/one-api")
|
||||||
|
63
relay/channel/perplexity/adaptor.go
Normal file
63
relay/channel/perplexity/adaptor.go
Normal file
@ -0,0 +1,63 @@
|
|||||||
|
package perplexity
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"github.com/gin-gonic/gin"
|
||||||
|
"io"
|
||||||
|
"net/http"
|
||||||
|
"one-api/dto"
|
||||||
|
"one-api/relay/channel"
|
||||||
|
"one-api/relay/channel/openai"
|
||||||
|
relaycommon "one-api/relay/common"
|
||||||
|
"one-api/service"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Adaptor struct {
|
||||||
|
}
|
||||||
|
|
||||||
|
func (a *Adaptor) Init(info *relaycommon.RelayInfo, request dto.GeneralOpenAIRequest) {
|
||||||
|
}
|
||||||
|
|
||||||
|
func (a *Adaptor) GetRequestURL(info *relaycommon.RelayInfo) (string, error) {
|
||||||
|
return fmt.Sprintf("%s/chat/completions", info.BaseUrl), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (a *Adaptor) SetupRequestHeader(c *gin.Context, req *http.Request, info *relaycommon.RelayInfo) error {
|
||||||
|
channel.SetupApiRequestHeader(info, c, req)
|
||||||
|
req.Header.Set("Authorization", "Bearer "+info.ApiKey)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (a *Adaptor) ConvertRequest(c *gin.Context, relayMode int, request *dto.GeneralOpenAIRequest) (any, error) {
|
||||||
|
if request == nil {
|
||||||
|
return nil, errors.New("request is nil")
|
||||||
|
}
|
||||||
|
if request.TopP >= 1 {
|
||||||
|
request.TopP = 0.99
|
||||||
|
}
|
||||||
|
return requestOpenAI2Perplexity(*request), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (a *Adaptor) DoRequest(c *gin.Context, info *relaycommon.RelayInfo, requestBody io.Reader) (*http.Response, error) {
|
||||||
|
return channel.DoApiRequest(a, c, info, requestBody)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (a *Adaptor) DoResponse(c *gin.Context, resp *http.Response, info *relaycommon.RelayInfo) (usage *dto.Usage, err *dto.OpenAIErrorWithStatusCode) {
|
||||||
|
if info.IsStream {
|
||||||
|
var responseText string
|
||||||
|
err, responseText = openai.OpenaiStreamHandler(c, resp, info.RelayMode)
|
||||||
|
usage = service.ResponseText2Usage(responseText, info.UpstreamModelName, info.PromptTokens)
|
||||||
|
} else {
|
||||||
|
err, usage = openai.OpenaiHandler(c, resp, info.PromptTokens, info.UpstreamModelName)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func (a *Adaptor) GetModelList() []string {
|
||||||
|
return ModelList
|
||||||
|
}
|
||||||
|
|
||||||
|
func (a *Adaptor) GetChannelName() string {
|
||||||
|
return ChannelName
|
||||||
|
}
|
7
relay/channel/perplexity/constants.go
Normal file
7
relay/channel/perplexity/constants.go
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
package perplexity
|
||||||
|
|
||||||
|
var ModelList = []string{
|
||||||
|
"sonar-small-chat", "sonar-small-online", "sonar-medium-chat", "sonar-medium-online", "mistral-7b-instruct", "mixtral-8x7b-instruct",
|
||||||
|
}
|
||||||
|
|
||||||
|
var ChannelName = "perplexity"
|
21
relay/channel/perplexity/relay-perplexity.go
Normal file
21
relay/channel/perplexity/relay-perplexity.go
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
package perplexity
|
||||||
|
|
||||||
|
import "one-api/dto"
|
||||||
|
|
||||||
|
func requestOpenAI2Perplexity(request dto.GeneralOpenAIRequest) *dto.GeneralOpenAIRequest {
|
||||||
|
messages := make([]dto.Message, 0, len(request.Messages))
|
||||||
|
for _, message := range request.Messages {
|
||||||
|
messages = append(messages, dto.Message{
|
||||||
|
Role: message.Role,
|
||||||
|
Content: message.Content,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
return &dto.GeneralOpenAIRequest{
|
||||||
|
Model: request.Model,
|
||||||
|
Stream: request.Stream,
|
||||||
|
Messages: messages,
|
||||||
|
Temperature: request.Temperature,
|
||||||
|
TopP: request.TopP,
|
||||||
|
MaxTokens: request.MaxTokens,
|
||||||
|
}
|
||||||
|
}
|
@ -36,6 +36,9 @@ func (a *Adaptor) ConvertRequest(c *gin.Context, relayMode int, request *dto.Gen
|
|||||||
if request == nil {
|
if request == nil {
|
||||||
return nil, errors.New("request is nil")
|
return nil, errors.New("request is nil")
|
||||||
}
|
}
|
||||||
|
if request.TopP >= 1 {
|
||||||
|
request.TopP = 0.99
|
||||||
|
}
|
||||||
return requestOpenAI2Zhipu(*request), nil
|
return requestOpenAI2Zhipu(*request), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -34,6 +34,9 @@ func (a *Adaptor) ConvertRequest(c *gin.Context, relayMode int, request *dto.Gen
|
|||||||
if request == nil {
|
if request == nil {
|
||||||
return nil, errors.New("request is nil")
|
return nil, errors.New("request is nil")
|
||||||
}
|
}
|
||||||
|
if request.TopP >= 1 {
|
||||||
|
request.TopP = 0.99
|
||||||
|
}
|
||||||
return requestOpenAI2Zhipu(*request), nil
|
return requestOpenAI2Zhipu(*request), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -24,6 +24,7 @@ type RelayInfo struct {
|
|||||||
ApiVersion string
|
ApiVersion string
|
||||||
PromptTokens int
|
PromptTokens int
|
||||||
ApiKey string
|
ApiKey string
|
||||||
|
Organization string
|
||||||
BaseUrl string
|
BaseUrl string
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -52,6 +53,7 @@ func GenRelayInfo(c *gin.Context) *RelayInfo {
|
|||||||
ApiType: apiType,
|
ApiType: apiType,
|
||||||
ApiVersion: c.GetString("api_version"),
|
ApiVersion: c.GetString("api_version"),
|
||||||
ApiKey: strings.TrimPrefix(c.Request.Header.Get("Authorization"), "Bearer "),
|
ApiKey: strings.TrimPrefix(c.Request.Header.Get("Authorization"), "Bearer "),
|
||||||
|
Organization: c.GetString("channel_organization"),
|
||||||
}
|
}
|
||||||
if info.BaseUrl == "" {
|
if info.BaseUrl == "" {
|
||||||
info.BaseUrl = common.ChannelBaseURLs[channelType]
|
info.BaseUrl = common.ChannelBaseURLs[channelType]
|
||||||
|
@ -16,6 +16,8 @@ const (
|
|||||||
APITypeTencent
|
APITypeTencent
|
||||||
APITypeGemini
|
APITypeGemini
|
||||||
APITypeZhipu_v4
|
APITypeZhipu_v4
|
||||||
|
APITypeOllama
|
||||||
|
APITypePerplexity
|
||||||
|
|
||||||
APITypeDummy // this one is only for count, do not add any channel after this
|
APITypeDummy // this one is only for count, do not add any channel after this
|
||||||
)
|
)
|
||||||
@ -43,6 +45,10 @@ func ChannelType2APIType(channelType int) int {
|
|||||||
apiType = APITypeGemini
|
apiType = APITypeGemini
|
||||||
case common.ChannelTypeZhipu_v4:
|
case common.ChannelTypeZhipu_v4:
|
||||||
apiType = APITypeZhipu_v4
|
apiType = APITypeZhipu_v4
|
||||||
|
case common.ChannelTypeOllama:
|
||||||
|
apiType = APITypeOllama
|
||||||
|
case common.ChannelTypePerplexity:
|
||||||
|
apiType = APITypePerplexity
|
||||||
}
|
}
|
||||||
return apiType
|
return apiType
|
||||||
}
|
}
|
||||||
|
@ -266,24 +266,6 @@ func RelayMidjourneyTaskImageSeed(c *gin.Context) *dto.MidjourneyResponse {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return &midjResponseWithStatus.Response
|
return &midjResponseWithStatus.Response
|
||||||
}
|
}
|
||||||
//defer func(ctx context.Context) {
|
|
||||||
// err := model.PostConsumeTokenQuota(tokenId, userQuota, quota, 0, true)
|
|
||||||
// if err != nil {
|
|
||||||
// common.SysError("error consuming token remain quota: " + err.Error())
|
|
||||||
// }
|
|
||||||
// err = model.CacheUpdateUserQuota(userId)
|
|
||||||
// if err != nil {
|
|
||||||
// common.SysError("error update user quota cache: " + err.Error())
|
|
||||||
// }
|
|
||||||
// if quota != 0 {
|
|
||||||
// tokenName := c.GetString("token_name")
|
|
||||||
// logContent := fmt.Sprintf("模型固定价格 %.2f,分组倍率 %.2f,操作 %s", modelPrice, groupRatio, midjRequest.Action)
|
|
||||||
// model.RecordConsumeLog(ctx, userId, channelId, 0, 0, modelName, tokenName, quota, logContent, tokenId, userQuota, 0, false)
|
|
||||||
// model.UpdateUserUsedQuotaAndRequestCount(userId, quota)
|
|
||||||
// channelId := c.GetInt("channel_id")
|
|
||||||
// model.UpdateChannelUsedQuota(channelId, quota)
|
|
||||||
// }
|
|
||||||
//}(c.Request.Context())
|
|
||||||
midjResponse := &midjResponseWithStatus.Response
|
midjResponse := &midjResponseWithStatus.Response
|
||||||
c.Writer.WriteHeader(midjResponseWithStatus.StatusCode)
|
c.Writer.WriteHeader(midjResponseWithStatus.StatusCode)
|
||||||
respBody, err := json.Marshal(midjResponse)
|
respBody, err := json.Marshal(midjResponse)
|
||||||
|
@ -6,8 +6,10 @@ import (
|
|||||||
"one-api/relay/channel/baidu"
|
"one-api/relay/channel/baidu"
|
||||||
"one-api/relay/channel/claude"
|
"one-api/relay/channel/claude"
|
||||||
"one-api/relay/channel/gemini"
|
"one-api/relay/channel/gemini"
|
||||||
|
"one-api/relay/channel/ollama"
|
||||||
"one-api/relay/channel/openai"
|
"one-api/relay/channel/openai"
|
||||||
"one-api/relay/channel/palm"
|
"one-api/relay/channel/palm"
|
||||||
|
"one-api/relay/channel/perplexity"
|
||||||
"one-api/relay/channel/tencent"
|
"one-api/relay/channel/tencent"
|
||||||
"one-api/relay/channel/xunfei"
|
"one-api/relay/channel/xunfei"
|
||||||
"one-api/relay/channel/zhipu"
|
"one-api/relay/channel/zhipu"
|
||||||
@ -39,6 +41,10 @@ func GetAdaptor(apiType int) channel.Adaptor {
|
|||||||
return &zhipu.Adaptor{}
|
return &zhipu.Adaptor{}
|
||||||
case constant.APITypeZhipu_v4:
|
case constant.APITypeZhipu_v4:
|
||||||
return &zhipu_4v.Adaptor{}
|
return &zhipu_4v.Adaptor{}
|
||||||
|
case constant.APITypeOllama:
|
||||||
|
return &ollama.Adaptor{}
|
||||||
|
case constant.APITypePerplexity:
|
||||||
|
return &perplexity.Adaptor{}
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
@ -7,6 +7,7 @@ import (
|
|||||||
"io"
|
"io"
|
||||||
"log"
|
"log"
|
||||||
"net/http"
|
"net/http"
|
||||||
|
"one-api/common"
|
||||||
"one-api/constant"
|
"one-api/constant"
|
||||||
"one-api/dto"
|
"one-api/dto"
|
||||||
relayconstant "one-api/relay/constant"
|
relayconstant "one-api/relay/constant"
|
||||||
@ -158,14 +159,19 @@ func DoMidjourneyHttpRequest(c *gin.Context, timeout time.Duration, fullRequestU
|
|||||||
//requestBody = c.Request.Body
|
//requestBody = c.Request.Body
|
||||||
// read request body to json, delete accountFilter and notifyHook
|
// read request body to json, delete accountFilter and notifyHook
|
||||||
var mapResult map[string]interface{}
|
var mapResult map[string]interface{}
|
||||||
err := json.NewDecoder(c.Request.Body).Decode(&mapResult)
|
// if get request, no need to read request body
|
||||||
if err != nil {
|
if c.Request.Method != "GET" {
|
||||||
return MidjourneyErrorWithStatusCodeWrapper(constant.MjErrorUnknown, "read_request_body_failed", http.StatusInternalServerError), nullBytes, err
|
err := json.NewDecoder(c.Request.Body).Decode(&mapResult)
|
||||||
|
if err != nil {
|
||||||
|
return MidjourneyErrorWithStatusCodeWrapper(constant.MjErrorUnknown, "read_request_body_failed", http.StatusInternalServerError), nullBytes, err
|
||||||
|
}
|
||||||
|
delete(mapResult, "accountFilter")
|
||||||
|
if !constant.MjNotifyEnabled {
|
||||||
|
delete(mapResult, "notifyHook")
|
||||||
|
}
|
||||||
|
//req, err := http.NewRequest(c.Request.Method, fullRequestURL, requestBody)
|
||||||
|
// make new request with mapResult
|
||||||
}
|
}
|
||||||
delete(mapResult, "accountFilter")
|
|
||||||
delete(mapResult, "notifyHook")
|
|
||||||
//req, err := http.NewRequest(c.Request.Method, fullRequestURL, requestBody)
|
|
||||||
// make new request with mapResult
|
|
||||||
reqBody, err := json.Marshal(mapResult)
|
reqBody, err := json.Marshal(mapResult)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return MidjourneyErrorWithStatusCodeWrapper(constant.MjErrorUnknown, "marshal_request_body_failed", http.StatusInternalServerError), nullBytes, err
|
return MidjourneyErrorWithStatusCodeWrapper(constant.MjErrorUnknown, "marshal_request_body_failed", http.StatusInternalServerError), nullBytes, err
|
||||||
@ -183,10 +189,11 @@ func DoMidjourneyHttpRequest(c *gin.Context, timeout time.Duration, fullRequestU
|
|||||||
defer cancel()
|
defer cancel()
|
||||||
resp, err := GetHttpClient().Do(req)
|
resp, err := GetHttpClient().Do(req)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
common.SysError("do request failed: " + err.Error())
|
||||||
return MidjourneyErrorWithStatusCodeWrapper(constant.MjErrorUnknown, "do_request_failed", http.StatusInternalServerError), nullBytes, err
|
return MidjourneyErrorWithStatusCodeWrapper(constant.MjErrorUnknown, "do_request_failed", http.StatusInternalServerError), nullBytes, err
|
||||||
}
|
}
|
||||||
statusCode := resp.StatusCode
|
statusCode := resp.StatusCode
|
||||||
//if statusCode != 200 {
|
//if statusCode != 200 {
|
||||||
// return MidjourneyErrorWithStatusCodeWrapper(constant.MjErrorUnknown, "bad_response_status_code", statusCode), nullBytes, nil
|
// return MidjourneyErrorWithStatusCodeWrapper(constant.MjErrorUnknown, "bad_response_status_code", statusCode), nullBytes, nil
|
||||||
//}
|
//}
|
||||||
err = req.Body.Close()
|
err = req.Body.Close()
|
||||||
@ -207,11 +214,15 @@ func DoMidjourneyHttpRequest(c *gin.Context, timeout time.Duration, fullRequestU
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return MidjourneyErrorWithStatusCodeWrapper(constant.MjErrorUnknown, "close_response_body_failed", statusCode), responseBody, err
|
return MidjourneyErrorWithStatusCodeWrapper(constant.MjErrorUnknown, "close_response_body_failed", statusCode), responseBody, err
|
||||||
}
|
}
|
||||||
|
respStr := string(responseBody)
|
||||||
err = json.Unmarshal(responseBody, &midjResponse)
|
log.Printf("responseBody: %s", respStr)
|
||||||
log.Printf("responseBody: %s", string(responseBody))
|
if respStr == "" {
|
||||||
if err != nil {
|
return MidjourneyErrorWithStatusCodeWrapper(constant.MjErrorUnknown, "empty_response_body", statusCode), responseBody, nil
|
||||||
return MidjourneyErrorWithStatusCodeWrapper(constant.MjErrorUnknown, "unmarshal_response_body_failed", statusCode), responseBody, err
|
} else {
|
||||||
|
err = json.Unmarshal(responseBody, &midjResponse)
|
||||||
|
if err != nil {
|
||||||
|
return MidjourneyErrorWithStatusCodeWrapper(constant.MjErrorUnknown, "unmarshal_response_body_failed", statusCode), responseBody, err
|
||||||
|
}
|
||||||
}
|
}
|
||||||
//log.Printf("midjResponse: %v", midjResponse)
|
//log.Printf("midjResponse: %v", midjResponse)
|
||||||
//for k, v := range resp.Header {
|
//for k, v := range resp.Header {
|
||||||
|
@ -49,7 +49,7 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"prettier": "^2.7.1",
|
"prettier": "2.8.8",
|
||||||
"typescript": "4.4.2"
|
"typescript": "4.4.2"
|
||||||
},
|
},
|
||||||
"prettier": {
|
"prettier": {
|
||||||
|
386
web/src/App.js
386
web/src/App.js
@ -8,13 +8,12 @@ import LoginForm from './components/LoginForm';
|
|||||||
import NotFound from './pages/NotFound';
|
import NotFound from './pages/NotFound';
|
||||||
import Setting from './pages/Setting';
|
import Setting from './pages/Setting';
|
||||||
import EditUser from './pages/User/EditUser';
|
import EditUser from './pages/User/EditUser';
|
||||||
import { API, getLogo, getSystemName, showError, showNotice } from './helpers';
|
import { getLogo, getSystemName } from './helpers';
|
||||||
import PasswordResetForm from './components/PasswordResetForm';
|
import PasswordResetForm from './components/PasswordResetForm';
|
||||||
import GitHubOAuth from './components/GitHubOAuth';
|
import GitHubOAuth from './components/GitHubOAuth';
|
||||||
import LinuxDoOAuth from "./components/LinuxDoOAuth";
|
import LinuxDoOAuth from "./components/LinuxDoOAuth";
|
||||||
import PasswordResetConfirm from './components/PasswordResetConfirm';
|
import PasswordResetConfirm from './components/PasswordResetConfirm';
|
||||||
import { UserContext } from './context/User';
|
import { UserContext } from './context/User';
|
||||||
import { StatusContext } from './context/Status';
|
|
||||||
import Channel from './pages/Channel';
|
import Channel from './pages/Channel';
|
||||||
import Token from './pages/Token';
|
import Token from './pages/Token';
|
||||||
import EditChannel from './pages/Channel/EditChannel';
|
import EditChannel from './pages/Channel/EditChannel';
|
||||||
@ -22,12 +21,13 @@ import Redemption from './pages/Redemption';
|
|||||||
import TopUp from './pages/TopUp';
|
import TopUp from './pages/TopUp';
|
||||||
import Log from './pages/Log';
|
import Log from './pages/Log';
|
||||||
import Chat from './pages/Chat';
|
import Chat from './pages/Chat';
|
||||||
import {Layout} from "@douyinfe/semi-ui";
|
import { Layout } from '@douyinfe/semi-ui';
|
||||||
import Midjourney from "./pages/Midjourney";
|
import Midjourney from './pages/Midjourney';
|
||||||
import Detail from "./pages/Detail";
|
import Detail from './pages/Detail';
|
||||||
|
|
||||||
const Home = lazy(() => import('./pages/Home'));
|
const Home = lazy(() => import('./pages/Home'));
|
||||||
const About = lazy(() => import('./pages/About'));
|
const About = lazy(() => import('./pages/About'));
|
||||||
|
|
||||||
function App() {
|
function App() {
|
||||||
const [userState, userDispatch] = useContext(UserContext);
|
const [userState, userDispatch] = useContext(UserContext);
|
||||||
// const [statusState, statusDispatch] = useContext(StatusContext);
|
// const [statusState, statusDispatch] = useContext(StatusContext);
|
||||||
@ -48,7 +48,7 @@ function App() {
|
|||||||
}
|
}
|
||||||
let logo = getLogo();
|
let logo = getLogo();
|
||||||
if (logo) {
|
if (logo) {
|
||||||
let linkElement = document.querySelector("link[rel~='icon']");
|
let linkElement = document.querySelector('link[rel~=\'icon\']');
|
||||||
if (linkElement) {
|
if (linkElement) {
|
||||||
linkElement.href = logo;
|
linkElement.href = logo;
|
||||||
}
|
}
|
||||||
@ -57,193 +57,193 @@ function App() {
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<Layout>
|
<Layout>
|
||||||
<Layout.Content>
|
<Layout.Content>
|
||||||
<Routes>
|
<Routes>
|
||||||
<Route
|
<Route
|
||||||
path='/'
|
path="/"
|
||||||
element={
|
element={
|
||||||
<Suspense fallback={<Loading></Loading>}>
|
<Suspense fallback={<Loading></Loading>}>
|
||||||
<Home />
|
<Home />
|
||||||
</Suspense>
|
</Suspense>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
<Route
|
<Route
|
||||||
path='/channel'
|
path="/channel"
|
||||||
element={
|
element={
|
||||||
<PrivateRoute>
|
<PrivateRoute>
|
||||||
<Channel />
|
<Channel />
|
||||||
</PrivateRoute>
|
</PrivateRoute>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
<Route
|
<Route
|
||||||
path='/channel/edit/:id'
|
path="/channel/edit/:id"
|
||||||
element={
|
element={
|
||||||
<Suspense fallback={<Loading></Loading>}>
|
<Suspense fallback={<Loading></Loading>}>
|
||||||
<EditChannel />
|
<EditChannel />
|
||||||
</Suspense>
|
</Suspense>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
<Route
|
<Route
|
||||||
path='/channel/add'
|
path="/channel/add"
|
||||||
element={
|
element={
|
||||||
<Suspense fallback={<Loading></Loading>}>
|
<Suspense fallback={<Loading></Loading>}>
|
||||||
<EditChannel />
|
<EditChannel />
|
||||||
</Suspense>
|
</Suspense>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
<Route
|
<Route
|
||||||
path='/token'
|
path="/token"
|
||||||
element={
|
element={
|
||||||
<PrivateRoute>
|
<PrivateRoute>
|
||||||
<Token />
|
<Token />
|
||||||
</PrivateRoute>
|
</PrivateRoute>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
<Route
|
<Route
|
||||||
path='/redemption'
|
path="/redemption"
|
||||||
element={
|
element={
|
||||||
<PrivateRoute>
|
<PrivateRoute>
|
||||||
<Redemption />
|
<Redemption />
|
||||||
</PrivateRoute>
|
</PrivateRoute>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
<Route
|
<Route
|
||||||
path='/user'
|
path="/user"
|
||||||
element={
|
element={
|
||||||
<PrivateRoute>
|
<PrivateRoute>
|
||||||
<User />
|
<User />
|
||||||
</PrivateRoute>
|
</PrivateRoute>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
<Route
|
<Route
|
||||||
path='/user/edit/:id'
|
path="/user/edit/:id"
|
||||||
element={
|
element={
|
||||||
<Suspense fallback={<Loading></Loading>}>
|
<Suspense fallback={<Loading></Loading>}>
|
||||||
<EditUser />
|
<EditUser />
|
||||||
</Suspense>
|
</Suspense>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
<Route
|
<Route
|
||||||
path='/user/edit'
|
path="/user/edit"
|
||||||
element={
|
element={
|
||||||
<Suspense fallback={<Loading></Loading>}>
|
<Suspense fallback={<Loading></Loading>}>
|
||||||
<EditUser />
|
<EditUser />
|
||||||
</Suspense>
|
</Suspense>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
<Route
|
<Route
|
||||||
path='/user/reset'
|
path="/user/reset"
|
||||||
element={
|
element={
|
||||||
<Suspense fallback={<Loading></Loading>}>
|
<Suspense fallback={<Loading></Loading>}>
|
||||||
<PasswordResetConfirm />
|
<PasswordResetConfirm />
|
||||||
</Suspense>
|
</Suspense>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
<Route
|
<Route
|
||||||
path='/login'
|
path="/login"
|
||||||
element={
|
element={
|
||||||
<Suspense fallback={<Loading></Loading>}>
|
<Suspense fallback={<Loading></Loading>}>
|
||||||
<LoginForm />
|
<LoginForm />
|
||||||
</Suspense>
|
</Suspense>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
<Route
|
<Route
|
||||||
path='/register'
|
path="/register"
|
||||||
element={
|
element={
|
||||||
<Suspense fallback={<Loading></Loading>}>
|
<Suspense fallback={<Loading></Loading>}>
|
||||||
<RegisterForm />
|
<RegisterForm />
|
||||||
</Suspense>
|
</Suspense>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
<Route
|
<Route
|
||||||
path='/reset'
|
path="/reset"
|
||||||
element={
|
element={
|
||||||
<Suspense fallback={<Loading></Loading>}>
|
<Suspense fallback={<Loading></Loading>}>
|
||||||
<PasswordResetForm />
|
<PasswordResetForm />
|
||||||
</Suspense>
|
</Suspense>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
<Route
|
<Route
|
||||||
path='/oauth/github'
|
path="/oauth/github"
|
||||||
element={
|
element={
|
||||||
<Suspense fallback={<Loading></Loading>}>
|
<Suspense fallback={<Loading></Loading>}>
|
||||||
<GitHubOAuth />
|
<GitHubOAuth />
|
||||||
</Suspense>
|
</Suspense>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
<Route
|
<Route
|
||||||
path='/oauth/linuxdo'
|
path="/oauth/linuxdo"
|
||||||
element={
|
element={
|
||||||
<Suspense fallback={<Loading></Loading>}>
|
<Suspense fallback={<Loading></Loading>}>
|
||||||
<LinuxDoOAuth />
|
<LinuxDoOAuth />
|
||||||
</Suspense>
|
</Suspense>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
<Route
|
<Route
|
||||||
path='/setting'
|
path="/setting"
|
||||||
element={
|
element={
|
||||||
<PrivateRoute>
|
<PrivateRoute>
|
||||||
<Suspense fallback={<Loading></Loading>}>
|
<Suspense fallback={<Loading></Loading>}>
|
||||||
<Setting />
|
<Setting />
|
||||||
</Suspense>
|
</Suspense>
|
||||||
</PrivateRoute>
|
</PrivateRoute>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
<Route
|
<Route
|
||||||
path='/topup'
|
path="/topup"
|
||||||
element={
|
element={
|
||||||
<PrivateRoute>
|
<PrivateRoute>
|
||||||
<Suspense fallback={<Loading></Loading>}>
|
<Suspense fallback={<Loading></Loading>}>
|
||||||
<TopUp />
|
<TopUp />
|
||||||
</Suspense>
|
</Suspense>
|
||||||
</PrivateRoute>
|
</PrivateRoute>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
<Route
|
<Route
|
||||||
path='/log'
|
path="/log"
|
||||||
element={
|
element={
|
||||||
<PrivateRoute>
|
<PrivateRoute>
|
||||||
<Log />
|
<Log />
|
||||||
</PrivateRoute>
|
</PrivateRoute>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
<Route
|
<Route
|
||||||
path='/detail'
|
path="/detail"
|
||||||
element={
|
element={
|
||||||
<PrivateRoute>
|
<PrivateRoute>
|
||||||
<Detail />
|
<Detail />
|
||||||
</PrivateRoute>
|
</PrivateRoute>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
<Route
|
<Route
|
||||||
path='/midjourney'
|
path="/midjourney"
|
||||||
element={
|
element={
|
||||||
<PrivateRoute>
|
<PrivateRoute>
|
||||||
<Midjourney />
|
<Midjourney />
|
||||||
</PrivateRoute>
|
</PrivateRoute>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
<Route
|
<Route
|
||||||
path='/about'
|
path="/about"
|
||||||
element={
|
element={
|
||||||
<Suspense fallback={<Loading></Loading>}>
|
<Suspense fallback={<Loading></Loading>}>
|
||||||
<About />
|
<About />
|
||||||
</Suspense>
|
</Suspense>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
<Route
|
<Route
|
||||||
path='/chat'
|
path="/chat"
|
||||||
element={
|
element={
|
||||||
<Suspense fallback={<Loading></Loading>}>
|
<Suspense fallback={<Loading></Loading>}>
|
||||||
<Chat />
|
<Chat />
|
||||||
</Suspense>
|
</Suspense>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
<Route path='*' element={
|
<Route path="*" element={
|
||||||
<NotFound />
|
<NotFound />
|
||||||
} />
|
} />
|
||||||
</Routes>
|
</Routes>
|
||||||
</Layout.Content>
|
</Layout.Content>
|
||||||
</Layout>
|
</Layout>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
File diff suppressed because it is too large
Load Diff
@ -1,7 +1,7 @@
|
|||||||
import React, { useEffect, useState } from 'react';
|
import React, { useEffect, useState } from 'react';
|
||||||
|
|
||||||
import { getFooterHTML, getSystemName } from '../helpers';
|
import { getFooterHTML, getSystemName } from '../helpers';
|
||||||
import {Layout} from "@douyinfe/semi-ui";
|
import { Layout } from '@douyinfe/semi-ui';
|
||||||
|
|
||||||
const Footer = () => {
|
const Footer = () => {
|
||||||
const systemName = getSystemName();
|
const systemName = getSystemName();
|
||||||
@ -29,30 +29,30 @@ const Footer = () => {
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<Layout>
|
<Layout>
|
||||||
<Layout.Content style={{textAlign: 'center'}}>
|
<Layout.Content style={{ textAlign: 'center' }}>
|
||||||
{footer ? (
|
{footer ? (
|
||||||
<div
|
<div
|
||||||
className='custom-footer'
|
className="custom-footer"
|
||||||
dangerouslySetInnerHTML={{ __html: footer }}
|
dangerouslySetInnerHTML={{ __html: footer }}
|
||||||
></div>
|
></div>
|
||||||
) : (
|
) : (
|
||||||
<div className='custom-footer'>
|
<div className="custom-footer">
|
||||||
<a
|
<a
|
||||||
href='https://github.com/Calcium-Ion/new-api'
|
href="https://github.com/Calcium-Ion/new-api"
|
||||||
target='_blank'
|
target="_blank" rel="noreferrer"
|
||||||
>
|
>
|
||||||
New API {process.env.REACT_APP_VERSION}{' '}
|
New API {process.env.REACT_APP_VERSION}{' '}
|
||||||
</a>
|
</a>
|
||||||
由{' '}
|
由{' '}
|
||||||
<a href='https://github.com/Calcium-Ion' target='_blank'>
|
<a href="https://github.com/Calcium-Ion" target="_blank" rel="noreferrer">
|
||||||
Calcium-Ion
|
Calcium-Ion
|
||||||
</a>{' '}
|
</a>{' '}
|
||||||
开发,基于{' '}
|
开发,基于{' '}
|
||||||
<a href='https://github.com/songquanpeng/one-api' target='_blank'>
|
<a href="https://github.com/songquanpeng/one-api" target="_blank" rel="noreferrer">
|
||||||
One API v0.5.4
|
One API v0.5.4
|
||||||
</a>{' '}
|
</a>{' '}
|
||||||
,本项目根据{' '}
|
,本项目根据{' '}
|
||||||
<a href='https://opensource.org/licenses/mit-license.php'>
|
<a href="https://opensource.org/licenses/mit-license.php">
|
||||||
MIT 许可证
|
MIT 许可证
|
||||||
</a>{' '}
|
</a>{' '}
|
||||||
授权
|
授权
|
||||||
|
@ -58,7 +58,7 @@ const GitHubOAuth = () => {
|
|||||||
return (
|
return (
|
||||||
<Segment style={{ minHeight: '300px' }}>
|
<Segment style={{ minHeight: '300px' }}>
|
||||||
<Dimmer active inverted>
|
<Dimmer active inverted>
|
||||||
<Loader size='large'>{prompt}</Loader>
|
<Loader size="large">{prompt}</Loader>
|
||||||
</Dimmer>
|
</Dimmer>
|
||||||
</Segment>
|
</Segment>
|
||||||
);
|
);
|
||||||
|
@ -1,165 +1,161 @@
|
|||||||
import React, {useContext, useEffect, useRef, useState} from 'react';
|
import React, { useContext, useEffect, useState } from 'react';
|
||||||
import {Link, useNavigate} from 'react-router-dom';
|
import { Link, useNavigate } from 'react-router-dom';
|
||||||
import {UserContext} from '../context/User';
|
import { UserContext } from '../context/User';
|
||||||
|
|
||||||
import {API, getLogo, getSystemName, isAdmin, isMobile, showSuccess} from '../helpers';
|
import { API, getLogo, getSystemName, showSuccess } from '../helpers';
|
||||||
import '../index.css';
|
import '../index.css';
|
||||||
|
|
||||||
import fireworks from 'react-fireworks';
|
import fireworks from 'react-fireworks';
|
||||||
|
|
||||||
import {
|
import { IconHelpCircle, IconKey, IconUser } from '@douyinfe/semi-icons';
|
||||||
IconKey,
|
import { Avatar, Dropdown, Layout, Nav, Switch } from '@douyinfe/semi-ui';
|
||||||
IconUser,
|
import { stringToColor } from '../helpers/render';
|
||||||
IconHelpCircle
|
|
||||||
} from '@douyinfe/semi-icons';
|
|
||||||
import {Nav, Avatar, Dropdown, Layout, Switch} from '@douyinfe/semi-ui';
|
|
||||||
import {stringToColor} from "../helpers/render";
|
|
||||||
|
|
||||||
// HeaderBar Buttons
|
// HeaderBar Buttons
|
||||||
let headerButtons = [
|
let headerButtons = [
|
||||||
{
|
{
|
||||||
text: '关于',
|
text: '关于',
|
||||||
itemKey: 'about',
|
itemKey: 'about',
|
||||||
to: '/about',
|
to: '/about',
|
||||||
icon: <IconHelpCircle/>
|
icon: <IconHelpCircle />
|
||||||
},
|
}
|
||||||
];
|
];
|
||||||
|
|
||||||
if (localStorage.getItem('chat_link')) {
|
if (localStorage.getItem('chat_link')) {
|
||||||
headerButtons.splice(1, 0, {
|
headerButtons.splice(1, 0, {
|
||||||
name: '聊天',
|
name: '聊天',
|
||||||
to: '/chat',
|
to: '/chat',
|
||||||
icon: 'comments'
|
icon: 'comments'
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
const HeaderBar = () => {
|
const HeaderBar = () => {
|
||||||
const [userState, userDispatch] = useContext(UserContext);
|
const [userState, userDispatch] = useContext(UserContext);
|
||||||
let navigate = useNavigate();
|
let navigate = useNavigate();
|
||||||
|
|
||||||
const [showSidebar, setShowSidebar] = useState(false);
|
const [showSidebar, setShowSidebar] = useState(false);
|
||||||
const [dark, setDark] = useState(false);
|
const [dark, setDark] = useState(false);
|
||||||
const systemName = getSystemName();
|
const systemName = getSystemName();
|
||||||
const logo = getLogo();
|
const logo = getLogo();
|
||||||
var themeMode = localStorage.getItem('theme-mode');
|
var themeMode = localStorage.getItem('theme-mode');
|
||||||
const currentDate = new Date();
|
const currentDate = new Date();
|
||||||
// enable fireworks on new year(1.1 and 2.9-2.24)
|
// enable fireworks on new year(1.1 and 2.9-2.24)
|
||||||
const isNewYear = (currentDate.getMonth() === 0 && currentDate.getDate() === 1) || (currentDate.getMonth() === 1 && currentDate.getDate() >= 9 && currentDate.getDate() <= 24);
|
const isNewYear = (currentDate.getMonth() === 0 && currentDate.getDate() === 1) || (currentDate.getMonth() === 1 && currentDate.getDate() >= 9 && currentDate.getDate() <= 24);
|
||||||
|
|
||||||
async function logout() {
|
async function logout() {
|
||||||
setShowSidebar(false);
|
setShowSidebar(false);
|
||||||
await API.get('/api/user/logout');
|
await API.get('/api/user/logout');
|
||||||
showSuccess('注销成功!');
|
showSuccess('注销成功!');
|
||||||
userDispatch({type: 'logout'});
|
userDispatch({ type: 'logout' });
|
||||||
localStorage.removeItem('user');
|
localStorage.removeItem('user');
|
||||||
navigate('/login');
|
navigate('/login');
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleNewYearClick = () => {
|
||||||
|
fireworks.init('root', {});
|
||||||
|
fireworks.start();
|
||||||
|
setTimeout(() => {
|
||||||
|
fireworks.stop();
|
||||||
|
setTimeout(() => {
|
||||||
|
window.location.reload();
|
||||||
|
}, 10000);
|
||||||
|
}, 3000);
|
||||||
|
};
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (themeMode === 'dark') {
|
||||||
|
switchMode(true);
|
||||||
}
|
}
|
||||||
|
if (isNewYear) {
|
||||||
|
console.log('Happy New Year!');
|
||||||
|
}
|
||||||
|
}, []);
|
||||||
|
|
||||||
const handleNewYearClick = () => {
|
const switchMode = (model) => {
|
||||||
fireworks.init("root",{});
|
const body = document.body;
|
||||||
fireworks.start();
|
if (!model) {
|
||||||
setTimeout(() => {
|
body.removeAttribute('theme-mode');
|
||||||
fireworks.stop();
|
localStorage.setItem('theme-mode', 'light');
|
||||||
setTimeout(() => {
|
} else {
|
||||||
window.location.reload();
|
body.setAttribute('theme-mode', 'dark');
|
||||||
}, 10000);
|
localStorage.setItem('theme-mode', 'dark');
|
||||||
}, 3000);
|
}
|
||||||
};
|
setDark(model);
|
||||||
|
};
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Layout>
|
||||||
|
<div style={{ width: '100%' }}>
|
||||||
|
<Nav
|
||||||
|
mode={'horizontal'}
|
||||||
|
// bodyStyle={{ height: 100 }}
|
||||||
|
renderWrapper={({ itemElement, isSubNav, isInSubNav, props }) => {
|
||||||
|
const routerMap = {
|
||||||
|
about: '/about',
|
||||||
|
login: '/login',
|
||||||
|
register: '/register'
|
||||||
|
};
|
||||||
|
return (
|
||||||
|
<Link
|
||||||
|
style={{ textDecoration: 'none' }}
|
||||||
|
to={routerMap[props.itemKey]}
|
||||||
|
>
|
||||||
|
{itemElement}
|
||||||
|
</Link>
|
||||||
|
);
|
||||||
|
}}
|
||||||
|
selectedKeys={[]}
|
||||||
|
// items={headerButtons}
|
||||||
|
onSelect={key => {
|
||||||
|
|
||||||
useEffect(() => {
|
}}
|
||||||
if (themeMode === 'dark') {
|
footer={
|
||||||
switchMode(true);
|
<>
|
||||||
}
|
{isNewYear &&
|
||||||
if (isNewYear) {
|
// happy new year
|
||||||
console.log('Happy New Year!');
|
<Dropdown
|
||||||
}
|
position="bottomRight"
|
||||||
}, []);
|
render={
|
||||||
|
<Dropdown.Menu>
|
||||||
const switchMode = (model) => {
|
<Dropdown.Item onClick={handleNewYearClick}>Happy New Year!!!</Dropdown.Item>
|
||||||
const body = document.body;
|
</Dropdown.Menu>
|
||||||
if (!model) {
|
}
|
||||||
body.removeAttribute('theme-mode');
|
>
|
||||||
localStorage.setItem('theme-mode', 'light');
|
<Nav.Item itemKey={'new-year'} text={'🏮'} />
|
||||||
} else {
|
</Dropdown>
|
||||||
body.setAttribute('theme-mode', 'dark');
|
}
|
||||||
localStorage.setItem('theme-mode', 'dark');
|
<Nav.Item itemKey={'about'} icon={<IconHelpCircle />} />
|
||||||
}
|
<Switch checkedText="🌞" size={'large'} checked={dark} uncheckedText="🌙" onChange={switchMode} />
|
||||||
setDark(model);
|
{userState.user ?
|
||||||
};
|
<>
|
||||||
return (
|
<Dropdown
|
||||||
<>
|
position="bottomRight"
|
||||||
<Layout>
|
render={
|
||||||
<div style={{width: '100%'}}>
|
<Dropdown.Menu>
|
||||||
<Nav
|
<Dropdown.Item onClick={logout}>退出</Dropdown.Item>
|
||||||
mode={'horizontal'}
|
</Dropdown.Menu>
|
||||||
// bodyStyle={{ height: 100 }}
|
}
|
||||||
renderWrapper={({itemElement, isSubNav, isInSubNav, props}) => {
|
|
||||||
const routerMap = {
|
|
||||||
about: "/about",
|
|
||||||
login: "/login",
|
|
||||||
register: "/register",
|
|
||||||
};
|
|
||||||
return (
|
|
||||||
<Link
|
|
||||||
style={{textDecoration: "none"}}
|
|
||||||
to={routerMap[props.itemKey]}
|
|
||||||
>
|
|
||||||
{itemElement}
|
|
||||||
</Link>
|
|
||||||
);
|
|
||||||
}}
|
|
||||||
selectedKeys={[]}
|
|
||||||
// items={headerButtons}
|
|
||||||
onSelect={key => {
|
|
||||||
|
|
||||||
}}
|
|
||||||
footer={
|
|
||||||
<>
|
|
||||||
{isNewYear &&
|
|
||||||
// happy new year
|
|
||||||
<Dropdown
|
|
||||||
position="bottomRight"
|
|
||||||
render={
|
|
||||||
<Dropdown.Menu>
|
|
||||||
<Dropdown.Item onClick={handleNewYearClick}>Happy New Year!!!</Dropdown.Item>
|
|
||||||
</Dropdown.Menu>
|
|
||||||
}
|
|
||||||
>
|
|
||||||
<Nav.Item itemKey={'new-year'} text={'🏮'}/>
|
|
||||||
</Dropdown>
|
|
||||||
}
|
|
||||||
<Nav.Item itemKey={'about'} icon={<IconHelpCircle />} />
|
|
||||||
<Switch checkedText="🌞" size={'large'} checked={dark} uncheckedText="🌙" onChange={switchMode} />
|
|
||||||
{userState.user ?
|
|
||||||
<>
|
|
||||||
<Dropdown
|
|
||||||
position="bottomRight"
|
|
||||||
render={
|
|
||||||
<Dropdown.Menu>
|
|
||||||
<Dropdown.Item onClick={logout}>退出</Dropdown.Item>
|
|
||||||
</Dropdown.Menu>
|
|
||||||
}
|
|
||||||
>
|
|
||||||
<Avatar size="small" color={stringToColor(userState.user.username)} style={{ margin: 4 }}>
|
|
||||||
{userState.user.username[0]}
|
|
||||||
</Avatar>
|
|
||||||
<span>{userState.user.username}</span>
|
|
||||||
</Dropdown>
|
|
||||||
</>
|
|
||||||
:
|
|
||||||
<>
|
|
||||||
<Nav.Item itemKey={'login'} text={'登录'} icon={<IconKey />} />
|
|
||||||
<Nav.Item itemKey={'register'} text={'注册'} icon={<IconUser />} />
|
|
||||||
</>
|
|
||||||
}
|
|
||||||
</>
|
|
||||||
}
|
|
||||||
>
|
>
|
||||||
</Nav>
|
<Avatar size="small" color={stringToColor(userState.user.username)} style={{ margin: 4 }}>
|
||||||
</div>
|
{userState.user.username[0]}
|
||||||
</Layout>
|
</Avatar>
|
||||||
</>
|
<span>{userState.user.username}</span>
|
||||||
);
|
</Dropdown>
|
||||||
|
</>
|
||||||
|
:
|
||||||
|
<>
|
||||||
|
<Nav.Item itemKey={'login'} text={'登录'} icon={<IconKey />} />
|
||||||
|
<Nav.Item itemKey={'register'} text={'注册'} icon={<IconUser />} />
|
||||||
|
</>
|
||||||
|
}
|
||||||
|
</>
|
||||||
|
}
|
||||||
|
>
|
||||||
|
</Nav>
|
||||||
|
</div>
|
||||||
|
</Layout>
|
||||||
|
</>
|
||||||
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export default HeaderBar;
|
export default HeaderBar;
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { Segment, Dimmer, Loader } from 'semantic-ui-react';
|
import { Dimmer, Loader, Segment } from 'semantic-ui-react';
|
||||||
|
|
||||||
const Loading = ({ prompt: name = 'page' }) => {
|
const Loading = ({ prompt: name = 'page' }) => {
|
||||||
return (
|
return (
|
||||||
|
@ -1,12 +1,12 @@
|
|||||||
import React, { useContext, useEffect, useState } from 'react';
|
import React, { useContext, useEffect, useState } from 'react';
|
||||||
import { Link, useNavigate, useSearchParams } from 'react-router-dom';
|
import { Link, useNavigate, useSearchParams } from 'react-router-dom';
|
||||||
import { UserContext } from '../context/User';
|
import { UserContext } from '../context/User';
|
||||||
import { API, getLogo, isMobile, showError, showInfo, showSuccess, showWarning } from '../helpers';
|
import { API, getLogo, showError, showInfo, showSuccess } from '../helpers';
|
||||||
import { onGitHubOAuthClicked, onLinuxDoOAuthClicked } from './utils';
|
import { onGitHubOAuthClicked, onLinuxDoOAuthClicked } from './utils';
|
||||||
import Turnstile from "react-turnstile";
|
import Turnstile from 'react-turnstile';
|
||||||
import { Layout, Card, Image, Form, Button, Divider, Modal, Icon } from '@douyinfe/semi-ui';
|
import { Button, Card, Divider, Form, Icon, Layout, Modal } from '@douyinfe/semi-ui';
|
||||||
import Title from "@douyinfe/semi-ui/lib/es/typography/title";
|
import Title from '@douyinfe/semi-ui/lib/es/typography/title';
|
||||||
import Text from "@douyinfe/semi-ui/lib/es/typography/text";
|
import Text from '@douyinfe/semi-ui/lib/es/typography/text';
|
||||||
import TelegramLoginButton from 'react-telegram-login';
|
import TelegramLoginButton from 'react-telegram-login';
|
||||||
|
|
||||||
import { IconGithubLogo } from '@douyinfe/semi-icons';
|
import { IconGithubLogo } from '@douyinfe/semi-icons';
|
||||||
@ -14,252 +14,252 @@ import LinuxDoIcon from './LinuxDoIcon';
|
|||||||
import WeChatIcon from './WeChatIcon';
|
import WeChatIcon from './WeChatIcon';
|
||||||
|
|
||||||
const LoginForm = () => {
|
const LoginForm = () => {
|
||||||
const [inputs, setInputs] = useState({
|
const [inputs, setInputs] = useState({
|
||||||
username: '',
|
username: '',
|
||||||
password: '',
|
password: '',
|
||||||
wechat_verification_code: ''
|
wechat_verification_code: ''
|
||||||
});
|
});
|
||||||
const [searchParams, setSearchParams] = useSearchParams();
|
const [searchParams, setSearchParams] = useSearchParams();
|
||||||
const [submitted, setSubmitted] = useState(false);
|
const [submitted, setSubmitted] = useState(false);
|
||||||
const { username, password } = inputs;
|
const { username, password } = inputs;
|
||||||
const [userState, userDispatch] = useContext(UserContext);
|
const [userState, userDispatch] = useContext(UserContext);
|
||||||
const [turnstileEnabled, setTurnstileEnabled] = useState(false);
|
const [turnstileEnabled, setTurnstileEnabled] = useState(false);
|
||||||
const [turnstileSiteKey, setTurnstileSiteKey] = useState('');
|
const [turnstileSiteKey, setTurnstileSiteKey] = useState('');
|
||||||
const [turnstileToken, setTurnstileToken] = useState('');
|
const [turnstileToken, setTurnstileToken] = useState('');
|
||||||
let navigate = useNavigate();
|
let navigate = useNavigate();
|
||||||
const [status, setStatus] = useState({});
|
const [status, setStatus] = useState({});
|
||||||
const logo = getLogo();
|
const logo = getLogo();
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (searchParams.get('expired')) {
|
if (searchParams.get('expired')) {
|
||||||
showError('未登录或登录已过期,请重新登录!');
|
showError('未登录或登录已过期,请重新登录!');
|
||||||
}
|
|
||||||
let status = localStorage.getItem('status');
|
|
||||||
if (status) {
|
|
||||||
status = JSON.parse(status);
|
|
||||||
setStatus(status);
|
|
||||||
if (status.turnstile_check) {
|
|
||||||
setTurnstileEnabled(true);
|
|
||||||
setTurnstileSiteKey(status.turnstile_site_key);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
const [showWeChatLoginModal, setShowWeChatLoginModal] = useState(false);
|
|
||||||
|
|
||||||
const onWeChatLoginClicked = () => {
|
|
||||||
setShowWeChatLoginModal(true);
|
|
||||||
};
|
|
||||||
|
|
||||||
const onSubmitWeChatVerificationCode = async () => {
|
|
||||||
if (turnstileEnabled && turnstileToken === '') {
|
|
||||||
showInfo('请稍后几秒重试,Turnstile 正在检查用户环境!');
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const res = await API.get(
|
|
||||||
`/api/oauth/wechat?code=${inputs.wechat_verification_code}`
|
|
||||||
);
|
|
||||||
const { success, message, data } = res.data;
|
|
||||||
if (success) {
|
|
||||||
userDispatch({ type: 'login', payload: data });
|
|
||||||
localStorage.setItem('user', JSON.stringify(data));
|
|
||||||
navigate('/');
|
|
||||||
showSuccess('登录成功!');
|
|
||||||
setShowWeChatLoginModal(false);
|
|
||||||
} else {
|
|
||||||
showError(message);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
function handleChange(name, value) {
|
|
||||||
setInputs((inputs) => ({ ...inputs, [name]: value }));
|
|
||||||
}
|
}
|
||||||
|
let status = localStorage.getItem('status');
|
||||||
async function handleSubmit(e) {
|
if (status) {
|
||||||
if (turnstileEnabled && turnstileToken === '') {
|
status = JSON.parse(status);
|
||||||
showInfo('请稍后几秒重试,Turnstile 正在检查用户环境!');
|
setStatus(status);
|
||||||
return;
|
if (status.turnstile_check) {
|
||||||
}
|
setTurnstileEnabled(true);
|
||||||
setSubmitted(true);
|
setTurnstileSiteKey(status.turnstile_site_key);
|
||||||
if (username && password) {
|
}
|
||||||
const res = await API.post(`/api/user/login?turnstile=${turnstileToken}`, {
|
|
||||||
username,
|
|
||||||
password
|
|
||||||
});
|
|
||||||
const { success, message, data } = res.data;
|
|
||||||
if (success) {
|
|
||||||
userDispatch({ type: 'login', payload: data });
|
|
||||||
localStorage.setItem('user', JSON.stringify(data));
|
|
||||||
showSuccess('登录成功!');
|
|
||||||
if (username === 'root' && password === '123456') {
|
|
||||||
Modal.error({ title: '您正在使用默认密码!', content: '请立刻修改默认密码!', centered: true });
|
|
||||||
}
|
|
||||||
navigate('/token');
|
|
||||||
} else {
|
|
||||||
showError(message);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
showError('请输入用户名和密码!');
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
}, []);
|
||||||
|
|
||||||
// 添加Telegram登录处理函数
|
const [showWeChatLoginModal, setShowWeChatLoginModal] = useState(false);
|
||||||
const onTelegramLoginClicked = async (response) => {
|
|
||||||
const fields = ["id", "first_name", "last_name", "username", "photo_url", "auth_date", "hash", "lang"];
|
|
||||||
const params = {};
|
|
||||||
fields.forEach((field) => {
|
|
||||||
if (response[field]) {
|
|
||||||
params[field] = response[field];
|
|
||||||
}
|
|
||||||
});
|
|
||||||
const res = await API.get(`/api/oauth/telegram/login`, { params });
|
|
||||||
const { success, message, data } = res.data;
|
|
||||||
if (success) {
|
|
||||||
userDispatch({ type: 'login', payload: data });
|
|
||||||
localStorage.setItem('user', JSON.stringify(data));
|
|
||||||
showSuccess('登录成功!');
|
|
||||||
navigate('/');
|
|
||||||
} else {
|
|
||||||
showError(message);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
const onWeChatLoginClicked = () => {
|
||||||
<div>
|
setShowWeChatLoginModal(true);
|
||||||
<Layout>
|
};
|
||||||
<Layout.Header>
|
|
||||||
</Layout.Header>
|
|
||||||
<Layout.Content>
|
|
||||||
<div style={{ justifyContent: 'center', display: "flex", marginTop: 120 }}>
|
|
||||||
<div style={{ width: 500 }}>
|
|
||||||
<Card>
|
|
||||||
<Title heading={2} style={{ textAlign: 'center' }}>
|
|
||||||
用户登录
|
|
||||||
</Title>
|
|
||||||
<Form>
|
|
||||||
<Form.Input
|
|
||||||
field={'username'}
|
|
||||||
label={'用户名'}
|
|
||||||
placeholder='用户名'
|
|
||||||
name='username'
|
|
||||||
onChange={(value) => handleChange('username', value)}
|
|
||||||
/>
|
|
||||||
<Form.Input
|
|
||||||
field={'password'}
|
|
||||||
label={'密码'}
|
|
||||||
placeholder='密码'
|
|
||||||
name='password'
|
|
||||||
type='password'
|
|
||||||
onChange={(value) => handleChange('password', value)}
|
|
||||||
/>
|
|
||||||
|
|
||||||
<Button theme='solid' style={{ width: '100%' }} type={'primary'} size='large'
|
const onSubmitWeChatVerificationCode = async () => {
|
||||||
htmlType={'submit'} onClick={handleSubmit}>
|
if (turnstileEnabled && turnstileToken === '') {
|
||||||
登录
|
showInfo('请稍后几秒重试,Turnstile 正在检查用户环境!');
|
||||||
</Button>
|
return;
|
||||||
</Form>
|
}
|
||||||
<div style={{ display: 'flex', justifyContent: 'space-between', marginTop: 20 }}>
|
const res = await API.get(
|
||||||
<Text>
|
`/api/oauth/wechat?code=${inputs.wechat_verification_code}`
|
||||||
没有账号请先 <Link to='/register'>注册账号</Link>
|
|
||||||
</Text>
|
|
||||||
<Text>
|
|
||||||
忘记密码 <Link to='/reset'>点击重置</Link>
|
|
||||||
</Text>
|
|
||||||
</div>
|
|
||||||
{status.github_oauth || status.linuxdo_oauth || status.wechat_login || status.telegram_oauth ? (
|
|
||||||
<>
|
|
||||||
<Divider margin='12px' align='center'>
|
|
||||||
第三方登录
|
|
||||||
</Divider>
|
|
||||||
<div style={{ display: 'flex', justifyContent: 'center', marginTop: 20 }}>
|
|
||||||
{status.github_oauth ? (
|
|
||||||
<Button
|
|
||||||
type='primary'
|
|
||||||
icon={<IconGithubLogo />}
|
|
||||||
onClick={() => onGitHubOAuthClicked(status.github_client_id)}
|
|
||||||
/>
|
|
||||||
) : (
|
|
||||||
<></>
|
|
||||||
)}
|
|
||||||
{status.linuxdo_oauth ? (
|
|
||||||
<Button
|
|
||||||
type='primary'
|
|
||||||
icon={<LinuxDoIcon />}
|
|
||||||
style={{color: '#000'}}
|
|
||||||
onClick={() => onLinuxDoOAuthClicked(status.linuxdo_client_id)}
|
|
||||||
/>
|
|
||||||
) : (
|
|
||||||
<></>
|
|
||||||
)}
|
|
||||||
{status.wechat_login ? (
|
|
||||||
<Button
|
|
||||||
type='primary'
|
|
||||||
style={{color: 'rgba(var(--semi-green-5), 1)'}}
|
|
||||||
icon={<Icon svg={<WeChatIcon />} />}
|
|
||||||
onClick={onWeChatLoginClicked}
|
|
||||||
/>
|
|
||||||
) : (
|
|
||||||
<></>
|
|
||||||
)}
|
|
||||||
|
|
||||||
{status.telegram_oauth ? (
|
|
||||||
<TelegramLoginButton dataOnauth={onTelegramLoginClicked} botName={status.telegram_bot_name} />
|
|
||||||
) : (
|
|
||||||
<></>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
</>
|
|
||||||
) : (
|
|
||||||
<></>
|
|
||||||
)}
|
|
||||||
<Modal
|
|
||||||
title="微信扫码登录"
|
|
||||||
visible={showWeChatLoginModal}
|
|
||||||
maskClosable={true}
|
|
||||||
onOk={onSubmitWeChatVerificationCode}
|
|
||||||
onCancel={() => setShowWeChatLoginModal(false)}
|
|
||||||
okText={'登录'}
|
|
||||||
size={'small'}
|
|
||||||
centered={true}
|
|
||||||
>
|
|
||||||
<div style={{ display: 'flex', alignItem: 'center', flexDirection: 'column' }}>
|
|
||||||
<img src={status.wechat_qrcode}/>
|
|
||||||
</div>
|
|
||||||
<div style={{textAlign: 'center'}}>
|
|
||||||
<p>
|
|
||||||
微信扫码关注公众号,输入「验证码」获取验证码(三分钟内有效)
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
<Form size='large'>
|
|
||||||
<Form.Input
|
|
||||||
field={'wechat_verification_code'}
|
|
||||||
placeholder='验证码'
|
|
||||||
label={'验证码'}
|
|
||||||
value={inputs.wechat_verification_code}
|
|
||||||
onChange={(value) => handleChange('wechat_verification_code', value)}
|
|
||||||
/>
|
|
||||||
</Form>
|
|
||||||
</Modal>
|
|
||||||
</Card>
|
|
||||||
{turnstileEnabled ? (
|
|
||||||
<div style={{ display: 'flex', justifyContent: 'center', marginTop: 20 }}>
|
|
||||||
<Turnstile
|
|
||||||
sitekey={turnstileSiteKey}
|
|
||||||
onVerify={(token) => {
|
|
||||||
setTurnstileToken(token);
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
) : (
|
|
||||||
<></>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
</Layout.Content>
|
|
||||||
</Layout>
|
|
||||||
</div>
|
|
||||||
);
|
);
|
||||||
|
const { success, message, data } = res.data;
|
||||||
|
if (success) {
|
||||||
|
userDispatch({ type: 'login', payload: data });
|
||||||
|
localStorage.setItem('user', JSON.stringify(data));
|
||||||
|
navigate('/');
|
||||||
|
showSuccess('登录成功!');
|
||||||
|
setShowWeChatLoginModal(false);
|
||||||
|
} else {
|
||||||
|
showError(message);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
function handleChange(name, value) {
|
||||||
|
setInputs((inputs) => ({ ...inputs, [name]: value }));
|
||||||
|
}
|
||||||
|
|
||||||
|
async function handleSubmit(e) {
|
||||||
|
if (turnstileEnabled && turnstileToken === '') {
|
||||||
|
showInfo('请稍后几秒重试,Turnstile 正在检查用户环境!');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
setSubmitted(true);
|
||||||
|
if (username && password) {
|
||||||
|
const res = await API.post(`/api/user/login?turnstile=${turnstileToken}`, {
|
||||||
|
username,
|
||||||
|
password
|
||||||
|
});
|
||||||
|
const { success, message, data } = res.data;
|
||||||
|
if (success) {
|
||||||
|
userDispatch({ type: 'login', payload: data });
|
||||||
|
localStorage.setItem('user', JSON.stringify(data));
|
||||||
|
showSuccess('登录成功!');
|
||||||
|
if (username === 'root' && password === '123456') {
|
||||||
|
Modal.error({ title: '您正在使用默认密码!', content: '请立刻修改默认密码!', centered: true });
|
||||||
|
}
|
||||||
|
navigate('/token');
|
||||||
|
} else {
|
||||||
|
showError(message);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
showError('请输入用户名和密码!');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 添加Telegram登录处理函数
|
||||||
|
const onTelegramLoginClicked = async (response) => {
|
||||||
|
const fields = ['id', 'first_name', 'last_name', 'username', 'photo_url', 'auth_date', 'hash', 'lang'];
|
||||||
|
const params = {};
|
||||||
|
fields.forEach((field) => {
|
||||||
|
if (response[field]) {
|
||||||
|
params[field] = response[field];
|
||||||
|
}
|
||||||
|
});
|
||||||
|
const res = await API.get(`/api/oauth/telegram/login`, { params });
|
||||||
|
const { success, message, data } = res.data;
|
||||||
|
if (success) {
|
||||||
|
userDispatch({ type: 'login', payload: data });
|
||||||
|
localStorage.setItem('user', JSON.stringify(data));
|
||||||
|
showSuccess('登录成功!');
|
||||||
|
navigate('/');
|
||||||
|
} else {
|
||||||
|
showError(message);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div>
|
||||||
|
<Layout>
|
||||||
|
<Layout.Header>
|
||||||
|
</Layout.Header>
|
||||||
|
<Layout.Content>
|
||||||
|
<div style={{ justifyContent: 'center', display: 'flex', marginTop: 120 }}>
|
||||||
|
<div style={{ width: 500 }}>
|
||||||
|
<Card>
|
||||||
|
<Title heading={2} style={{ textAlign: 'center' }}>
|
||||||
|
用户登录
|
||||||
|
</Title>
|
||||||
|
<Form>
|
||||||
|
<Form.Input
|
||||||
|
field={'username'}
|
||||||
|
label={'用户名'}
|
||||||
|
placeholder="用户名"
|
||||||
|
name="username"
|
||||||
|
onChange={(value) => handleChange('username', value)}
|
||||||
|
/>
|
||||||
|
<Form.Input
|
||||||
|
field={'password'}
|
||||||
|
label={'密码'}
|
||||||
|
placeholder="密码"
|
||||||
|
name="password"
|
||||||
|
type="password"
|
||||||
|
onChange={(value) => handleChange('password', value)}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<Button theme="solid" style={{ width: '100%' }} type={'primary'} size="large"
|
||||||
|
htmlType={'submit'} onClick={handleSubmit}>
|
||||||
|
登录
|
||||||
|
</Button>
|
||||||
|
</Form>
|
||||||
|
<div style={{ display: 'flex', justifyContent: 'space-between', marginTop: 20 }}>
|
||||||
|
<Text>
|
||||||
|
没有账号请先 <Link to="/register">注册账号</Link>
|
||||||
|
</Text>
|
||||||
|
<Text>
|
||||||
|
忘记密码 <Link to="/reset">点击重置</Link>
|
||||||
|
</Text>
|
||||||
|
</div>
|
||||||
|
{status.github_oauth || status.linuxdo_oauth || status.wechat_login || status.telegram_oauth ? (
|
||||||
|
<>
|
||||||
|
<Divider margin="12px" align="center">
|
||||||
|
第三方登录
|
||||||
|
</Divider>
|
||||||
|
<div style={{ display: 'flex', justifyContent: 'center', marginTop: 20 }}>
|
||||||
|
{status.github_oauth ? (
|
||||||
|
<Button
|
||||||
|
type="primary"
|
||||||
|
icon={<IconGithubLogo />}
|
||||||
|
onClick={() => onGitHubOAuthClicked(status.github_client_id)}
|
||||||
|
/>
|
||||||
|
) : (
|
||||||
|
<></>
|
||||||
|
)}
|
||||||
|
{status.linuxdo_oauth ? (
|
||||||
|
<Button
|
||||||
|
type="primary"
|
||||||
|
icon={<LinuxDoIcon />}
|
||||||
|
style={{color: '#000'}}
|
||||||
|
onClick={() => onLinuxDoOAuthClicked(status.linuxdo_client_id)}
|
||||||
|
/>
|
||||||
|
) : (
|
||||||
|
<></>
|
||||||
|
)}
|
||||||
|
{status.wechat_login ? (
|
||||||
|
<Button
|
||||||
|
type="primary"
|
||||||
|
style={{ color: 'rgba(var(--semi-green-5), 1)' }}
|
||||||
|
icon={<Icon svg={<WeChatIcon />} />}
|
||||||
|
onClick={onWeChatLoginClicked}
|
||||||
|
/>
|
||||||
|
) : (
|
||||||
|
<></>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{status.telegram_oauth ? (
|
||||||
|
<TelegramLoginButton dataOnauth={onTelegramLoginClicked} botName={status.telegram_bot_name} />
|
||||||
|
) : (
|
||||||
|
<></>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</>
|
||||||
|
) : (
|
||||||
|
<></>
|
||||||
|
)}
|
||||||
|
<Modal
|
||||||
|
title="微信扫码登录"
|
||||||
|
visible={showWeChatLoginModal}
|
||||||
|
maskClosable={true}
|
||||||
|
onOk={onSubmitWeChatVerificationCode}
|
||||||
|
onCancel={() => setShowWeChatLoginModal(false)}
|
||||||
|
okText={'登录'}
|
||||||
|
size={'small'}
|
||||||
|
centered={true}
|
||||||
|
>
|
||||||
|
<div style={{ display: 'flex', alignItem: 'center', flexDirection: 'column' }}>
|
||||||
|
<img src={status.wechat_qrcode} />
|
||||||
|
</div>
|
||||||
|
<div style={{ textAlign: 'center' }}>
|
||||||
|
<p>
|
||||||
|
微信扫码关注公众号,输入「验证码」获取验证码(三分钟内有效)
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
<Form size="large">
|
||||||
|
<Form.Input
|
||||||
|
field={'wechat_verification_code'}
|
||||||
|
placeholder="验证码"
|
||||||
|
label={'验证码'}
|
||||||
|
value={inputs.wechat_verification_code}
|
||||||
|
onChange={(value) => handleChange('wechat_verification_code', value)}
|
||||||
|
/>
|
||||||
|
</Form>
|
||||||
|
</Modal>
|
||||||
|
</Card>
|
||||||
|
{turnstileEnabled ? (
|
||||||
|
<div style={{ display: 'flex', justifyContent: 'center', marginTop: 20 }}>
|
||||||
|
<Turnstile
|
||||||
|
sitekey={turnstileSiteKey}
|
||||||
|
onVerify={(token) => {
|
||||||
|
setTurnstileToken(token);
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
) : (
|
||||||
|
<></>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
</Layout.Content>
|
||||||
|
</Layout>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export default LoginForm;
|
export default LoginForm;
|
||||||
|
@ -1,501 +1,399 @@
|
|||||||
import React, {useEffect, useState} from 'react';
|
import React, { useEffect, useState } from 'react';
|
||||||
import {Label} from 'semantic-ui-react';
|
import { API, copy, isAdmin, showError, showSuccess, timestamp2string } from '../helpers';
|
||||||
import {API, copy, isAdmin, showError, showSuccess, timestamp2string} from '../helpers';
|
|
||||||
|
|
||||||
import {Table, Avatar, Tag, Form, Button, Layout, Select, Popover, Modal, Spin, Space} from '@douyinfe/semi-ui';
|
import { Avatar, Button, Form, Layout, Modal, Select, Space, Spin, Table, Tag } from '@douyinfe/semi-ui';
|
||||||
import {ITEMS_PER_PAGE} from '../constants';
|
import { ITEMS_PER_PAGE } from '../constants';
|
||||||
import {renderNumber, renderQuota, stringToColor} from '../helpers/render';
|
import { renderNumber, renderQuota, stringToColor } from '../helpers/render';
|
||||||
import {
|
import Paragraph from '@douyinfe/semi-ui/lib/es/typography/paragraph';
|
||||||
IconAt,
|
|
||||||
IconHistogram,
|
const { Header } = Layout;
|
||||||
IconGift,
|
|
||||||
IconKey,
|
|
||||||
IconUser,
|
|
||||||
IconLayers,
|
|
||||||
IconSetting,
|
|
||||||
IconCreditCard,
|
|
||||||
IconSemiLogo,
|
|
||||||
IconHome,
|
|
||||||
IconMore
|
|
||||||
} from '@douyinfe/semi-icons';
|
|
||||||
import Paragraph from "@douyinfe/semi-ui/lib/es/typography/paragraph";
|
|
||||||
|
|
||||||
const {Header} = Layout;
|
|
||||||
function renderTimestamp(timestamp) {
|
function renderTimestamp(timestamp) {
|
||||||
return (
|
return (<>
|
||||||
<>
|
{timestamp2string(timestamp)}
|
||||||
{timestamp2string(timestamp)}
|
</>);
|
||||||
</>
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const MODE_OPTIONS = [
|
const MODE_OPTIONS = [{ key: 'all', text: '全部用户', value: 'all' }, { key: 'self', text: '当前用户', value: 'self' }];
|
||||||
{key: 'all', text: '全部用户', value: 'all'},
|
|
||||||
{key: 'self', text: '当前用户', value: 'self'}
|
|
||||||
];
|
|
||||||
|
|
||||||
const colors = ['amber', 'blue', 'cyan', 'green', 'grey', 'indigo',
|
const colors = ['amber', 'blue', 'cyan', 'green', 'grey', 'indigo', 'light-blue', 'lime', 'orange', 'pink', 'purple', 'red', 'teal', 'violet', 'yellow'];
|
||||||
'light-blue', 'lime', 'orange', 'pink',
|
|
||||||
'purple', 'red', 'teal', 'violet', 'yellow'
|
|
||||||
]
|
|
||||||
|
|
||||||
function renderType(type) {
|
function renderType(type) {
|
||||||
switch (type) {
|
switch (type) {
|
||||||
case 1:
|
case 1:
|
||||||
return <Tag color='cyan' size='large'> 充值 </Tag>;
|
return <Tag color="cyan" size="large"> 充值 </Tag>;
|
||||||
case 2:
|
case 2:
|
||||||
return <Tag color='lime' size='large'> 消费 </Tag>;
|
return <Tag color="lime" size="large"> 消费 </Tag>;
|
||||||
case 3:
|
case 3:
|
||||||
return <Tag color='orange' size='large'> 管理 </Tag>;
|
return <Tag color="orange" size="large"> 管理 </Tag>;
|
||||||
case 4:
|
case 4:
|
||||||
return <Tag color='purple' size='large'> 系统 </Tag>;
|
return <Tag color="purple" size="large"> 系统 </Tag>;
|
||||||
default:
|
default:
|
||||||
return <Tag color='black' size='large'> 未知 </Tag>;
|
return <Tag color="black" size="large"> 未知 </Tag>;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function renderIsStream(bool) {
|
function renderIsStream(bool) {
|
||||||
if (bool) {
|
if (bool) {
|
||||||
return <Tag color='blue' size='large'>流</Tag>;
|
return <Tag color="blue" size="large">流</Tag>;
|
||||||
} else {
|
} else {
|
||||||
return <Tag color='purple' size='large'>非流</Tag>;
|
return <Tag color="purple" size="large">非流</Tag>;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function renderUseTime(type) {
|
function renderUseTime(type) {
|
||||||
const time = parseInt(type);
|
const time = parseInt(type);
|
||||||
if (time < 101) {
|
if (time < 101) {
|
||||||
return <Tag color='green' size='large'> {time} s </Tag>;
|
return <Tag color="green" size="large"> {time} s </Tag>;
|
||||||
} else if (time < 300) {
|
} else if (time < 300) {
|
||||||
return <Tag color='orange' size='large'> {time} s </Tag>;
|
return <Tag color="orange" size="large"> {time} s </Tag>;
|
||||||
} else {
|
} else {
|
||||||
return <Tag color='red' size='large'> {time} s </Tag>;
|
return <Tag color="red" size="large"> {time} s </Tag>;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const LogsTable = () => {
|
const LogsTable = () => {
|
||||||
const columns = [
|
const columns = [{
|
||||||
{
|
title: '时间', dataIndex: 'timestamp2string'
|
||||||
title: '时间',
|
}, {
|
||||||
dataIndex: 'timestamp2string',
|
title: '渠道',
|
||||||
},
|
dataIndex: 'channel',
|
||||||
{
|
className: isAdmin() ? 'tableShow' : 'tableHiddle',
|
||||||
title: '渠道',
|
render: (text, record, index) => {
|
||||||
dataIndex: 'channel',
|
return (isAdminUser ? record.type === 0 || record.type === 2 ? <div>
|
||||||
className: isAdmin() ? 'tableShow' : 'tableHiddle',
|
{<Tag color={colors[parseInt(text) % colors.length]} size="large"> {text} </Tag>}
|
||||||
render: (text, record, index) => {
|
</div> : <></> : <></>);
|
||||||
return (
|
|
||||||
isAdminUser ?
|
|
||||||
record.type === 0 || record.type === 2 ?
|
|
||||||
<div>
|
|
||||||
{<Tag color={colors[parseInt(text) % colors.length]} size='large'> {text} </Tag>}
|
|
||||||
</div>
|
|
||||||
:
|
|
||||||
<></>
|
|
||||||
:
|
|
||||||
<></>
|
|
||||||
);
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
title: '用户',
|
|
||||||
dataIndex: 'username',
|
|
||||||
className: isAdmin() ? 'tableShow' : 'tableHiddle',
|
|
||||||
render: (text, record, index) => {
|
|
||||||
return (
|
|
||||||
isAdminUser ?
|
|
||||||
<div>
|
|
||||||
<Avatar size="small" color={stringToColor(text)} style={{marginRight: 4}}
|
|
||||||
onClick={() => showUserInfo(record.user_id)}>
|
|
||||||
{typeof text === 'string' && text.slice(0, 1)}
|
|
||||||
</Avatar>
|
|
||||||
{text}
|
|
||||||
</div>
|
|
||||||
:
|
|
||||||
<></>
|
|
||||||
);
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
title: '令牌',
|
|
||||||
dataIndex: 'token_name',
|
|
||||||
render: (text, record, index) => {
|
|
||||||
return (
|
|
||||||
record.type === 0 || record.type === 2 ?
|
|
||||||
<div>
|
|
||||||
<Tag color='grey' size='large' onClick={() => {
|
|
||||||
copyText(text)
|
|
||||||
}}> {text} </Tag>
|
|
||||||
</div>
|
|
||||||
:
|
|
||||||
<></>
|
|
||||||
);
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
title: '类型',
|
|
||||||
dataIndex: 'type',
|
|
||||||
render: (text, record, index) => {
|
|
||||||
return (
|
|
||||||
<div>
|
|
||||||
{renderType(text)}
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
title: '模型',
|
|
||||||
dataIndex: 'model_name',
|
|
||||||
render: (text, record, index) => {
|
|
||||||
return (
|
|
||||||
record.type === 0 || record.type === 2 ?
|
|
||||||
<div>
|
|
||||||
<Tag color={stringToColor(text)} size='large' onClick={() => {
|
|
||||||
copyText(text)
|
|
||||||
}}> {text} </Tag>
|
|
||||||
</div>
|
|
||||||
:
|
|
||||||
<></>
|
|
||||||
);
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
title: '用时',
|
|
||||||
dataIndex: 'use_time',
|
|
||||||
render: (text, record, index) => {
|
|
||||||
return (
|
|
||||||
<div>
|
|
||||||
<Space>
|
|
||||||
{renderUseTime(text)}
|
|
||||||
{renderIsStream(record.is_stream)}
|
|
||||||
</Space>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
title: '提示',
|
|
||||||
dataIndex: 'prompt_tokens',
|
|
||||||
render: (text, record, index) => {
|
|
||||||
return (
|
|
||||||
record.type === 0 || record.type === 2 ?
|
|
||||||
<div>
|
|
||||||
{<span> {text} </span>}
|
|
||||||
</div>
|
|
||||||
:
|
|
||||||
<></>
|
|
||||||
);
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
title: '补全',
|
|
||||||
dataIndex: 'completion_tokens',
|
|
||||||
render: (text, record, index) => {
|
|
||||||
return (
|
|
||||||
parseInt(text) > 0 && (record.type === 0 || record.type === 2) ?
|
|
||||||
<div>
|
|
||||||
{<span> {text} </span>}
|
|
||||||
</div>
|
|
||||||
:
|
|
||||||
<></>
|
|
||||||
);
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
title: '花费',
|
|
||||||
dataIndex: 'quota',
|
|
||||||
render: (text, record, index) => {
|
|
||||||
return (
|
|
||||||
record.type === 0 || record.type === 2 ?
|
|
||||||
<div>
|
|
||||||
{
|
|
||||||
renderQuota(text, 6)
|
|
||||||
}
|
|
||||||
</div>
|
|
||||||
:
|
|
||||||
<></>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
title: '详情',
|
|
||||||
dataIndex: 'content',
|
|
||||||
render: (text, record, index) => {
|
|
||||||
return <Paragraph ellipsis={{ rows: 2, showTooltip: { type: 'popover', opts: { style: { width: 240 } } } }} style={{ maxWidth: 240}}>
|
|
||||||
{text}
|
|
||||||
</Paragraph>
|
|
||||||
}
|
|
||||||
}
|
|
||||||
];
|
|
||||||
|
|
||||||
const [logs, setLogs] = useState([]);
|
|
||||||
const [showStat, setShowStat] = useState(false);
|
|
||||||
const [loading, setLoading] = useState(false);
|
|
||||||
const [loadingStat, setLoadingStat] = useState(false);
|
|
||||||
const [activePage, setActivePage] = useState(1);
|
|
||||||
const [logCount, setLogCount] = useState(ITEMS_PER_PAGE);
|
|
||||||
const [searchKeyword, setSearchKeyword] = useState('');
|
|
||||||
const [searching, setSearching] = useState(false);
|
|
||||||
const [logType, setLogType] = useState(0);
|
|
||||||
const isAdminUser = isAdmin();
|
|
||||||
let now = new Date();
|
|
||||||
// 初始化start_timestamp为前一天
|
|
||||||
const [inputs, setInputs] = useState({
|
|
||||||
username: '',
|
|
||||||
token_name: '',
|
|
||||||
model_name: '',
|
|
||||||
start_timestamp: timestamp2string(now.getTime() / 1000 - 86400),
|
|
||||||
end_timestamp: timestamp2string(now.getTime() / 1000 + 3600),
|
|
||||||
channel: ''
|
|
||||||
});
|
|
||||||
const {username, token_name, model_name, start_timestamp, end_timestamp, channel} = inputs;
|
|
||||||
|
|
||||||
const [stat, setStat] = useState({
|
|
||||||
quota: 0,
|
|
||||||
token: 0
|
|
||||||
});
|
|
||||||
|
|
||||||
const handleInputChange = (value, name) => {
|
|
||||||
setInputs((inputs) => ({...inputs, [name]: value}));
|
|
||||||
};
|
|
||||||
|
|
||||||
const getLogSelfStat = async () => {
|
|
||||||
let localStartTimestamp = Date.parse(start_timestamp) / 1000;
|
|
||||||
let localEndTimestamp = Date.parse(end_timestamp) / 1000;
|
|
||||||
let res = await API.get(`/api/log/self/stat?type=${logType}&token_name=${token_name}&model_name=${model_name}&start_timestamp=${localStartTimestamp}&end_timestamp=${localEndTimestamp}`);
|
|
||||||
const {success, message, data} = res.data;
|
|
||||||
if (success) {
|
|
||||||
setStat(data);
|
|
||||||
} else {
|
|
||||||
showError(message);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const getLogStat = async () => {
|
|
||||||
let localStartTimestamp = Date.parse(start_timestamp) / 1000;
|
|
||||||
let localEndTimestamp = Date.parse(end_timestamp) / 1000;
|
|
||||||
let res = await API.get(`/api/log/stat?type=${logType}&username=${username}&token_name=${token_name}&model_name=${model_name}&start_timestamp=${localStartTimestamp}&end_timestamp=${localEndTimestamp}&channel=${channel}`);
|
|
||||||
const {success, message, data} = res.data;
|
|
||||||
if (success) {
|
|
||||||
setStat(data);
|
|
||||||
} else {
|
|
||||||
showError(message);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleEyeClick = async () => {
|
|
||||||
setLoadingStat(true);
|
|
||||||
if (isAdminUser) {
|
|
||||||
await getLogStat();
|
|
||||||
} else {
|
|
||||||
await getLogSelfStat();
|
|
||||||
}
|
|
||||||
setShowStat(true);
|
|
||||||
setLoadingStat(false);
|
|
||||||
};
|
|
||||||
|
|
||||||
const showUserInfo = async (userId) => {
|
|
||||||
if (!isAdminUser) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const res = await API.get(`/api/user/${userId}`);
|
|
||||||
const {success, message, data} = res.data;
|
|
||||||
if (success) {
|
|
||||||
Modal.info({
|
|
||||||
title: '用户信息',
|
|
||||||
content: <div style={{padding: 12}}>
|
|
||||||
<p>用户名: {data.username}</p>
|
|
||||||
<p>余额: {renderQuota(data.quota)}</p>
|
|
||||||
<p>已用额度:{renderQuota(data.used_quota)}</p>
|
|
||||||
<p>请求次数:{renderNumber(data.request_count)}</p>
|
|
||||||
</div>,
|
|
||||||
centered: true,
|
|
||||||
})
|
|
||||||
} else {
|
|
||||||
showError(message);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const setLogsFormat = (logs) => {
|
|
||||||
for (let i = 0; i < logs.length; i++) {
|
|
||||||
logs[i].timestamp2string = timestamp2string(logs[i].created_at);
|
|
||||||
logs[i].key = '' + logs[i].id;
|
|
||||||
}
|
|
||||||
// data.key = '' + data.id
|
|
||||||
setLogs(logs);
|
|
||||||
setLogCount(logs.length + ITEMS_PER_PAGE);
|
|
||||||
// console.log(logCount);
|
|
||||||
}
|
}
|
||||||
|
}, {
|
||||||
const loadLogs = async (startIdx) => {
|
title: '用户',
|
||||||
setLoading(true);
|
dataIndex: 'username',
|
||||||
|
className: isAdmin() ? 'tableShow' : 'tableHiddle',
|
||||||
let url = '';
|
render: (text, record, index) => {
|
||||||
let localStartTimestamp = Date.parse(start_timestamp) / 1000;
|
return (isAdminUser ? <div>
|
||||||
let localEndTimestamp = Date.parse(end_timestamp) / 1000;
|
<Avatar size="small" color={stringToColor(text)} style={{ marginRight: 4 }}
|
||||||
if (isAdminUser) {
|
onClick={() => showUserInfo(record.user_id)}>
|
||||||
url = `/api/log/?p=${startIdx}&type=${logType}&username=${username}&token_name=${token_name}&model_name=${model_name}&start_timestamp=${localStartTimestamp}&end_timestamp=${localEndTimestamp}&channel=${channel}`;
|
{typeof text === 'string' && text.slice(0, 1)}
|
||||||
} else {
|
</Avatar>
|
||||||
url = `/api/log/self/?p=${startIdx}&type=${logType}&token_name=${token_name}&model_name=${model_name}&start_timestamp=${localStartTimestamp}&end_timestamp=${localEndTimestamp}`;
|
{text}
|
||||||
}
|
</div> : <></>);
|
||||||
const res = await API.get(url);
|
|
||||||
const {success, message, data} = res.data;
|
|
||||||
if (success) {
|
|
||||||
if (startIdx === 0) {
|
|
||||||
setLogsFormat(data);
|
|
||||||
} else {
|
|
||||||
let newLogs = [...logs];
|
|
||||||
newLogs.splice(startIdx * ITEMS_PER_PAGE, data.length, ...data);
|
|
||||||
setLogsFormat(newLogs);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
showError(message);
|
|
||||||
}
|
|
||||||
setLoading(false);
|
|
||||||
};
|
|
||||||
|
|
||||||
const pageData = logs.slice((activePage - 1) * ITEMS_PER_PAGE, activePage * ITEMS_PER_PAGE);
|
|
||||||
|
|
||||||
const handlePageChange = page => {
|
|
||||||
setActivePage(page);
|
|
||||||
if (page === Math.ceil(logs.length / ITEMS_PER_PAGE) + 1) {
|
|
||||||
// In this case we have to load more data and then append them.
|
|
||||||
loadLogs(page - 1).then(r => {
|
|
||||||
});
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const refresh = async () => {
|
|
||||||
// setLoading(true);
|
|
||||||
setActivePage(1);
|
|
||||||
await loadLogs(0);
|
|
||||||
};
|
|
||||||
|
|
||||||
const copyText = async (text) => {
|
|
||||||
if (await copy(text)) {
|
|
||||||
showSuccess('已复制:' + text);
|
|
||||||
} else {
|
|
||||||
// setSearchKeyword(text);
|
|
||||||
Modal.error({title: '无法复制到剪贴板,请手动复制', content: text});
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
}, {
|
||||||
|
title: '令牌', dataIndex: 'token_name', render: (text, record, index) => {
|
||||||
|
return (record.type === 0 || record.type === 2 ? <div>
|
||||||
|
<Tag color="grey" size="large" onClick={() => {
|
||||||
|
copyText(text);
|
||||||
|
}}> {text} </Tag>
|
||||||
|
</div> : <></>);
|
||||||
|
}
|
||||||
|
}, {
|
||||||
|
title: '类型', dataIndex: 'type', render: (text, record, index) => {
|
||||||
|
return (<div>
|
||||||
|
{renderType(text)}
|
||||||
|
</div>);
|
||||||
|
}
|
||||||
|
}, {
|
||||||
|
title: '模型', dataIndex: 'model_name', render: (text, record, index) => {
|
||||||
|
return (record.type === 0 || record.type === 2 ? <div>
|
||||||
|
<Tag color={stringToColor(text)} size="large" onClick={() => {
|
||||||
|
copyText(text);
|
||||||
|
}}> {text} </Tag>
|
||||||
|
</div> : <></>);
|
||||||
|
}
|
||||||
|
}, {
|
||||||
|
title: '用时', dataIndex: 'use_time', render: (text, record, index) => {
|
||||||
|
return (<div>
|
||||||
|
<Space>
|
||||||
|
{renderUseTime(text)}
|
||||||
|
{renderIsStream(record.is_stream)}
|
||||||
|
</Space>
|
||||||
|
</div>);
|
||||||
|
}
|
||||||
|
}, {
|
||||||
|
title: '提示', dataIndex: 'prompt_tokens', render: (text, record, index) => {
|
||||||
|
return (record.type === 0 || record.type === 2 ? <div>
|
||||||
|
{<span> {text} </span>}
|
||||||
|
</div> : <></>);
|
||||||
|
}
|
||||||
|
}, {
|
||||||
|
title: '补全', dataIndex: 'completion_tokens', render: (text, record, index) => {
|
||||||
|
return (parseInt(text) > 0 && (record.type === 0 || record.type === 2) ? <div>
|
||||||
|
{<span> {text} </span>}
|
||||||
|
</div> : <></>);
|
||||||
|
}
|
||||||
|
}, {
|
||||||
|
title: '花费', dataIndex: 'quota', render: (text, record, index) => {
|
||||||
|
return (record.type === 0 || record.type === 2 ? <div>
|
||||||
|
{renderQuota(text, 6)}
|
||||||
|
</div> : <></>);
|
||||||
|
}
|
||||||
|
}, {
|
||||||
|
title: '详情', dataIndex: 'content', render: (text, record, index) => {
|
||||||
|
return <Paragraph ellipsis={{ rows: 2, showTooltip: { type: 'popover', opts: { style: { width: 240 } } } }}
|
||||||
|
style={{ maxWidth: 240 }}>
|
||||||
|
{text}
|
||||||
|
</Paragraph>;
|
||||||
|
}
|
||||||
|
}];
|
||||||
|
|
||||||
useEffect(() => {
|
const [logs, setLogs] = useState([]);
|
||||||
refresh().then();
|
const [showStat, setShowStat] = useState(false);
|
||||||
}, [logType]);
|
const [loading, setLoading] = useState(false);
|
||||||
|
const [loadingStat, setLoadingStat] = useState(false);
|
||||||
|
const [activePage, setActivePage] = useState(1);
|
||||||
|
const [logCount, setLogCount] = useState(ITEMS_PER_PAGE);
|
||||||
|
const [pageSize, setPageSize] = useState(ITEMS_PER_PAGE);
|
||||||
|
const [searchKeyword, setSearchKeyword] = useState('');
|
||||||
|
const [searching, setSearching] = useState(false);
|
||||||
|
const [logType, setLogType] = useState(0);
|
||||||
|
const isAdminUser = isAdmin();
|
||||||
|
let now = new Date();
|
||||||
|
// 初始化start_timestamp为前一天
|
||||||
|
const [inputs, setInputs] = useState({
|
||||||
|
username: '',
|
||||||
|
token_name: '',
|
||||||
|
model_name: '',
|
||||||
|
start_timestamp: timestamp2string(now.getTime() / 1000 - 86400),
|
||||||
|
end_timestamp: timestamp2string(now.getTime() / 1000 + 3600),
|
||||||
|
channel: ''
|
||||||
|
});
|
||||||
|
const { username, token_name, model_name, start_timestamp, end_timestamp, channel } = inputs;
|
||||||
|
|
||||||
const searchLogs = async () => {
|
const [stat, setStat] = useState({
|
||||||
if (searchKeyword === '') {
|
quota: 0, token: 0
|
||||||
// if keyword is blank, load files instead.
|
});
|
||||||
await loadLogs(0);
|
|
||||||
setActivePage(1);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
setSearching(true);
|
|
||||||
const res = await API.get(`/api/log/self/search?keyword=${searchKeyword}`);
|
|
||||||
const {success, message, data} = res.data;
|
|
||||||
if (success) {
|
|
||||||
setLogs(data);
|
|
||||||
setActivePage(1);
|
|
||||||
} else {
|
|
||||||
showError(message);
|
|
||||||
}
|
|
||||||
setSearching(false);
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleKeywordChange = async (e, {value}) => {
|
const handleInputChange = (value, name) => {
|
||||||
setSearchKeyword(value.trim());
|
setInputs((inputs) => ({ ...inputs, [name]: value }));
|
||||||
};
|
};
|
||||||
|
|
||||||
const sortLog = (key) => {
|
const getLogSelfStat = async () => {
|
||||||
if (logs.length === 0) return;
|
let localStartTimestamp = Date.parse(start_timestamp) / 1000;
|
||||||
setLoading(true);
|
let localEndTimestamp = Date.parse(end_timestamp) / 1000;
|
||||||
let sortedLogs = [...logs];
|
let res = await API.get(`/api/log/self/stat?type=${logType}&token_name=${token_name}&model_name=${model_name}&start_timestamp=${localStartTimestamp}&end_timestamp=${localEndTimestamp}`);
|
||||||
if (typeof sortedLogs[0][key] === 'string') {
|
const { success, message, data } = res.data;
|
||||||
sortedLogs.sort((a, b) => {
|
if (success) {
|
||||||
return ('' + a[key]).localeCompare(b[key]);
|
setStat(data);
|
||||||
});
|
} else {
|
||||||
} else {
|
showError(message);
|
||||||
sortedLogs.sort((a, b) => {
|
}
|
||||||
if (a[key] === b[key]) return 0;
|
};
|
||||||
if (a[key] > b[key]) return -1;
|
|
||||||
if (a[key] < b[key]) return 1;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
if (sortedLogs[0].id === logs[0].id) {
|
|
||||||
sortedLogs.reverse();
|
|
||||||
}
|
|
||||||
setLogs(sortedLogs);
|
|
||||||
setLoading(false);
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
const getLogStat = async () => {
|
||||||
|
let localStartTimestamp = Date.parse(start_timestamp) / 1000;
|
||||||
|
let localEndTimestamp = Date.parse(end_timestamp) / 1000;
|
||||||
|
let res = await API.get(`/api/log/stat?type=${logType}&username=${username}&token_name=${token_name}&model_name=${model_name}&start_timestamp=${localStartTimestamp}&end_timestamp=${localEndTimestamp}&channel=${channel}`);
|
||||||
|
const { success, message, data } = res.data;
|
||||||
|
if (success) {
|
||||||
|
setStat(data);
|
||||||
|
} else {
|
||||||
|
showError(message);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleEyeClick = async () => {
|
||||||
|
setLoadingStat(true);
|
||||||
|
if (isAdminUser) {
|
||||||
|
await getLogStat();
|
||||||
|
} else {
|
||||||
|
await getLogSelfStat();
|
||||||
|
}
|
||||||
|
setShowStat(true);
|
||||||
|
setLoadingStat(false);
|
||||||
|
};
|
||||||
|
|
||||||
|
const showUserInfo = async (userId) => {
|
||||||
|
if (!isAdminUser) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const res = await API.get(`/api/user/${userId}`);
|
||||||
|
const { success, message, data } = res.data;
|
||||||
|
if (success) {
|
||||||
|
Modal.info({
|
||||||
|
title: '用户信息', content: <div style={{ padding: 12 }}>
|
||||||
|
<p>用户名: {data.username}</p>
|
||||||
|
<p>余额: {renderQuota(data.quota)}</p>
|
||||||
|
<p>已用额度:{renderQuota(data.used_quota)}</p>
|
||||||
|
<p>请求次数:{renderNumber(data.request_count)}</p>
|
||||||
|
</div>, centered: true
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
showError(message);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const setLogsFormat = (logs) => {
|
||||||
|
for (let i = 0; i < logs.length; i++) {
|
||||||
|
logs[i].timestamp2string = timestamp2string(logs[i].created_at);
|
||||||
|
logs[i].key = '' + logs[i].id;
|
||||||
|
}
|
||||||
|
// data.key = '' + data.id
|
||||||
|
setLogs(logs);
|
||||||
|
setLogCount(logs.length + ITEMS_PER_PAGE);
|
||||||
|
// console.log(logCount);
|
||||||
|
};
|
||||||
|
|
||||||
|
const loadLogs = async (startIdx, pageSize, logType = 0) => {
|
||||||
|
setLoading(true);
|
||||||
|
|
||||||
|
let url = '';
|
||||||
|
let localStartTimestamp = Date.parse(start_timestamp) / 1000;
|
||||||
|
let localEndTimestamp = Date.parse(end_timestamp) / 1000;
|
||||||
|
if (isAdminUser) {
|
||||||
|
url = `/api/log/?p=${startIdx}&page_size=${pageSize}&type=${logType}&username=${username}&token_name=${token_name}&model_name=${model_name}&start_timestamp=${localStartTimestamp}&end_timestamp=${localEndTimestamp}&channel=${channel}`;
|
||||||
|
} else {
|
||||||
|
url = `/api/log/self/?p=${startIdx}&page_size=${pageSize}&type=${logType}&token_name=${token_name}&model_name=${model_name}&start_timestamp=${localStartTimestamp}&end_timestamp=${localEndTimestamp}`;
|
||||||
|
}
|
||||||
|
const res = await API.get(url);
|
||||||
|
const { success, message, data } = res.data;
|
||||||
|
if (success) {
|
||||||
|
if (startIdx === 0) {
|
||||||
|
setLogsFormat(data);
|
||||||
|
} else {
|
||||||
|
let newLogs = [...logs];
|
||||||
|
newLogs.splice(startIdx * pageSize, data.length, ...data);
|
||||||
|
setLogsFormat(newLogs);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
showError(message);
|
||||||
|
}
|
||||||
|
setLoading(false);
|
||||||
|
};
|
||||||
|
|
||||||
|
const pageData = logs.slice((activePage - 1) * pageSize, activePage * pageSize);
|
||||||
|
|
||||||
|
const handlePageChange = page => {
|
||||||
|
setActivePage(page);
|
||||||
|
if (page === Math.ceil(logs.length / pageSize) + 1) {
|
||||||
|
// In this case we have to load more data and then append them.
|
||||||
|
loadLogs(page - 1, pageSize, logType).then(r => {
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handlePageSizeChange = async (size) => {
|
||||||
|
localStorage.setItem('page-size', size + '');
|
||||||
|
setPageSize(size);
|
||||||
|
setActivePage(1);
|
||||||
|
loadLogs(0, size)
|
||||||
|
.then()
|
||||||
|
.catch((reason) => {
|
||||||
|
showError(reason);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
const refresh = async (localLogType) => {
|
||||||
|
// setLoading(true);
|
||||||
|
setActivePage(1);
|
||||||
|
await loadLogs(0, pageSize, localLogType);
|
||||||
|
};
|
||||||
|
|
||||||
|
const copyText = async (text) => {
|
||||||
|
if (await copy(text)) {
|
||||||
|
showSuccess('已复制:' + text);
|
||||||
|
} else {
|
||||||
|
// setSearchKeyword(text);
|
||||||
|
Modal.error({ title: '无法复制到剪贴板,请手动复制', content: text });
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
// console.log('default effect')
|
||||||
|
const localPageSize = parseInt(localStorage.getItem('page-size')) || ITEMS_PER_PAGE;
|
||||||
|
setPageSize(localPageSize);
|
||||||
|
loadLogs(0, localPageSize)
|
||||||
|
.then()
|
||||||
|
.catch((reason) => {
|
||||||
|
showError(reason);
|
||||||
|
});
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const searchLogs = async () => {
|
||||||
|
if (searchKeyword === '') {
|
||||||
|
// if keyword is blank, load files instead.
|
||||||
|
await loadLogs(0, pageSize);
|
||||||
|
setActivePage(1);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
setSearching(true);
|
||||||
|
const res = await API.get(`/api/log/self/search?keyword=${searchKeyword}`);
|
||||||
|
const { success, message, data } = res.data;
|
||||||
|
if (success) {
|
||||||
|
setLogs(data);
|
||||||
|
setActivePage(1);
|
||||||
|
} else {
|
||||||
|
showError(message);
|
||||||
|
}
|
||||||
|
setSearching(false);
|
||||||
|
};
|
||||||
|
|
||||||
|
return (<>
|
||||||
|
<Layout>
|
||||||
|
<Header>
|
||||||
|
<Spin spinning={loadingStat}>
|
||||||
|
<h3>使用明细(总消耗额度:
|
||||||
|
<span onClick={handleEyeClick} style={{
|
||||||
|
cursor: 'pointer', color: 'gray'
|
||||||
|
}}>{showStat ? renderQuota(stat.quota) : '点击查看'}</span>
|
||||||
|
)
|
||||||
|
</h3>
|
||||||
|
</Spin>
|
||||||
|
</Header>
|
||||||
|
<Form layout="horizontal" style={{ marginTop: 10 }}>
|
||||||
<>
|
<>
|
||||||
<Layout>
|
<Form.Input field="token_name" label="令牌名称" style={{ width: 176 }} value={token_name}
|
||||||
<Header>
|
placeholder={'可选值'} name="token_name"
|
||||||
<Spin spinning={loadingStat}>
|
onChange={value => handleInputChange(value, 'token_name')} />
|
||||||
<h3>使用明细(总消耗额度:
|
<Form.Input field="model_name" label="模型名称" style={{ width: 176 }} value={model_name}
|
||||||
<span onClick={handleEyeClick} style={{cursor: 'pointer', color: 'gray'}}>{showStat?renderQuota(stat.quota):"点击查看"}</span>
|
placeholder="可选值"
|
||||||
)
|
name="model_name"
|
||||||
</h3>
|
onChange={value => handleInputChange(value, 'model_name')} />
|
||||||
</Spin>
|
<Form.DatePicker field="start_timestamp" label="起始时间" style={{ width: 272 }}
|
||||||
</Header>
|
initValue={start_timestamp}
|
||||||
<Form layout='horizontal' style={{marginTop: 10}}>
|
value={start_timestamp} type="dateTime"
|
||||||
<>
|
name="start_timestamp"
|
||||||
<Form.Input field="token_name" label='令牌名称' style={{width: 176}} value={token_name}
|
onChange={value => handleInputChange(value, 'start_timestamp')} />
|
||||||
placeholder={'可选值'} name='token_name'
|
<Form.DatePicker field="end_timestamp" fluid label="结束时间" style={{ width: 272 }}
|
||||||
onChange={value => handleInputChange(value, 'token_name')}/>
|
initValue={end_timestamp}
|
||||||
<Form.Input field="model_name" label='模型名称' style={{width: 176}} value={model_name}
|
value={end_timestamp} type="dateTime"
|
||||||
placeholder='可选值'
|
name="end_timestamp"
|
||||||
name='model_name'
|
onChange={value => handleInputChange(value, 'end_timestamp')} />
|
||||||
onChange={value => handleInputChange(value, 'model_name')}/>
|
{isAdminUser && <>
|
||||||
<Form.DatePicker field="start_timestamp" label='起始时间' style={{width: 272}}
|
<Form.Input field="channel" label="渠道 ID" style={{ width: 176 }} value={channel}
|
||||||
initValue={start_timestamp}
|
placeholder="可选值" name="channel"
|
||||||
value={start_timestamp} type='dateTime'
|
onChange={value => handleInputChange(value, 'channel')} />
|
||||||
name='start_timestamp'
|
<Form.Input field="username" label="用户名称" style={{ width: 176 }} value={username}
|
||||||
onChange={value => handleInputChange(value, 'start_timestamp')}/>
|
placeholder={'可选值'} name="username"
|
||||||
<Form.DatePicker field="end_timestamp" fluid label='结束时间' style={{width: 272}}
|
onChange={value => handleInputChange(value, 'username')} />
|
||||||
initValue={end_timestamp}
|
</>}
|
||||||
value={end_timestamp} type='dateTime'
|
<Form.Section>
|
||||||
name='end_timestamp'
|
<Button label="查询" type="primary" htmlType="submit" className="btn-margin-right"
|
||||||
onChange={value => handleInputChange(value, 'end_timestamp')}/>
|
onClick={refresh} loading={loading}>查询</Button>
|
||||||
{
|
</Form.Section>
|
||||||
isAdminUser && <>
|
|
||||||
<Form.Input field="channel" label='渠道 ID' style={{width: 176}} value={channel}
|
|
||||||
placeholder='可选值' name='channel'
|
|
||||||
onChange={value => handleInputChange(value, 'channel')}/>
|
|
||||||
<Form.Input field="username" label='用户名称' style={{width: 176}} value={username}
|
|
||||||
placeholder={'可选值'} name='username'
|
|
||||||
onChange={value => handleInputChange(value, 'username')}/>
|
|
||||||
</>
|
|
||||||
}
|
|
||||||
<Form.Section>
|
|
||||||
<Button label='查询' type="primary" htmlType="submit" className="btn-margin-right"
|
|
||||||
onClick={refresh} loading={loading}>查询</Button>
|
|
||||||
</Form.Section>
|
|
||||||
</>
|
|
||||||
</Form>
|
|
||||||
<Table style={{marginTop: 5}} columns={columns} dataSource={pageData} pagination={{
|
|
||||||
currentPage: activePage,
|
|
||||||
pageSize: ITEMS_PER_PAGE,
|
|
||||||
total: logCount,
|
|
||||||
pageSizeOpts: [10, 20, 50, 100],
|
|
||||||
onPageChange: handlePageChange,
|
|
||||||
}}/>
|
|
||||||
<Select defaultValue="0" style={{width: 120}} onChange={
|
|
||||||
(value) => {
|
|
||||||
setLogType(parseInt(value));
|
|
||||||
}
|
|
||||||
}>
|
|
||||||
<Select.Option value="0">全部</Select.Option>
|
|
||||||
<Select.Option value="1">充值</Select.Option>
|
|
||||||
<Select.Option value="2">消费</Select.Option>
|
|
||||||
<Select.Option value="3">管理</Select.Option>
|
|
||||||
<Select.Option value="4">系统</Select.Option>
|
|
||||||
</Select>
|
|
||||||
</Layout>
|
|
||||||
</>
|
</>
|
||||||
);
|
</Form>
|
||||||
|
<Table style={{ marginTop: 5 }} columns={columns} dataSource={pageData} pagination={{
|
||||||
|
currentPage: activePage,
|
||||||
|
pageSize: pageSize,
|
||||||
|
total: logCount,
|
||||||
|
pageSizeOpts: [10, 20, 50, 100],
|
||||||
|
showSizeChanger: true,
|
||||||
|
onPageSizeChange: (size) => {
|
||||||
|
handlePageSizeChange(size).then();
|
||||||
|
},
|
||||||
|
onPageChange: handlePageChange
|
||||||
|
}} />
|
||||||
|
<Select defaultValue="0" style={{ width: 120 }} onChange={(value) => {
|
||||||
|
setLogType(parseInt(value));
|
||||||
|
refresh(parseInt(value)).then();
|
||||||
|
}}>
|
||||||
|
<Select.Option value="0">全部</Select.Option>
|
||||||
|
<Select.Option value="1">充值</Select.Option>
|
||||||
|
<Select.Option value="2">消费</Select.Option>
|
||||||
|
<Select.Option value="3">管理</Select.Option>
|
||||||
|
<Select.Option value="4">系统</Select.Option>
|
||||||
|
</Select>
|
||||||
|
</Layout>
|
||||||
|
</>);
|
||||||
};
|
};
|
||||||
|
|
||||||
export default LogsTable;
|
export default LogsTable;
|
||||||
|
@ -1,454 +1,454 @@
|
|||||||
import React, {useEffect, useState} from 'react';
|
import React, { useEffect, useState } from 'react';
|
||||||
import {API, copy, isAdmin, showError, showSuccess, timestamp2string} from '../helpers';
|
import { API, copy, isAdmin, showError, showSuccess, timestamp2string } from '../helpers';
|
||||||
|
|
||||||
import {
|
import { Banner, Button, Form, ImagePreview, Layout, Modal, Progress, Table, Tag, Typography } from '@douyinfe/semi-ui';
|
||||||
Table,
|
import { ITEMS_PER_PAGE } from '../constants';
|
||||||
Avatar,
|
|
||||||
Tag,
|
|
||||||
Form,
|
|
||||||
Button,
|
|
||||||
Layout,
|
|
||||||
Select,
|
|
||||||
Popover,
|
|
||||||
Modal,
|
|
||||||
ImagePreview,
|
|
||||||
Typography, Progress
|
|
||||||
} from '@douyinfe/semi-ui';
|
|
||||||
import {ITEMS_PER_PAGE} from '../constants';
|
|
||||||
import {renderNumber, renderQuota, stringToColor} from '../helpers/render';
|
|
||||||
|
|
||||||
|
|
||||||
const colors = ['amber', 'blue', 'cyan', 'green', 'grey', 'indigo',
|
const colors = ['amber', 'blue', 'cyan', 'green', 'grey', 'indigo',
|
||||||
'light-blue', 'lime', 'orange', 'pink',
|
'light-blue', 'lime', 'orange', 'pink',
|
||||||
'purple', 'red', 'teal', 'violet', 'yellow'
|
'purple', 'red', 'teal', 'violet', 'yellow'
|
||||||
]
|
];
|
||||||
|
|
||||||
function renderType(type) {
|
function renderType(type) {
|
||||||
switch (type) {
|
switch (type) {
|
||||||
case 'IMAGINE':
|
case 'IMAGINE':
|
||||||
return <Tag color="blue" size='large'>绘图</Tag>;
|
return <Tag color="blue" size="large">绘图</Tag>;
|
||||||
case 'UPSCALE':
|
case 'UPSCALE':
|
||||||
return <Tag color="orange" size='large'>放大</Tag>;
|
return <Tag color="orange" size="large">放大</Tag>;
|
||||||
case 'VARIATION':
|
case 'VARIATION':
|
||||||
return <Tag color="purple" size='large'>变换</Tag>;
|
return <Tag color="purple" size="large">变换</Tag>;
|
||||||
case 'HIGH_VARIATION':
|
case 'HIGH_VARIATION':
|
||||||
return <Tag color="purple" size='large'>强变换</Tag>;
|
return <Tag color="purple" size="large">强变换</Tag>;
|
||||||
case 'LOW_VARIATION':
|
case 'LOW_VARIATION':
|
||||||
return <Tag color="purple" size='large'>弱变换</Tag>;
|
return <Tag color="purple" size="large">弱变换</Tag>;
|
||||||
case 'PAN':
|
case 'PAN':
|
||||||
return <Tag color="cyan" size='large'>平移</Tag>;
|
return <Tag color="cyan" size="large">平移</Tag>;
|
||||||
case 'DESCRIBE':
|
case 'DESCRIBE':
|
||||||
return <Tag color="yellow" size='large'>图生文</Tag>;
|
return <Tag color="yellow" size="large">图生文</Tag>;
|
||||||
case 'BLEND':
|
case 'BLEND':
|
||||||
return <Tag color="lime" size='large'>图混合</Tag>;
|
return <Tag color="lime" size="large">图混合</Tag>;
|
||||||
case 'SHORTEN':
|
case 'SHORTEN':
|
||||||
return <Tag color="pink" size='large'>缩词</Tag>;
|
return <Tag color="pink" size="large">缩词</Tag>;
|
||||||
case 'REROLL':
|
case 'REROLL':
|
||||||
return <Tag color="indigo" size='large'>重绘</Tag>;
|
return <Tag color="indigo" size="large">重绘</Tag>;
|
||||||
case 'INPAINT':
|
case 'INPAINT':
|
||||||
return <Tag color="violet" size='large'>局部重绘-提交</Tag>;
|
return <Tag color="violet" size="large">局部重绘-提交</Tag>;
|
||||||
case 'ZOOM':
|
case 'ZOOM':
|
||||||
return <Tag color="teal" size='large'>变焦</Tag>;
|
return <Tag color="teal" size="large">变焦</Tag>;
|
||||||
case 'CUSTOM_ZOOM':
|
case 'CUSTOM_ZOOM':
|
||||||
return <Tag color="teal" size='large'>自定义变焦-提交</Tag>;
|
return <Tag color="teal" size="large">自定义变焦-提交</Tag>;
|
||||||
case 'MODAL':
|
case 'MODAL':
|
||||||
return <Tag color="green" size='large'>窗口处理</Tag>;
|
return <Tag color="green" size="large">窗口处理</Tag>;
|
||||||
case 'SWAP_FACE':
|
case 'SWAP_FACE':
|
||||||
return <Tag color="light-green" size='large'>换脸</Tag>;
|
return <Tag color="light-green" size="large">换脸</Tag>;
|
||||||
default:
|
default:
|
||||||
return <Tag color="white" size='large'>未知</Tag>;
|
return <Tag color="white" size="large">未知</Tag>;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
function renderCode(code) {
|
function renderCode(code) {
|
||||||
switch (code) {
|
switch (code) {
|
||||||
case 1:
|
case 1:
|
||||||
return <Tag color="green" size='large'>已提交</Tag>;
|
return <Tag color="green" size="large">已提交</Tag>;
|
||||||
case 21:
|
case 21:
|
||||||
return <Tag color="lime" size='large'>等待中</Tag>;
|
return <Tag color="lime" size="large">等待中</Tag>;
|
||||||
case 22:
|
case 22:
|
||||||
return <Tag color="orange" size='large'>重复提交</Tag>;
|
return <Tag color="orange" size="large">重复提交</Tag>;
|
||||||
case 0:
|
case 0:
|
||||||
return <Tag color="yellow" size='large'>未提交</Tag>;
|
return <Tag color="yellow" size="large">未提交</Tag>;
|
||||||
default:
|
default:
|
||||||
return <Tag color="white" size='large'>未知</Tag>;
|
return <Tag color="white" size="large">未知</Tag>;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
function renderStatus(type) {
|
function renderStatus(type) {
|
||||||
// Ensure all cases are string literals by adding quotes.
|
// Ensure all cases are string literals by adding quotes.
|
||||||
switch (type) {
|
switch (type) {
|
||||||
case 'SUCCESS':
|
case 'SUCCESS':
|
||||||
return <Tag color="green" size='large'>成功</Tag>;
|
return <Tag color="green" size="large">成功</Tag>;
|
||||||
case 'NOT_START':
|
case 'NOT_START':
|
||||||
return <Tag color="grey" size='large'>未启动</Tag>;
|
return <Tag color="grey" size="large">未启动</Tag>;
|
||||||
case 'SUBMITTED':
|
case 'SUBMITTED':
|
||||||
return <Tag color="yellow" size='large'>队列中</Tag>;
|
return <Tag color="yellow" size="large">队列中</Tag>;
|
||||||
case 'IN_PROGRESS':
|
case 'IN_PROGRESS':
|
||||||
return <Tag color="blue" size='large'>执行中</Tag>;
|
return <Tag color="blue" size="large">执行中</Tag>;
|
||||||
case 'FAILURE':
|
case 'FAILURE':
|
||||||
return <Tag color="red" size='large'>失败</Tag>;
|
return <Tag color="red" size="large">失败</Tag>;
|
||||||
case 'MODAL':
|
case 'MODAL':
|
||||||
return <Tag color="yellow" size='large'>窗口等待</Tag>;
|
return <Tag color="yellow" size="large">窗口等待</Tag>;
|
||||||
default:
|
default:
|
||||||
return <Tag color="white" size='large'>未知</Tag>;
|
return <Tag color="white" size="large">未知</Tag>;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const renderTimestamp = (timestampInSeconds) => {
|
const renderTimestamp = (timestampInSeconds) => {
|
||||||
const date = new Date(timestampInSeconds * 1000); // 从秒转换为毫秒
|
const date = new Date(timestampInSeconds * 1000); // 从秒转换为毫秒
|
||||||
|
|
||||||
const year = date.getFullYear(); // 获取年份
|
const year = date.getFullYear(); // 获取年份
|
||||||
const month = ('0' + (date.getMonth() + 1)).slice(-2); // 获取月份,从0开始需要+1,并保证两位数
|
const month = ('0' + (date.getMonth() + 1)).slice(-2); // 获取月份,从0开始需要+1,并保证两位数
|
||||||
const day = ('0' + date.getDate()).slice(-2); // 获取日期,并保证两位数
|
const day = ('0' + date.getDate()).slice(-2); // 获取日期,并保证两位数
|
||||||
const hours = ('0' + date.getHours()).slice(-2); // 获取小时,并保证两位数
|
const hours = ('0' + date.getHours()).slice(-2); // 获取小时,并保证两位数
|
||||||
const minutes = ('0' + date.getMinutes()).slice(-2); // 获取分钟,并保证两位数
|
const minutes = ('0' + date.getMinutes()).slice(-2); // 获取分钟,并保证两位数
|
||||||
const seconds = ('0' + date.getSeconds()).slice(-2); // 获取秒钟,并保证两位数
|
const seconds = ('0' + date.getSeconds()).slice(-2); // 获取秒钟,并保证两位数
|
||||||
|
|
||||||
return `${year}-${month}-${day} ${hours}:${minutes}:${seconds}`; // 格式化输出
|
return `${year}-${month}-${day} ${hours}:${minutes}:${seconds}`; // 格式化输出
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
const LogsTable = () => {
|
const LogsTable = () => {
|
||||||
const [isModalOpen, setIsModalOpen] = useState(false);
|
const [isModalOpen, setIsModalOpen] = useState(false);
|
||||||
const [modalContent, setModalContent] = useState('');
|
const [modalContent, setModalContent] = useState('');
|
||||||
const columns = [
|
const columns = [
|
||||||
{
|
{
|
||||||
title: '提交时间',
|
title: '提交时间',
|
||||||
dataIndex: 'submit_time',
|
dataIndex: 'submit_time',
|
||||||
render: (text, record, index) => {
|
render: (text, record, index) => {
|
||||||
return (
|
return (
|
||||||
<div>
|
<div>
|
||||||
{renderTimestamp(text / 1000)}
|
{renderTimestamp(text / 1000)}
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
},
|
}
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: '渠道',
|
title: '渠道',
|
||||||
dataIndex: 'channel_id',
|
dataIndex: 'channel_id',
|
||||||
className: isAdmin() ? 'tableShow' : 'tableHiddle',
|
className: isAdmin() ? 'tableShow' : 'tableHiddle',
|
||||||
render: (text, record, index) => {
|
render: (text, record, index) => {
|
||||||
return (
|
return (
|
||||||
|
|
||||||
<div>
|
<div>
|
||||||
<Tag color={colors[parseInt(text) % colors.length]} size='large' onClick={() => {
|
<Tag color={colors[parseInt(text) % colors.length]} size="large" onClick={() => {
|
||||||
copyText(text); // 假设copyText是用于文本复制的函数
|
copyText(text); // 假设copyText是用于文本复制的函数
|
||||||
}}> {text} </Tag>
|
}}> {text} </Tag>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
);
|
);
|
||||||
},
|
}
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: '类型',
|
title: '类型',
|
||||||
dataIndex: 'action',
|
dataIndex: 'action',
|
||||||
render: (text, record, index) => {
|
render: (text, record, index) => {
|
||||||
return (
|
return (
|
||||||
<div>
|
<div>
|
||||||
{renderType(text)}
|
{renderType(text)}
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
},
|
}
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: '任务ID',
|
title: '任务ID',
|
||||||
dataIndex: 'mj_id',
|
dataIndex: 'mj_id',
|
||||||
render: (text, record, index) => {
|
render: (text, record, index) => {
|
||||||
return (
|
return (
|
||||||
<div>
|
<div>
|
||||||
{text}
|
{text}
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
},
|
}
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: '提交结果',
|
title: '提交结果',
|
||||||
dataIndex: 'code',
|
dataIndex: 'code',
|
||||||
className: isAdmin() ? 'tableShow' : 'tableHiddle',
|
className: isAdmin() ? 'tableShow' : 'tableHiddle',
|
||||||
render: (text, record, index) => {
|
render: (text, record, index) => {
|
||||||
return (
|
return (
|
||||||
<div>
|
<div>
|
||||||
{renderCode(text)}
|
{renderCode(text)}
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
},
|
}
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: '任务状态',
|
title: '任务状态',
|
||||||
dataIndex: 'status',
|
dataIndex: 'status',
|
||||||
className: isAdmin() ? 'tableShow' : 'tableHiddle',
|
className: isAdmin() ? 'tableShow' : 'tableHiddle',
|
||||||
render: (text, record, index) => {
|
render: (text, record, index) => {
|
||||||
return (
|
return (
|
||||||
<div>
|
<div>
|
||||||
{renderStatus(text)}
|
{renderStatus(text)}
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
},
|
}
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: '进度',
|
title: '进度',
|
||||||
dataIndex: 'progress',
|
dataIndex: 'progress',
|
||||||
render: (text, record, index) => {
|
render: (text, record, index) => {
|
||||||
return (
|
return (
|
||||||
<div>
|
<div>
|
||||||
{
|
{
|
||||||
// 转换例如100%为数字100,如果text未定义,返回0
|
// 转换例如100%为数字100,如果text未定义,返回0
|
||||||
<Progress stroke={record.status === "FAILURE"?"var(--semi-color-warning)":null} percent={text ? parseInt(text.replace('%', '')) : 0} showInfo={true}
|
<Progress stroke={record.status === 'FAILURE' ? 'var(--semi-color-warning)' : null}
|
||||||
aria-label="drawing progress"/>
|
percent={text ? parseInt(text.replace('%', '')) : 0} showInfo={true}
|
||||||
}
|
aria-label="drawing progress" />
|
||||||
</div>
|
|
||||||
);
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
title: '结果图片',
|
|
||||||
dataIndex: 'image_url',
|
|
||||||
render: (text, record, index) => {
|
|
||||||
if (!text) {
|
|
||||||
return '无';
|
|
||||||
}
|
|
||||||
return (
|
|
||||||
<Button
|
|
||||||
onClick={() => {
|
|
||||||
setModalImageUrl(text); // 更新图片URL状态
|
|
||||||
setIsModalOpenurl(true); // 打开模态框
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
查看图片
|
|
||||||
</Button>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
title: 'Prompt',
|
|
||||||
dataIndex: 'prompt',
|
|
||||||
render: (text, record, index) => {
|
|
||||||
// 如果text未定义,返回替代文本,例如空字符串''或其他
|
|
||||||
if (!text) {
|
|
||||||
return '无';
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Typography.Text
|
|
||||||
ellipsis={{showTooltip: true}}
|
|
||||||
style={{width: 100}}
|
|
||||||
onClick={() => {
|
|
||||||
setModalContent(text);
|
|
||||||
setIsModalOpen(true);
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
{text}
|
|
||||||
</Typography.Text>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
title: 'PromptEn',
|
|
||||||
dataIndex: 'prompt_en',
|
|
||||||
render: (text, record, index) => {
|
|
||||||
// 如果text未定义,返回替代文本,例如空字符串''或其他
|
|
||||||
if (!text) {
|
|
||||||
return '无';
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Typography.Text
|
|
||||||
ellipsis={{showTooltip: true}}
|
|
||||||
style={{width: 100}}
|
|
||||||
onClick={() => {
|
|
||||||
setModalContent(text);
|
|
||||||
setIsModalOpen(true);
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
{text}
|
|
||||||
</Typography.Text>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
title: '失败原因',
|
|
||||||
dataIndex: 'fail_reason',
|
|
||||||
render: (text, record, index) => {
|
|
||||||
// 如果text未定义,返回替代文本,例如空字符串''或其他
|
|
||||||
if (!text) {
|
|
||||||
return '无';
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Typography.Text
|
|
||||||
ellipsis={{showTooltip: true}}
|
|
||||||
style={{width: 100}}
|
|
||||||
onClick={() => {
|
|
||||||
setModalContent(text);
|
|
||||||
setIsModalOpen(true);
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
{text}
|
|
||||||
</Typography.Text>
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: '结果图片',
|
||||||
|
dataIndex: 'image_url',
|
||||||
|
render: (text, record, index) => {
|
||||||
|
if (!text) {
|
||||||
|
return '无';
|
||||||
|
}
|
||||||
|
return (
|
||||||
|
<Button
|
||||||
|
onClick={() => {
|
||||||
|
setModalImageUrl(text); // 更新图片URL状态
|
||||||
|
setIsModalOpenurl(true); // 打开模态框
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
查看图片
|
||||||
|
</Button>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: 'Prompt',
|
||||||
|
dataIndex: 'prompt',
|
||||||
|
render: (text, record, index) => {
|
||||||
|
// 如果text未定义,返回替代文本,例如空字符串''或其他
|
||||||
|
if (!text) {
|
||||||
|
return '无';
|
||||||
}
|
}
|
||||||
|
|
||||||
];
|
return (
|
||||||
|
<Typography.Text
|
||||||
const [logs, setLogs] = useState([]);
|
ellipsis={{ showTooltip: true }}
|
||||||
const [loading, setLoading] = useState(true);
|
style={{ width: 100 }}
|
||||||
const [activePage, setActivePage] = useState(1);
|
onClick={() => {
|
||||||
const [logCount, setLogCount] = useState(ITEMS_PER_PAGE);
|
setModalContent(text);
|
||||||
const [logType, setLogType] = useState(0);
|
setIsModalOpen(true);
|
||||||
const isAdminUser = isAdmin();
|
}}
|
||||||
const [isModalOpenurl, setIsModalOpenurl] = useState(false);
|
>
|
||||||
|
{text}
|
||||||
// 定义模态框图片URL的状态和更新函数
|
</Typography.Text>
|
||||||
const [modalImageUrl, setModalImageUrl] = useState('');
|
);
|
||||||
let now = new Date();
|
}
|
||||||
// 初始化start_timestamp为前一天
|
},
|
||||||
const [inputs, setInputs] = useState({
|
{
|
||||||
channel_id: '',
|
title: 'PromptEn',
|
||||||
mj_id: '',
|
dataIndex: 'prompt_en',
|
||||||
start_timestamp: timestamp2string(now.getTime() / 1000 - 2592000),
|
render: (text, record, index) => {
|
||||||
end_timestamp: timestamp2string(now.getTime() / 1000 + 3600),
|
// 如果text未定义,返回替代文本,例如空字符串''或其他
|
||||||
});
|
if (!text) {
|
||||||
const {channel_id, mj_id, start_timestamp, end_timestamp} = inputs;
|
return '无';
|
||||||
|
|
||||||
const [stat, setStat] = useState({
|
|
||||||
quota: 0,
|
|
||||||
token: 0
|
|
||||||
});
|
|
||||||
|
|
||||||
const handleInputChange = (value, name) => {
|
|
||||||
setInputs((inputs) => ({...inputs, [name]: value}));
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
const setLogsFormat = (logs) => {
|
|
||||||
for (let i = 0; i < logs.length; i++) {
|
|
||||||
logs[i].timestamp2string = timestamp2string(logs[i].created_at);
|
|
||||||
logs[i].key = '' + logs[i].id;
|
|
||||||
}
|
}
|
||||||
// data.key = '' + data.id
|
|
||||||
setLogs(logs);
|
return (
|
||||||
setLogCount(logs.length + ITEMS_PER_PAGE);
|
<Typography.Text
|
||||||
// console.log(logCount);
|
ellipsis={{ showTooltip: true }}
|
||||||
|
style={{ width: 100 }}
|
||||||
|
onClick={() => {
|
||||||
|
setModalContent(text);
|
||||||
|
setIsModalOpen(true);
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{text}
|
||||||
|
</Typography.Text>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: '失败原因',
|
||||||
|
dataIndex: 'fail_reason',
|
||||||
|
render: (text, record, index) => {
|
||||||
|
// 如果text未定义,返回替代文本,例如空字符串''或其他
|
||||||
|
if (!text) {
|
||||||
|
return '无';
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Typography.Text
|
||||||
|
ellipsis={{ showTooltip: true }}
|
||||||
|
style={{ width: 100 }}
|
||||||
|
onClick={() => {
|
||||||
|
setModalContent(text);
|
||||||
|
setIsModalOpen(true);
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{text}
|
||||||
|
</Typography.Text>
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const loadLogs = async (startIdx) => {
|
];
|
||||||
setLoading(true);
|
|
||||||
|
|
||||||
let url = '';
|
const [logs, setLogs] = useState([]);
|
||||||
let localStartTimestamp = Date.parse(start_timestamp);
|
const [loading, setLoading] = useState(true);
|
||||||
let localEndTimestamp = Date.parse(end_timestamp);
|
const [activePage, setActivePage] = useState(1);
|
||||||
if (isAdminUser) {
|
const [logCount, setLogCount] = useState(ITEMS_PER_PAGE);
|
||||||
url = `/api/mj/?p=${startIdx}&channel_id=${channel_id}&mj_id=${mj_id}&start_timestamp=${localStartTimestamp}&end_timestamp=${localEndTimestamp}`;
|
const [logType, setLogType] = useState(0);
|
||||||
} else {
|
const isAdminUser = isAdmin();
|
||||||
url = `/api/mj/self/?p=${startIdx}&mj_id=${mj_id}&start_timestamp=${localStartTimestamp}&end_timestamp=${localEndTimestamp}`;
|
const [isModalOpenurl, setIsModalOpenurl] = useState(false);
|
||||||
}
|
const [showBanner, setShowBanner] = useState(false);
|
||||||
const res = await API.get(url);
|
|
||||||
const {success, message, data} = res.data;
|
|
||||||
if (success) {
|
|
||||||
if (startIdx === 0) {
|
|
||||||
setLogsFormat(data);
|
|
||||||
} else {
|
|
||||||
let newLogs = [...logs];
|
|
||||||
newLogs.splice(startIdx * ITEMS_PER_PAGE, data.length, ...data);
|
|
||||||
setLogsFormat(newLogs);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
showError(message);
|
|
||||||
}
|
|
||||||
setLoading(false);
|
|
||||||
};
|
|
||||||
|
|
||||||
const pageData = logs.slice((activePage - 1) * ITEMS_PER_PAGE, activePage * ITEMS_PER_PAGE);
|
// 定义模态框图片URL的状态和更新函数
|
||||||
|
const [modalImageUrl, setModalImageUrl] = useState('');
|
||||||
|
let now = new Date();
|
||||||
|
// 初始化start_timestamp为前一天
|
||||||
|
const [inputs, setInputs] = useState({
|
||||||
|
channel_id: '',
|
||||||
|
mj_id: '',
|
||||||
|
start_timestamp: timestamp2string(now.getTime() / 1000 - 2592000),
|
||||||
|
end_timestamp: timestamp2string(now.getTime() / 1000 + 3600)
|
||||||
|
});
|
||||||
|
const { channel_id, mj_id, start_timestamp, end_timestamp } = inputs;
|
||||||
|
|
||||||
const handlePageChange = page => {
|
const [stat, setStat] = useState({
|
||||||
setActivePage(page);
|
quota: 0,
|
||||||
if (page === Math.ceil(logs.length / ITEMS_PER_PAGE) + 1) {
|
token: 0
|
||||||
// In this case we have to load more data and then append them.
|
});
|
||||||
loadLogs(page - 1).then(r => {
|
|
||||||
});
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const refresh = async () => {
|
const handleInputChange = (value, name) => {
|
||||||
// setLoading(true);
|
setInputs((inputs) => ({ ...inputs, [name]: value }));
|
||||||
setActivePage(1);
|
};
|
||||||
await loadLogs(0);
|
|
||||||
};
|
|
||||||
|
|
||||||
const copyText = async (text) => {
|
|
||||||
if (await copy(text)) {
|
const setLogsFormat = (logs) => {
|
||||||
showSuccess('已复制:' + text);
|
for (let i = 0; i < logs.length; i++) {
|
||||||
} else {
|
logs[i].timestamp2string = timestamp2string(logs[i].created_at);
|
||||||
// setSearchKeyword(text);
|
logs[i].key = '' + logs[i].id;
|
||||||
Modal.error({title: '无法复制到剪贴板,请手动复制', content: text});
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
// data.key = '' + data.id
|
||||||
|
setLogs(logs);
|
||||||
|
setLogCount(logs.length + ITEMS_PER_PAGE);
|
||||||
|
// console.log(logCount);
|
||||||
|
};
|
||||||
|
|
||||||
useEffect(() => {
|
const loadLogs = async (startIdx) => {
|
||||||
refresh().then();
|
setLoading(true);
|
||||||
}, [logType]);
|
|
||||||
|
|
||||||
|
let url = '';
|
||||||
|
let localStartTimestamp = Date.parse(start_timestamp);
|
||||||
|
let localEndTimestamp = Date.parse(end_timestamp);
|
||||||
|
if (isAdminUser) {
|
||||||
|
url = `/api/mj/?p=${startIdx}&channel_id=${channel_id}&mj_id=${mj_id}&start_timestamp=${localStartTimestamp}&end_timestamp=${localEndTimestamp}`;
|
||||||
|
} else {
|
||||||
|
url = `/api/mj/self/?p=${startIdx}&mj_id=${mj_id}&start_timestamp=${localStartTimestamp}&end_timestamp=${localEndTimestamp}`;
|
||||||
|
}
|
||||||
|
const res = await API.get(url);
|
||||||
|
const { success, message, data } = res.data;
|
||||||
|
if (success) {
|
||||||
|
if (startIdx === 0) {
|
||||||
|
setLogsFormat(data);
|
||||||
|
} else {
|
||||||
|
let newLogs = [...logs];
|
||||||
|
newLogs.splice(startIdx * ITEMS_PER_PAGE, data.length, ...data);
|
||||||
|
setLogsFormat(newLogs);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
showError(message);
|
||||||
|
}
|
||||||
|
setLoading(false);
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
const pageData = logs.slice((activePage - 1) * ITEMS_PER_PAGE, activePage * ITEMS_PER_PAGE);
|
||||||
<>
|
|
||||||
|
|
||||||
<Layout>
|
const handlePageChange = page => {
|
||||||
<Form layout='horizontal' style={{marginTop: 10}}>
|
setActivePage(page);
|
||||||
<>
|
if (page === Math.ceil(logs.length / ITEMS_PER_PAGE) + 1) {
|
||||||
<Form.Input field="channel_id" label='渠道 ID' style={{width: 176}} value={channel_id}
|
// In this case we have to load more data and then append them.
|
||||||
placeholder={'可选值'} name='channel_id'
|
loadLogs(page - 1).then(r => {
|
||||||
onChange={value => handleInputChange(value, 'channel_id')}/>
|
});
|
||||||
<Form.Input field="mj_id" label='任务 ID' style={{width: 176}} value={mj_id}
|
}
|
||||||
placeholder='可选值'
|
};
|
||||||
name='mj_id'
|
|
||||||
onChange={value => handleInputChange(value, 'mj_id')}/>
|
|
||||||
<Form.DatePicker field="start_timestamp" label='起始时间' style={{width: 272}}
|
|
||||||
initValue={start_timestamp}
|
|
||||||
value={start_timestamp} type='dateTime'
|
|
||||||
name='start_timestamp'
|
|
||||||
onChange={value => handleInputChange(value, 'start_timestamp')}/>
|
|
||||||
<Form.DatePicker field="end_timestamp" fluid label='结束时间' style={{width: 272}}
|
|
||||||
initValue={end_timestamp}
|
|
||||||
value={end_timestamp} type='dateTime'
|
|
||||||
name='end_timestamp'
|
|
||||||
onChange={value => handleInputChange(value, 'end_timestamp')}/>
|
|
||||||
|
|
||||||
<Form.Section>
|
const refresh = async () => {
|
||||||
<Button label='查询' type="primary" htmlType="submit" className="btn-margin-right"
|
// setLoading(true);
|
||||||
onClick={refresh}>查询</Button>
|
setActivePage(1);
|
||||||
</Form.Section>
|
await loadLogs(0);
|
||||||
</>
|
};
|
||||||
</Form>
|
|
||||||
<Table style={{marginTop: 5}} columns={columns} dataSource={pageData} pagination={{
|
|
||||||
currentPage: activePage,
|
|
||||||
pageSize: ITEMS_PER_PAGE,
|
|
||||||
total: logCount,
|
|
||||||
pageSizeOpts: [10, 20, 50, 100],
|
|
||||||
onPageChange: handlePageChange,
|
|
||||||
}} loading={loading}/>
|
|
||||||
<Modal
|
|
||||||
visible={isModalOpen}
|
|
||||||
onOk={() => setIsModalOpen(false)}
|
|
||||||
onCancel={() => setIsModalOpen(false)}
|
|
||||||
closable={null}
|
|
||||||
bodyStyle={{height: '400px', overflow: 'auto'}} // 设置模态框内容区域样式
|
|
||||||
width={800} // 设置模态框宽度
|
|
||||||
>
|
|
||||||
<p style={{whiteSpace: 'pre-line'}}>{modalContent}</p>
|
|
||||||
</Modal>
|
|
||||||
<ImagePreview
|
|
||||||
src={modalImageUrl}
|
|
||||||
visible={isModalOpenurl}
|
|
||||||
onVisibleChange={(visible) => setIsModalOpenurl(visible)}
|
|
||||||
/>
|
|
||||||
|
|
||||||
</Layout>
|
const copyText = async (text) => {
|
||||||
</>
|
if (await copy(text)) {
|
||||||
);
|
showSuccess('已复制:' + text);
|
||||||
|
} else {
|
||||||
|
// setSearchKeyword(text);
|
||||||
|
Modal.error({ title: '无法复制到剪贴板,请手动复制', content: text });
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
refresh().then();
|
||||||
|
}, [logType]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const mjNotifyEnabled = localStorage.getItem('mj_notify_enabled');
|
||||||
|
if (mjNotifyEnabled !== 'true') {
|
||||||
|
setShowBanner(true);
|
||||||
|
}
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
|
||||||
|
<Layout>
|
||||||
|
{isAdminUser && showBanner ? <Banner
|
||||||
|
type="info"
|
||||||
|
description="当前未开启Midjourney回调,部分项目可能无法获得绘图结果,可在运营设置中开启。"
|
||||||
|
/> : <></>
|
||||||
|
}
|
||||||
|
<Form layout="horizontal" style={{ marginTop: 10 }}>
|
||||||
|
<>
|
||||||
|
<Form.Input field="channel_id" label="渠道 ID" style={{ width: 176 }} value={channel_id}
|
||||||
|
placeholder={'可选值'} name="channel_id"
|
||||||
|
onChange={value => handleInputChange(value, 'channel_id')} />
|
||||||
|
<Form.Input field="mj_id" label="任务 ID" style={{ width: 176 }} value={mj_id}
|
||||||
|
placeholder="可选值"
|
||||||
|
name="mj_id"
|
||||||
|
onChange={value => handleInputChange(value, 'mj_id')} />
|
||||||
|
<Form.DatePicker field="start_timestamp" label="起始时间" style={{ width: 272 }}
|
||||||
|
initValue={start_timestamp}
|
||||||
|
value={start_timestamp} type="dateTime"
|
||||||
|
name="start_timestamp"
|
||||||
|
onChange={value => handleInputChange(value, 'start_timestamp')} />
|
||||||
|
<Form.DatePicker field="end_timestamp" fluid label="结束时间" style={{ width: 272 }}
|
||||||
|
initValue={end_timestamp}
|
||||||
|
value={end_timestamp} type="dateTime"
|
||||||
|
name="end_timestamp"
|
||||||
|
onChange={value => handleInputChange(value, 'end_timestamp')} />
|
||||||
|
|
||||||
|
<Form.Section>
|
||||||
|
<Button label="查询" type="primary" htmlType="submit" className="btn-margin-right"
|
||||||
|
onClick={refresh}>查询</Button>
|
||||||
|
</Form.Section>
|
||||||
|
</>
|
||||||
|
</Form>
|
||||||
|
<Table style={{ marginTop: 5 }} columns={columns} dataSource={pageData} pagination={{
|
||||||
|
currentPage: activePage,
|
||||||
|
pageSize: ITEMS_PER_PAGE,
|
||||||
|
total: logCount,
|
||||||
|
pageSizeOpts: [10, 20, 50, 100],
|
||||||
|
onPageChange: handlePageChange
|
||||||
|
}} loading={loading} />
|
||||||
|
<Modal
|
||||||
|
visible={isModalOpen}
|
||||||
|
onOk={() => setIsModalOpen(false)}
|
||||||
|
onCancel={() => setIsModalOpen(false)}
|
||||||
|
closable={null}
|
||||||
|
bodyStyle={{ height: '400px', overflow: 'auto' }} // 设置模态框内容区域样式
|
||||||
|
width={800} // 设置模态框宽度
|
||||||
|
>
|
||||||
|
<p style={{ whiteSpace: 'pre-line' }}>{modalContent}</p>
|
||||||
|
</Modal>
|
||||||
|
<ImagePreview
|
||||||
|
src={modalImageUrl}
|
||||||
|
visible={isModalOpenurl}
|
||||||
|
onVisibleChange={(visible) => setIsModalOpenurl(visible)}
|
||||||
|
/>
|
||||||
|
|
||||||
|
</Layout>
|
||||||
|
</>
|
||||||
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export default LogsTable;
|
export default LogsTable;
|
||||||
|
@ -1,453 +1,468 @@
|
|||||||
import React, {useEffect, useState} from 'react';
|
import React, { useEffect, useState } from 'react';
|
||||||
import {Divider, Form, Grid, Header} from 'semantic-ui-react';
|
import { Divider, Form, Grid, Header } from 'semantic-ui-react';
|
||||||
import {API, showError, showSuccess, timestamp2string, verifyJSON} from '../helpers';
|
import { API, showError, showSuccess, timestamp2string, verifyJSON } from '../helpers';
|
||||||
|
|
||||||
const OperationSetting = () => {
|
const OperationSetting = () => {
|
||||||
let now = new Date();
|
let now = new Date();
|
||||||
let [inputs, setInputs] = useState({
|
let [inputs, setInputs] = useState({
|
||||||
QuotaForNewUser: 0,
|
QuotaForNewUser: 0,
|
||||||
QuotaForInviter: 0,
|
QuotaForInviter: 0,
|
||||||
QuotaForInvitee: 0,
|
QuotaForInvitee: 0,
|
||||||
QuotaRemindThreshold: 0,
|
QuotaRemindThreshold: 0,
|
||||||
PreConsumedQuota: 0,
|
PreConsumedQuota: 0,
|
||||||
ModelRatio: '',
|
ModelRatio: '',
|
||||||
ModelPrice: '',
|
ModelPrice: '',
|
||||||
GroupRatio: '',
|
GroupRatio: '',
|
||||||
TopUpLink: '',
|
TopUpLink: '',
|
||||||
ChatLink: '',
|
ChatLink: '',
|
||||||
ChatLink2: '', // 添加的新状态变量
|
ChatLink2: '', // 添加的新状态变量
|
||||||
QuotaPerUnit: 0,
|
QuotaPerUnit: 0,
|
||||||
AutomaticDisableChannelEnabled: '',
|
AutomaticDisableChannelEnabled: '',
|
||||||
AutomaticEnableChannelEnabled: '',
|
AutomaticEnableChannelEnabled: '',
|
||||||
ChannelDisableThreshold: 0,
|
ChannelDisableThreshold: 0,
|
||||||
LogConsumeEnabled: '',
|
LogConsumeEnabled: '',
|
||||||
DisplayInCurrencyEnabled: '',
|
DisplayInCurrencyEnabled: '',
|
||||||
DisplayTokenStatEnabled: '',
|
DisplayTokenStatEnabled: '',
|
||||||
DrawingEnabled: '',
|
MjNotifyEnabled: '',
|
||||||
DataExportEnabled: '',
|
DrawingEnabled: '',
|
||||||
DataExportDefaultTime: 'hour',
|
DataExportEnabled: '',
|
||||||
DataExportInterval: 5,
|
DataExportDefaultTime: 'hour',
|
||||||
DefaultCollapseSidebar: '', // 默认折叠侧边栏
|
DataExportInterval: 5,
|
||||||
RetryTimes: 0
|
DefaultCollapseSidebar: '', // 默认折叠侧边栏
|
||||||
|
RetryTimes: 0
|
||||||
|
});
|
||||||
|
const [originInputs, setOriginInputs] = useState({});
|
||||||
|
let [loading, setLoading] = useState(false);
|
||||||
|
let [historyTimestamp, setHistoryTimestamp] = useState(timestamp2string(now.getTime() / 1000 - 30 * 24 * 3600)); // a month ago
|
||||||
|
// 精确时间选项(小时,天,周)
|
||||||
|
const timeOptions = [
|
||||||
|
{ key: 'hour', text: '小时', value: 'hour' },
|
||||||
|
{ key: 'day', text: '天', value: 'day' },
|
||||||
|
{ key: 'week', text: '周', value: 'week' }
|
||||||
|
];
|
||||||
|
const getOptions = async () => {
|
||||||
|
const res = await API.get('/api/option/');
|
||||||
|
const { success, message, data } = res.data;
|
||||||
|
if (success) {
|
||||||
|
let newInputs = {};
|
||||||
|
data.forEach((item) => {
|
||||||
|
if (item.key === 'ModelRatio' || item.key === 'GroupRatio' || item.key === 'ModelPrice') {
|
||||||
|
item.value = JSON.stringify(JSON.parse(item.value), null, 2);
|
||||||
|
}
|
||||||
|
newInputs[item.key] = item.value;
|
||||||
|
});
|
||||||
|
setInputs(newInputs);
|
||||||
|
setOriginInputs(newInputs);
|
||||||
|
} else {
|
||||||
|
showError(message);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
getOptions().then();
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const updateOption = async (key, value) => {
|
||||||
|
setLoading(true);
|
||||||
|
if (key.endsWith('Enabled')) {
|
||||||
|
value = inputs[key] === 'true' ? 'false' : 'true';
|
||||||
|
}
|
||||||
|
if (key === 'DefaultCollapseSidebar') {
|
||||||
|
value = inputs[key] === 'true' ? 'false' : 'true';
|
||||||
|
}
|
||||||
|
console.log(key, value);
|
||||||
|
const res = await API.put('/api/option/', {
|
||||||
|
key,
|
||||||
|
value
|
||||||
});
|
});
|
||||||
const [originInputs, setOriginInputs] = useState({});
|
const { success, message } = res.data;
|
||||||
let [loading, setLoading] = useState(false);
|
if (success) {
|
||||||
let [historyTimestamp, setHistoryTimestamp] = useState(timestamp2string(now.getTime() / 1000 - 30 * 24 * 3600)); // a month ago
|
setInputs((inputs) => ({ ...inputs, [key]: value }));
|
||||||
// 精确时间选项(小时,天,周)
|
} else {
|
||||||
const timeOptions = [
|
showError(message);
|
||||||
{key: 'hour', text: '小时', value: 'hour'},
|
}
|
||||||
{key: 'day', text: '天', value: 'day'},
|
setLoading(false);
|
||||||
{key: 'week', text: '周', value: 'week'}
|
};
|
||||||
];
|
|
||||||
const getOptions = async () => {
|
|
||||||
const res = await API.get('/api/option/');
|
|
||||||
const {success, message, data} = res.data;
|
|
||||||
if (success) {
|
|
||||||
let newInputs = {};
|
|
||||||
data.forEach((item) => {
|
|
||||||
if (item.key === 'ModelRatio' || item.key === 'GroupRatio' || item.key === 'ModelPrice') {
|
|
||||||
item.value = JSON.stringify(JSON.parse(item.value), null, 2);
|
|
||||||
}
|
|
||||||
newInputs[item.key] = item.value;
|
|
||||||
});
|
|
||||||
setInputs(newInputs);
|
|
||||||
setOriginInputs(newInputs);
|
|
||||||
} else {
|
|
||||||
showError(message);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
useEffect(() => {
|
const handleInputChange = async (e, { name, value }) => {
|
||||||
getOptions().then();
|
if (name.endsWith('Enabled') || name === 'DataExportInterval' || name === 'DataExportDefaultTime' || name === 'DefaultCollapseSidebar') {
|
||||||
}, []);
|
if (name === 'DataExportDefaultTime') {
|
||||||
|
localStorage.setItem('data_export_default_time', value);
|
||||||
|
} else if (name === 'MjNotifyEnabled') {
|
||||||
|
localStorage.setItem('mj_notify_enabled', value);
|
||||||
|
}
|
||||||
|
await updateOption(name, value);
|
||||||
|
} else {
|
||||||
|
setInputs((inputs) => ({ ...inputs, [name]: value }));
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
const updateOption = async (key, value) => {
|
const submitConfig = async (group) => {
|
||||||
setLoading(true);
|
switch (group) {
|
||||||
if (key.endsWith('Enabled')) {
|
case 'monitor':
|
||||||
value = inputs[key] === 'true' ? 'false' : 'true';
|
if (originInputs['ChannelDisableThreshold'] !== inputs.ChannelDisableThreshold) {
|
||||||
|
await updateOption('ChannelDisableThreshold', inputs.ChannelDisableThreshold);
|
||||||
}
|
}
|
||||||
if (key === 'DefaultCollapseSidebar') {
|
if (originInputs['QuotaRemindThreshold'] !== inputs.QuotaRemindThreshold) {
|
||||||
value = inputs[key] === 'true' ? 'false' : 'true';
|
await updateOption('QuotaRemindThreshold', inputs.QuotaRemindThreshold);
|
||||||
}
|
}
|
||||||
console.log(key, value)
|
break;
|
||||||
const res = await API.put('/api/option/', {
|
case 'ratio':
|
||||||
key,
|
if (originInputs['ModelRatio'] !== inputs.ModelRatio) {
|
||||||
value
|
if (!verifyJSON(inputs.ModelRatio)) {
|
||||||
});
|
showError('模型倍率不是合法的 JSON 字符串');
|
||||||
const {success, message} = res.data;
|
|
||||||
if (success) {
|
|
||||||
setInputs((inputs) => ({...inputs, [key]: value}));
|
|
||||||
} else {
|
|
||||||
showError(message);
|
|
||||||
}
|
|
||||||
setLoading(false);
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleInputChange = async (e, {name, value}) => {
|
|
||||||
if (name.endsWith('Enabled') || name === 'DataExportInterval' || name === 'DataExportDefaultTime' || name === 'DefaultCollapseSidebar') {
|
|
||||||
if (name === 'DataExportDefaultTime') {
|
|
||||||
localStorage.setItem('data_export_default_time', value);
|
|
||||||
}
|
|
||||||
await updateOption(name, value);
|
|
||||||
} else {
|
|
||||||
setInputs((inputs) => ({...inputs, [name]: value}));
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const submitConfig = async (group) => {
|
|
||||||
switch (group) {
|
|
||||||
case 'monitor':
|
|
||||||
if (originInputs['ChannelDisableThreshold'] !== inputs.ChannelDisableThreshold) {
|
|
||||||
await updateOption('ChannelDisableThreshold', inputs.ChannelDisableThreshold);
|
|
||||||
}
|
|
||||||
if (originInputs['QuotaRemindThreshold'] !== inputs.QuotaRemindThreshold) {
|
|
||||||
await updateOption('QuotaRemindThreshold', inputs.QuotaRemindThreshold);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case 'ratio':
|
|
||||||
if (originInputs['ModelRatio'] !== inputs.ModelRatio) {
|
|
||||||
if (!verifyJSON(inputs.ModelRatio)) {
|
|
||||||
showError('模型倍率不是合法的 JSON 字符串');
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
await updateOption('ModelRatio', inputs.ModelRatio);
|
|
||||||
}
|
|
||||||
if (originInputs['GroupRatio'] !== inputs.GroupRatio) {
|
|
||||||
if (!verifyJSON(inputs.GroupRatio)) {
|
|
||||||
showError('分组倍率不是合法的 JSON 字符串');
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
await updateOption('GroupRatio', inputs.GroupRatio);
|
|
||||||
}
|
|
||||||
if (originInputs['ModelPrice'] !== inputs.ModelPrice) {
|
|
||||||
if (!verifyJSON(inputs.ModelPrice)) {
|
|
||||||
showError('模型固定价格不是合法的 JSON 字符串');
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
await updateOption('ModelPrice', inputs.ModelPrice);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case 'quota':
|
|
||||||
if (originInputs['QuotaForNewUser'] !== inputs.QuotaForNewUser) {
|
|
||||||
await updateOption('QuotaForNewUser', inputs.QuotaForNewUser);
|
|
||||||
}
|
|
||||||
if (originInputs['QuotaForInvitee'] !== inputs.QuotaForInvitee) {
|
|
||||||
await updateOption('QuotaForInvitee', inputs.QuotaForInvitee);
|
|
||||||
}
|
|
||||||
if (originInputs['QuotaForInviter'] !== inputs.QuotaForInviter) {
|
|
||||||
await updateOption('QuotaForInviter', inputs.QuotaForInviter);
|
|
||||||
}
|
|
||||||
if (originInputs['PreConsumedQuota'] !== inputs.PreConsumedQuota) {
|
|
||||||
await updateOption('PreConsumedQuota', inputs.PreConsumedQuota);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case 'general':
|
|
||||||
if (originInputs['TopUpLink'] !== inputs.TopUpLink) {
|
|
||||||
await updateOption('TopUpLink', inputs.TopUpLink);
|
|
||||||
}
|
|
||||||
if (originInputs['ChatLink'] !== inputs.ChatLink) {
|
|
||||||
await updateOption('ChatLink', inputs.ChatLink);
|
|
||||||
}
|
|
||||||
if (originInputs['ChatLink2'] !== inputs.ChatLink2) {
|
|
||||||
await updateOption('ChatLink2', inputs.ChatLink2);
|
|
||||||
}
|
|
||||||
if (originInputs['QuotaPerUnit'] !== inputs.QuotaPerUnit) {
|
|
||||||
await updateOption('QuotaPerUnit', inputs.QuotaPerUnit);
|
|
||||||
}
|
|
||||||
if (originInputs['RetryTimes'] !== inputs.RetryTimes) {
|
|
||||||
await updateOption('RetryTimes', inputs.RetryTimes);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const deleteHistoryLogs = async () => {
|
|
||||||
console.log(inputs);
|
|
||||||
const res = await API.delete(`/api/log/?target_timestamp=${Date.parse(historyTimestamp) / 1000}`);
|
|
||||||
const {success, message, data} = res.data;
|
|
||||||
if (success) {
|
|
||||||
showSuccess(`${data} 条日志已清理!`);
|
|
||||||
return;
|
return;
|
||||||
|
}
|
||||||
|
await updateOption('ModelRatio', inputs.ModelRatio);
|
||||||
}
|
}
|
||||||
showError('日志清理失败:' + message);
|
if (originInputs['GroupRatio'] !== inputs.GroupRatio) {
|
||||||
};
|
if (!verifyJSON(inputs.GroupRatio)) {
|
||||||
return (
|
showError('分组倍率不是合法的 JSON 字符串');
|
||||||
<Grid columns={1}>
|
return;
|
||||||
<Grid.Column>
|
}
|
||||||
<Form loading={loading}>
|
await updateOption('GroupRatio', inputs.GroupRatio);
|
||||||
<Header as='h3'>
|
}
|
||||||
通用设置
|
if (originInputs['ModelPrice'] !== inputs.ModelPrice) {
|
||||||
</Header>
|
if (!verifyJSON(inputs.ModelPrice)) {
|
||||||
<Form.Group widths={4}>
|
showError('模型固定价格不是合法的 JSON 字符串');
|
||||||
<Form.Input
|
return;
|
||||||
label='充值链接'
|
}
|
||||||
name='TopUpLink'
|
await updateOption('ModelPrice', inputs.ModelPrice);
|
||||||
onChange={handleInputChange}
|
}
|
||||||
autoComplete='new-password'
|
break;
|
||||||
value={inputs.TopUpLink}
|
case 'quota':
|
||||||
type='link'
|
if (originInputs['QuotaForNewUser'] !== inputs.QuotaForNewUser) {
|
||||||
placeholder='例如发卡网站的购买链接'
|
await updateOption('QuotaForNewUser', inputs.QuotaForNewUser);
|
||||||
/>
|
}
|
||||||
<Form.Input
|
if (originInputs['QuotaForInvitee'] !== inputs.QuotaForInvitee) {
|
||||||
label='默认聊天页面链接'
|
await updateOption('QuotaForInvitee', inputs.QuotaForInvitee);
|
||||||
name='ChatLink'
|
}
|
||||||
onChange={handleInputChange}
|
if (originInputs['QuotaForInviter'] !== inputs.QuotaForInviter) {
|
||||||
autoComplete='new-password'
|
await updateOption('QuotaForInviter', inputs.QuotaForInviter);
|
||||||
value={inputs.ChatLink}
|
}
|
||||||
type='link'
|
if (originInputs['PreConsumedQuota'] !== inputs.PreConsumedQuota) {
|
||||||
placeholder='例如 ChatGPT Next Web 的部署地址'
|
await updateOption('PreConsumedQuota', inputs.PreConsumedQuota);
|
||||||
/>
|
}
|
||||||
<Form.Input
|
break;
|
||||||
label='聊天页面2链接'
|
case 'general':
|
||||||
name='ChatLink2'
|
if (originInputs['TopUpLink'] !== inputs.TopUpLink) {
|
||||||
onChange={handleInputChange}
|
await updateOption('TopUpLink', inputs.TopUpLink);
|
||||||
autoComplete='new-password'
|
}
|
||||||
value={inputs.ChatLink2}
|
if (originInputs['ChatLink'] !== inputs.ChatLink) {
|
||||||
type='link'
|
await updateOption('ChatLink', inputs.ChatLink);
|
||||||
placeholder='例如 ChatGPT Web & Midjourney 的部署地址'
|
}
|
||||||
/>
|
if (originInputs['ChatLink2'] !== inputs.ChatLink2) {
|
||||||
<Form.Input
|
await updateOption('ChatLink2', inputs.ChatLink2);
|
||||||
label='单位美元额度'
|
}
|
||||||
name='QuotaPerUnit'
|
if (originInputs['QuotaPerUnit'] !== inputs.QuotaPerUnit) {
|
||||||
onChange={handleInputChange}
|
await updateOption('QuotaPerUnit', inputs.QuotaPerUnit);
|
||||||
autoComplete='new-password'
|
}
|
||||||
value={inputs.QuotaPerUnit}
|
if (originInputs['RetryTimes'] !== inputs.RetryTimes) {
|
||||||
type='number'
|
await updateOption('RetryTimes', inputs.RetryTimes);
|
||||||
step='0.01'
|
}
|
||||||
placeholder='一单位货币能兑换的额度'
|
break;
|
||||||
/>
|
}
|
||||||
<Form.Input
|
};
|
||||||
label='失败重试次数'
|
|
||||||
name='RetryTimes'
|
|
||||||
type={'number'}
|
|
||||||
step='1'
|
|
||||||
min='0'
|
|
||||||
onChange={handleInputChange}
|
|
||||||
autoComplete='new-password'
|
|
||||||
value={inputs.RetryTimes}
|
|
||||||
placeholder='失败重试次数'
|
|
||||||
/>
|
|
||||||
</Form.Group>
|
|
||||||
<Form.Group inline>
|
|
||||||
|
|
||||||
<Form.Checkbox
|
const deleteHistoryLogs = async () => {
|
||||||
checked={inputs.DisplayInCurrencyEnabled === 'true'}
|
console.log(inputs);
|
||||||
label='以货币形式显示额度'
|
const res = await API.delete(`/api/log/?target_timestamp=${Date.parse(historyTimestamp) / 1000}`);
|
||||||
name='DisplayInCurrencyEnabled'
|
const { success, message, data } = res.data;
|
||||||
onChange={handleInputChange}
|
if (success) {
|
||||||
/>
|
showSuccess(`${data} 条日志已清理!`);
|
||||||
<Form.Checkbox
|
return;
|
||||||
checked={inputs.DisplayTokenStatEnabled === 'true'}
|
}
|
||||||
label='Billing 相关 API 显示令牌额度而非用户额度'
|
showError('日志清理失败:' + message);
|
||||||
name='DisplayTokenStatEnabled'
|
};
|
||||||
onChange={handleInputChange}
|
return (
|
||||||
/>
|
<Grid columns={1}>
|
||||||
<Form.Checkbox
|
<Grid.Column>
|
||||||
checked={inputs.DrawingEnabled === 'true'}
|
<Form loading={loading}>
|
||||||
label='启用绘图功能'
|
<Header as="h3">
|
||||||
name='DrawingEnabled'
|
通用设置
|
||||||
onChange={handleInputChange}
|
</Header>
|
||||||
/>
|
<Form.Group widths={4}>
|
||||||
<Form.Checkbox
|
<Form.Input
|
||||||
checked={inputs.DefaultCollapseSidebar === 'true'}
|
label="充值链接"
|
||||||
label='默认折叠侧边栏'
|
name="TopUpLink"
|
||||||
name='DefaultCollapseSidebar'
|
onChange={handleInputChange}
|
||||||
onChange={handleInputChange}
|
autoComplete="new-password"
|
||||||
/>
|
value={inputs.TopUpLink}
|
||||||
</Form.Group>
|
type="link"
|
||||||
<Form.Button onClick={() => {
|
placeholder="例如发卡网站的购买链接"
|
||||||
submitConfig('general').then();
|
/>
|
||||||
}}>保存通用设置</Form.Button><Divider/>
|
<Form.Input
|
||||||
<Header as='h3'>
|
label="默认聊天页面链接"
|
||||||
日志设置
|
name="ChatLink"
|
||||||
</Header>
|
onChange={handleInputChange}
|
||||||
<Form.Group inline>
|
autoComplete="new-password"
|
||||||
<Form.Checkbox
|
value={inputs.ChatLink}
|
||||||
checked={inputs.LogConsumeEnabled === 'true'}
|
type="link"
|
||||||
label='启用额度消费日志记录'
|
placeholder="例如 ChatGPT Next Web 的部署地址"
|
||||||
name='LogConsumeEnabled'
|
/>
|
||||||
onChange={handleInputChange}
|
<Form.Input
|
||||||
/>
|
label="聊天页面2链接"
|
||||||
</Form.Group>
|
name="ChatLink2"
|
||||||
<Form.Group widths={4}>
|
onChange={handleInputChange}
|
||||||
<Form.Input label='目标时间' value={historyTimestamp} type='datetime-local'
|
autoComplete="new-password"
|
||||||
name='history_timestamp'
|
value={inputs.ChatLink2}
|
||||||
onChange={(e, {name, value}) => {
|
type="link"
|
||||||
setHistoryTimestamp(value);
|
placeholder="例如 ChatGPT Web & Midjourney 的部署地址"
|
||||||
}}/>
|
/>
|
||||||
</Form.Group>
|
<Form.Input
|
||||||
<Form.Button onClick={() => {
|
label="单位美元额度"
|
||||||
deleteHistoryLogs().then();
|
name="QuotaPerUnit"
|
||||||
}}>清理历史日志</Form.Button>
|
onChange={handleInputChange}
|
||||||
<Divider/>
|
autoComplete="new-password"
|
||||||
<Header as='h3'>
|
value={inputs.QuotaPerUnit}
|
||||||
数据看板
|
type="number"
|
||||||
</Header>
|
step="0.01"
|
||||||
<Form.Checkbox
|
placeholder="一单位货币能兑换的额度"
|
||||||
checked={inputs.DataExportEnabled === 'true'}
|
/>
|
||||||
label='启用数据看板(实验性)'
|
<Form.Input
|
||||||
name='DataExportEnabled'
|
label="失败重试次数"
|
||||||
onChange={handleInputChange}
|
name="RetryTimes"
|
||||||
/>
|
type={'number'}
|
||||||
<Form.Group>
|
step="1"
|
||||||
<Form.Input
|
min="0"
|
||||||
label='数据看板更新间隔(分钟,设置过短会影响数据库性能)'
|
onChange={handleInputChange}
|
||||||
name='DataExportInterval'
|
autoComplete="new-password"
|
||||||
type={'number'}
|
value={inputs.RetryTimes}
|
||||||
step='1'
|
placeholder="失败重试次数"
|
||||||
min='1'
|
/>
|
||||||
onChange={handleInputChange}
|
</Form.Group>
|
||||||
autoComplete='new-password'
|
<Form.Group inline>
|
||||||
value={inputs.DataExportInterval}
|
<Form.Checkbox
|
||||||
placeholder='数据看板更新间隔(分钟,设置过短会影响数据库性能)'
|
checked={inputs.DisplayInCurrencyEnabled === 'true'}
|
||||||
/>
|
label="以货币形式显示额度"
|
||||||
<Form.Select
|
name="DisplayInCurrencyEnabled"
|
||||||
label='数据看板默认时间粒度(仅修改展示粒度,统计精确到小时)'
|
onChange={handleInputChange}
|
||||||
options={timeOptions}
|
/>
|
||||||
name='DataExportDefaultTime'
|
<Form.Checkbox
|
||||||
onChange={handleInputChange}
|
checked={inputs.DisplayTokenStatEnabled === 'true'}
|
||||||
autoComplete='new-password'
|
label="Billing 相关 API 显示令牌额度而非用户额度"
|
||||||
value={inputs.DataExportDefaultTime}
|
name="DisplayTokenStatEnabled"
|
||||||
placeholder='数据看板默认时间粒度'
|
onChange={handleInputChange}
|
||||||
/>
|
/>
|
||||||
</Form.Group>
|
<Form.Checkbox
|
||||||
<Divider/>
|
checked={inputs.DefaultCollapseSidebar === 'true'}
|
||||||
<Header as='h3'>
|
label="默认折叠侧边栏"
|
||||||
监控设置
|
name="DefaultCollapseSidebar"
|
||||||
</Header>
|
onChange={handleInputChange}
|
||||||
<Form.Group widths={3}>
|
/>
|
||||||
<Form.Input
|
</Form.Group>
|
||||||
label='最长响应时间'
|
<Form.Button onClick={() => {
|
||||||
name='ChannelDisableThreshold'
|
submitConfig('general').then();
|
||||||
onChange={handleInputChange}
|
}}>保存通用设置</Form.Button>
|
||||||
autoComplete='new-password'
|
<Divider />
|
||||||
value={inputs.ChannelDisableThreshold}
|
<Header as="h3">
|
||||||
type='number'
|
绘图设置
|
||||||
min='0'
|
</Header>
|
||||||
placeholder='单位秒,当运行通道全部测试时,超过此时间将自动禁用通道'
|
<Form.Group inline>
|
||||||
/>
|
<Form.Checkbox
|
||||||
<Form.Input
|
checked={inputs.DrawingEnabled === 'true'}
|
||||||
label='额度提醒阈值'
|
label="启用绘图功能"
|
||||||
name='QuotaRemindThreshold'
|
name="DrawingEnabled"
|
||||||
onChange={handleInputChange}
|
onChange={handleInputChange}
|
||||||
autoComplete='new-password'
|
/>
|
||||||
value={inputs.QuotaRemindThreshold}
|
<Form.Checkbox
|
||||||
type='number'
|
checked={inputs.MjNotifyEnabled === 'true'}
|
||||||
min='0'
|
label="允许回调(会泄露服务器ip地址)"
|
||||||
placeholder='低于此额度时将发送邮件提醒用户'
|
name="MjNotifyEnabled"
|
||||||
/>
|
onChange={handleInputChange}
|
||||||
</Form.Group>
|
/>
|
||||||
<Form.Group inline>
|
</Form.Group>
|
||||||
<Form.Checkbox
|
<Divider />
|
||||||
checked={inputs.AutomaticDisableChannelEnabled === 'true'}
|
<Header as="h3">
|
||||||
label='失败时自动禁用通道'
|
日志设置
|
||||||
name='AutomaticDisableChannelEnabled'
|
</Header>
|
||||||
onChange={handleInputChange}
|
<Form.Group inline>
|
||||||
/>
|
<Form.Checkbox
|
||||||
<Form.Checkbox
|
checked={inputs.LogConsumeEnabled === 'true'}
|
||||||
checked={inputs.AutomaticEnableChannelEnabled === 'true'}
|
label="启用额度消费日志记录"
|
||||||
label='成功时自动启用通道'
|
name="LogConsumeEnabled"
|
||||||
name='AutomaticEnableChannelEnabled'
|
onChange={handleInputChange}
|
||||||
onChange={handleInputChange}
|
/>
|
||||||
/>
|
</Form.Group>
|
||||||
</Form.Group>
|
<Form.Group widths={4}>
|
||||||
<Form.Button onClick={() => {
|
<Form.Input label="目标时间" value={historyTimestamp} type="datetime-local"
|
||||||
submitConfig('monitor').then();
|
name="history_timestamp"
|
||||||
}}>保存监控设置</Form.Button>
|
onChange={(e, { name, value }) => {
|
||||||
<Divider/>
|
setHistoryTimestamp(value);
|
||||||
<Header as='h3'>
|
}} />
|
||||||
额度设置
|
</Form.Group>
|
||||||
</Header>
|
<Form.Button onClick={() => {
|
||||||
<Form.Group widths={4}>
|
deleteHistoryLogs().then();
|
||||||
<Form.Input
|
}}>清理历史日志</Form.Button>
|
||||||
label='新用户初始额度'
|
<Divider />
|
||||||
name='QuotaForNewUser'
|
<Header as="h3">
|
||||||
onChange={handleInputChange}
|
数据看板
|
||||||
autoComplete='new-password'
|
</Header>
|
||||||
value={inputs.QuotaForNewUser}
|
<Form.Checkbox
|
||||||
type='number'
|
checked={inputs.DataExportEnabled === 'true'}
|
||||||
min='0'
|
label="启用数据看板(实验性)"
|
||||||
placeholder='例如:100'
|
name="DataExportEnabled"
|
||||||
/>
|
onChange={handleInputChange}
|
||||||
<Form.Input
|
/>
|
||||||
label='请求预扣费额度'
|
<Form.Group>
|
||||||
name='PreConsumedQuota'
|
<Form.Input
|
||||||
onChange={handleInputChange}
|
label="数据看板更新间隔(分钟,设置过短会影响数据库性能)"
|
||||||
autoComplete='new-password'
|
name="DataExportInterval"
|
||||||
value={inputs.PreConsumedQuota}
|
type={'number'}
|
||||||
type='number'
|
step="1"
|
||||||
min='0'
|
min="1"
|
||||||
placeholder='请求结束后多退少补'
|
onChange={handleInputChange}
|
||||||
/>
|
autoComplete="new-password"
|
||||||
<Form.Input
|
value={inputs.DataExportInterval}
|
||||||
label='邀请新用户奖励额度'
|
placeholder="数据看板更新间隔(分钟,设置过短会影响数据库性能)"
|
||||||
name='QuotaForInviter'
|
/>
|
||||||
onChange={handleInputChange}
|
<Form.Select
|
||||||
autoComplete='new-password'
|
label="数据看板默认时间粒度(仅修改展示粒度,统计精确到小时)"
|
||||||
value={inputs.QuotaForInviter}
|
options={timeOptions}
|
||||||
type='number'
|
name="DataExportDefaultTime"
|
||||||
min='0'
|
onChange={handleInputChange}
|
||||||
placeholder='例如:2000'
|
autoComplete="new-password"
|
||||||
/>
|
value={inputs.DataExportDefaultTime}
|
||||||
<Form.Input
|
placeholder="数据看板默认时间粒度"
|
||||||
label='新用户使用邀请码奖励额度'
|
/>
|
||||||
name='QuotaForInvitee'
|
</Form.Group>
|
||||||
onChange={handleInputChange}
|
<Divider />
|
||||||
autoComplete='new-password'
|
<Header as="h3">
|
||||||
value={inputs.QuotaForInvitee}
|
监控设置
|
||||||
type='number'
|
</Header>
|
||||||
min='0'
|
<Form.Group widths={3}>
|
||||||
placeholder='例如:1000'
|
<Form.Input
|
||||||
/>
|
label="最长响应时间"
|
||||||
</Form.Group>
|
name="ChannelDisableThreshold"
|
||||||
<Form.Button onClick={() => {
|
onChange={handleInputChange}
|
||||||
submitConfig('quota').then();
|
autoComplete="new-password"
|
||||||
}}>保存额度设置</Form.Button>
|
value={inputs.ChannelDisableThreshold}
|
||||||
<Divider/>
|
type="number"
|
||||||
<Header as='h3'>
|
min="0"
|
||||||
倍率设置
|
placeholder="单位秒,当运行通道全部测试时,超过此时间将自动禁用通道"
|
||||||
</Header>
|
/>
|
||||||
<Form.Group widths='equal'>
|
<Form.Input
|
||||||
<Form.TextArea
|
label="额度提醒阈值"
|
||||||
label='模型固定价格(一次调用消耗多少刀,优先级大于模型倍率)'
|
name="QuotaRemindThreshold"
|
||||||
name='ModelPrice'
|
onChange={handleInputChange}
|
||||||
onChange={handleInputChange}
|
autoComplete="new-password"
|
||||||
style={{minHeight: 250, fontFamily: 'JetBrains Mono, Consolas'}}
|
value={inputs.QuotaRemindThreshold}
|
||||||
autoComplete='new-password'
|
type="number"
|
||||||
value={inputs.ModelPrice}
|
min="0"
|
||||||
placeholder='为一个 JSON 文本,键为模型名称,值为一次调用消耗多少刀,比如 "gpt-4-gizmo-*": 0.1,一次消耗0.1刀'
|
placeholder="低于此额度时将发送邮件提醒用户"
|
||||||
/>
|
/>
|
||||||
</Form.Group>
|
</Form.Group>
|
||||||
<Form.Group widths='equal'>
|
<Form.Group inline>
|
||||||
<Form.TextArea
|
<Form.Checkbox
|
||||||
label='模型倍率'
|
checked={inputs.AutomaticDisableChannelEnabled === 'true'}
|
||||||
name='ModelRatio'
|
label="失败时自动禁用通道"
|
||||||
onChange={handleInputChange}
|
name="AutomaticDisableChannelEnabled"
|
||||||
style={{minHeight: 250, fontFamily: 'JetBrains Mono, Consolas'}}
|
onChange={handleInputChange}
|
||||||
autoComplete='new-password'
|
/>
|
||||||
value={inputs.ModelRatio}
|
<Form.Checkbox
|
||||||
placeholder='为一个 JSON 文本,键为模型名称,值为倍率'
|
checked={inputs.AutomaticEnableChannelEnabled === 'true'}
|
||||||
/>
|
label="成功时自动启用通道"
|
||||||
</Form.Group>
|
name="AutomaticEnableChannelEnabled"
|
||||||
<Form.Group widths='equal'>
|
onChange={handleInputChange}
|
||||||
<Form.TextArea
|
/>
|
||||||
label='分组倍率'
|
</Form.Group>
|
||||||
name='GroupRatio'
|
<Form.Button onClick={() => {
|
||||||
onChange={handleInputChange}
|
submitConfig('monitor').then();
|
||||||
style={{minHeight: 250, fontFamily: 'JetBrains Mono, Consolas'}}
|
}}>保存监控设置</Form.Button>
|
||||||
autoComplete='new-password'
|
<Divider />
|
||||||
value={inputs.GroupRatio}
|
<Header as="h3">
|
||||||
placeholder='为一个 JSON 文本,键为分组名称,值为倍率'
|
额度设置
|
||||||
/>
|
</Header>
|
||||||
</Form.Group>
|
<Form.Group widths={4}>
|
||||||
<Form.Button onClick={() => {
|
<Form.Input
|
||||||
submitConfig('ratio').then();
|
label="新用户初始额度"
|
||||||
}}>保存倍率设置</Form.Button>
|
name="QuotaForNewUser"
|
||||||
</Form>
|
onChange={handleInputChange}
|
||||||
</Grid.Column>
|
autoComplete="new-password"
|
||||||
</Grid>
|
value={inputs.QuotaForNewUser}
|
||||||
)
|
type="number"
|
||||||
;
|
min="0"
|
||||||
|
placeholder="例如:100"
|
||||||
|
/>
|
||||||
|
<Form.Input
|
||||||
|
label="请求预扣费额度"
|
||||||
|
name="PreConsumedQuota"
|
||||||
|
onChange={handleInputChange}
|
||||||
|
autoComplete="new-password"
|
||||||
|
value={inputs.PreConsumedQuota}
|
||||||
|
type="number"
|
||||||
|
min="0"
|
||||||
|
placeholder="请求结束后多退少补"
|
||||||
|
/>
|
||||||
|
<Form.Input
|
||||||
|
label="邀请新用户奖励额度"
|
||||||
|
name="QuotaForInviter"
|
||||||
|
onChange={handleInputChange}
|
||||||
|
autoComplete="new-password"
|
||||||
|
value={inputs.QuotaForInviter}
|
||||||
|
type="number"
|
||||||
|
min="0"
|
||||||
|
placeholder="例如:2000"
|
||||||
|
/>
|
||||||
|
<Form.Input
|
||||||
|
label="新用户使用邀请码奖励额度"
|
||||||
|
name="QuotaForInvitee"
|
||||||
|
onChange={handleInputChange}
|
||||||
|
autoComplete="new-password"
|
||||||
|
value={inputs.QuotaForInvitee}
|
||||||
|
type="number"
|
||||||
|
min="0"
|
||||||
|
placeholder="例如:1000"
|
||||||
|
/>
|
||||||
|
</Form.Group>
|
||||||
|
<Form.Button onClick={() => {
|
||||||
|
submitConfig('quota').then();
|
||||||
|
}}>保存额度设置</Form.Button>
|
||||||
|
<Divider />
|
||||||
|
<Header as="h3">
|
||||||
|
倍率设置
|
||||||
|
</Header>
|
||||||
|
<Form.Group widths="equal">
|
||||||
|
<Form.TextArea
|
||||||
|
label="模型固定价格(一次调用消耗多少刀,优先级大于模型倍率)"
|
||||||
|
name="ModelPrice"
|
||||||
|
onChange={handleInputChange}
|
||||||
|
style={{ minHeight: 250, fontFamily: 'JetBrains Mono, Consolas' }}
|
||||||
|
autoComplete="new-password"
|
||||||
|
value={inputs.ModelPrice}
|
||||||
|
placeholder='为一个 JSON 文本,键为模型名称,值为一次调用消耗多少刀,比如 "gpt-4-gizmo-*": 0.1,一次消耗0.1刀'
|
||||||
|
/>
|
||||||
|
</Form.Group>
|
||||||
|
<Form.Group widths="equal">
|
||||||
|
<Form.TextArea
|
||||||
|
label="模型倍率"
|
||||||
|
name="ModelRatio"
|
||||||
|
onChange={handleInputChange}
|
||||||
|
style={{ minHeight: 250, fontFamily: 'JetBrains Mono, Consolas' }}
|
||||||
|
autoComplete="new-password"
|
||||||
|
value={inputs.ModelRatio}
|
||||||
|
placeholder="为一个 JSON 文本,键为模型名称,值为倍率"
|
||||||
|
/>
|
||||||
|
</Form.Group>
|
||||||
|
<Form.Group widths="equal">
|
||||||
|
<Form.TextArea
|
||||||
|
label="分组倍率"
|
||||||
|
name="GroupRatio"
|
||||||
|
onChange={handleInputChange}
|
||||||
|
style={{ minHeight: 250, fontFamily: 'JetBrains Mono, Consolas' }}
|
||||||
|
autoComplete="new-password"
|
||||||
|
value={inputs.GroupRatio}
|
||||||
|
placeholder="为一个 JSON 文本,键为分组名称,值为倍率"
|
||||||
|
/>
|
||||||
|
</Form.Group>
|
||||||
|
<Form.Button onClick={() => {
|
||||||
|
submitConfig('ratio').then();
|
||||||
|
}}>保存倍率设置</Form.Button>
|
||||||
|
</Form>
|
||||||
|
</Grid.Column>
|
||||||
|
</Grid>
|
||||||
|
)
|
||||||
|
;
|
||||||
};
|
};
|
||||||
|
|
||||||
export default OperationSetting;
|
export default OperationSetting;
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
import React, { useEffect, useRef, useState } from 'react';
|
import React, { useEffect, useRef, useState } from 'react';
|
||||||
import { Col, Row , Form, Button, Banner } from '@douyinfe/semi-ui';
|
import { Banner, Button, Col, Form, Row } from '@douyinfe/semi-ui';
|
||||||
import { API, showError, showSuccess } from '../helpers';
|
import { API, showError, showSuccess } from '../helpers';
|
||||||
import { marked } from 'marked';
|
import { marked } from 'marked';
|
||||||
|
|
||||||
@ -57,8 +57,8 @@ const OtherSetting = () => {
|
|||||||
await updateOption('Notice', inputs.Notice);
|
await updateOption('Notice', inputs.Notice);
|
||||||
showSuccess('公告已更新');
|
showSuccess('公告已更新');
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error("公告更新失败", error);
|
console.error('公告更新失败', error);
|
||||||
showError("公告更新失败")
|
showError('公告更新失败');
|
||||||
} finally {
|
} finally {
|
||||||
setLoadingInput((loadingInput) => ({ ...loadingInput, Notice: false }));
|
setLoadingInput((loadingInput) => ({ ...loadingInput, Notice: false }));
|
||||||
}
|
}
|
||||||
@ -72,8 +72,8 @@ const OtherSetting = () => {
|
|||||||
await updateOption('SystemName', inputs.SystemName);
|
await updateOption('SystemName', inputs.SystemName);
|
||||||
showSuccess('系统名称已更新');
|
showSuccess('系统名称已更新');
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error("系统名称更新失败", error);
|
console.error('系统名称更新失败', error);
|
||||||
showError("系统名称更新失败")
|
showError('系统名称更新失败');
|
||||||
} finally {
|
} finally {
|
||||||
setLoadingInput((loadingInput) => ({ ...loadingInput, SystemName: false }));
|
setLoadingInput((loadingInput) => ({ ...loadingInput, SystemName: false }));
|
||||||
}
|
}
|
||||||
@ -86,8 +86,8 @@ const OtherSetting = () => {
|
|||||||
await updateOption('Logo', inputs.Logo);
|
await updateOption('Logo', inputs.Logo);
|
||||||
showSuccess('Logo 已更新');
|
showSuccess('Logo 已更新');
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error("Logo 更新失败", error);
|
console.error('Logo 更新失败', error);
|
||||||
showError("Logo 更新失败")
|
showError('Logo 更新失败');
|
||||||
} finally {
|
} finally {
|
||||||
setLoadingInput((loadingInput) => ({ ...loadingInput, Logo: false }));
|
setLoadingInput((loadingInput) => ({ ...loadingInput, Logo: false }));
|
||||||
}
|
}
|
||||||
@ -99,8 +99,8 @@ const OtherSetting = () => {
|
|||||||
await updateOption(key, inputs[key]);
|
await updateOption(key, inputs[key]);
|
||||||
showSuccess('首页内容已更新');
|
showSuccess('首页内容已更新');
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error("首页内容更新失败", error);
|
console.error('首页内容更新失败', error);
|
||||||
showError("首页内容更新失败")
|
showError('首页内容更新失败');
|
||||||
} finally {
|
} finally {
|
||||||
setLoadingInput((loadingInput) => ({ ...loadingInput, HomePageContent: false }));
|
setLoadingInput((loadingInput) => ({ ...loadingInput, HomePageContent: false }));
|
||||||
}
|
}
|
||||||
@ -112,8 +112,8 @@ const OtherSetting = () => {
|
|||||||
await updateOption('About', inputs.About);
|
await updateOption('About', inputs.About);
|
||||||
showSuccess('关于内容已更新');
|
showSuccess('关于内容已更新');
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error("关于内容更新失败", error);
|
console.error('关于内容更新失败', error);
|
||||||
showError("关于内容更新失败");
|
showError('关于内容更新失败');
|
||||||
} finally {
|
} finally {
|
||||||
setLoadingInput((loadingInput) => ({ ...loadingInput, About: false }));
|
setLoadingInput((loadingInput) => ({ ...loadingInput, About: false }));
|
||||||
}
|
}
|
||||||
@ -125,16 +125,14 @@ const OtherSetting = () => {
|
|||||||
await updateOption('Footer', inputs.Footer);
|
await updateOption('Footer', inputs.Footer);
|
||||||
showSuccess('页脚内容已更新');
|
showSuccess('页脚内容已更新');
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error("页脚内容更新失败", error);
|
console.error('页脚内容更新失败', error);
|
||||||
showError("页脚内容更新失败");
|
showError('页脚内容更新失败');
|
||||||
} finally {
|
} finally {
|
||||||
setLoadingInput((loadingInput) => ({ ...loadingInput, Footer: false }));
|
setLoadingInput((loadingInput) => ({ ...loadingInput, Footer: false }));
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
const openGitHubRelease = () => {
|
const openGitHubRelease = () => {
|
||||||
window.location =
|
window.location =
|
||||||
'https://github.com/songquanpeng/one-api/releases/latest';
|
'https://github.com/songquanpeng/one-api/releases/latest';
|
||||||
@ -173,16 +171,17 @@ const OtherSetting = () => {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
useEffect( () => {
|
useEffect(() => {
|
||||||
getOptions();
|
getOptions();
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Row >
|
<Row>
|
||||||
<Col span={24}>
|
<Col span={24}>
|
||||||
{/* 通用设置 */}
|
{/* 通用设置 */}
|
||||||
<Form values={inputs} getFormApi={formAPI => formAPISettingGeneral.current = formAPI} style={{marginBottom: 15}}>
|
<Form values={inputs} getFormApi={formAPI => formAPISettingGeneral.current = formAPI}
|
||||||
|
style={{ marginBottom: 15 }}>
|
||||||
<Form.Section text={'通用设置'}>
|
<Form.Section text={'通用设置'}>
|
||||||
<Form.TextArea
|
<Form.TextArea
|
||||||
label={'公告'}
|
label={'公告'}
|
||||||
@ -191,26 +190,27 @@ const OtherSetting = () => {
|
|||||||
onChange={handleInputChange}
|
onChange={handleInputChange}
|
||||||
style={{ fontFamily: 'JetBrains Mono, Consolas' }}
|
style={{ fontFamily: 'JetBrains Mono, Consolas' }}
|
||||||
autosize={{ minRows: 6, maxRows: 12 }}
|
autosize={{ minRows: 6, maxRows: 12 }}
|
||||||
/>
|
/>
|
||||||
<Button onClick={submitNotice} loading={loadingInput['Notice']}>设置公告</Button>
|
<Button onClick={submitNotice} loading={loadingInput['Notice']}>设置公告</Button>
|
||||||
</Form.Section>
|
</Form.Section>
|
||||||
</Form>
|
</Form>
|
||||||
{/* 个性化设置 */}
|
{/* 个性化设置 */}
|
||||||
<Form values={inputs} getFormApi={formAPI => formAPIPersonalization.current = formAPI} style={{marginBottom: 15}}>
|
<Form values={inputs} getFormApi={formAPI => formAPIPersonalization.current = formAPI}
|
||||||
|
style={{ marginBottom: 15 }}>
|
||||||
<Form.Section text={'个性化设置'}>
|
<Form.Section text={'个性化设置'}>
|
||||||
<Form.Input
|
<Form.Input
|
||||||
label={'系统名称'}
|
label={'系统名称'}
|
||||||
placeholder={'在此输入系统名称'}
|
placeholder={'在此输入系统名称'}
|
||||||
field={'SystemName'}
|
field={'SystemName'}
|
||||||
onChange={handleInputChange}
|
onChange={handleInputChange}
|
||||||
/>
|
/>
|
||||||
<Button onClick={submitSystemName} loading={loadingInput['SystemName']}>设置系统名称</Button>
|
<Button onClick={submitSystemName} loading={loadingInput['SystemName']}>设置系统名称</Button>
|
||||||
<Form.Input
|
<Form.Input
|
||||||
label={'Logo 图片地址'}
|
label={'Logo 图片地址'}
|
||||||
placeholder={'在此输入 Logo 图片地址'}
|
placeholder={'在此输入 Logo 图片地址'}
|
||||||
field={'Logo'}
|
field={'Logo'}
|
||||||
onChange={handleInputChange}
|
onChange={handleInputChange}
|
||||||
/>
|
/>
|
||||||
<Button onClick={submitLogo} loading={loadingInput['Logo']}>设置 Logo</Button>
|
<Button onClick={submitLogo} loading={loadingInput['Logo']}>设置 Logo</Button>
|
||||||
<Form.TextArea
|
<Form.TextArea
|
||||||
label={'首页内容'}
|
label={'首页内容'}
|
||||||
@ -219,8 +219,9 @@ const OtherSetting = () => {
|
|||||||
onChange={handleInputChange}
|
onChange={handleInputChange}
|
||||||
style={{ fontFamily: 'JetBrains Mono, Consolas' }}
|
style={{ fontFamily: 'JetBrains Mono, Consolas' }}
|
||||||
autosize={{ minRows: 6, maxRows: 12 }}
|
autosize={{ minRows: 6, maxRows: 12 }}
|
||||||
/>
|
/>
|
||||||
<Button onClick={() => submitOption('HomePageContent')} loading={loadingInput['HomePageContent']}>设置首页内容</Button>
|
<Button onClick={() => submitOption('HomePageContent')}
|
||||||
|
loading={loadingInput['HomePageContent']}>设置首页内容</Button>
|
||||||
<Form.TextArea
|
<Form.TextArea
|
||||||
label={'关于'}
|
label={'关于'}
|
||||||
placeholder={'在此输入新的关于内容,支持 Markdown & HTML 代码。如果输入的是一个链接,则会使用该链接作为 iframe 的 src 属性,这允许你设置任意网页作为关于页面。'}
|
placeholder={'在此输入新的关于内容,支持 Markdown & HTML 代码。如果输入的是一个链接,则会使用该链接作为 iframe 的 src 属性,这允许你设置任意网页作为关于页面。'}
|
||||||
@ -228,7 +229,7 @@ const OtherSetting = () => {
|
|||||||
onChange={handleInputChange}
|
onChange={handleInputChange}
|
||||||
style={{ fontFamily: 'JetBrains Mono, Consolas' }}
|
style={{ fontFamily: 'JetBrains Mono, Consolas' }}
|
||||||
autosize={{ minRows: 6, maxRows: 12 }}
|
autosize={{ minRows: 6, maxRows: 12 }}
|
||||||
/>
|
/>
|
||||||
<Button onClick={submitAbout} loading={loadingInput['About']}>设置关于</Button>
|
<Button onClick={submitAbout} loading={loadingInput['About']}>设置关于</Button>
|
||||||
{/* */}
|
{/* */}
|
||||||
<Banner
|
<Banner
|
||||||
@ -236,14 +237,14 @@ const OtherSetting = () => {
|
|||||||
type="info"
|
type="info"
|
||||||
description="移除 One API 的版权标识必须首先获得授权,项目维护需要花费大量精力,如果本项目对你有意义,请主动支持本项目。"
|
description="移除 One API 的版权标识必须首先获得授权,项目维护需要花费大量精力,如果本项目对你有意义,请主动支持本项目。"
|
||||||
closeIcon={null}
|
closeIcon={null}
|
||||||
style={{ marginTop: 15 }}
|
style={{ marginTop: 15 }}
|
||||||
/>
|
/>
|
||||||
<Form.Input
|
<Form.Input
|
||||||
label={'页脚'}
|
label={'页脚'}
|
||||||
placeholder={'在此输入新的页脚,留空则使用默认页脚,支持 HTML 代码'}
|
placeholder={'在此输入新的页脚,留空则使用默认页脚,支持 HTML 代码'}
|
||||||
field={'Footer'}
|
field={'Footer'}
|
||||||
onChange={handleInputChange}
|
onChange={handleInputChange}
|
||||||
/>
|
/>
|
||||||
<Button onClick={submitFooter} loading={loadingInput['Footer']}>设置页脚</Button>
|
<Button onClick={submitFooter} loading={loadingInput['Footer']}>设置页脚</Button>
|
||||||
</Form.Section>
|
</Form.Section>
|
||||||
</Form>
|
</Form>
|
||||||
@ -270,7 +271,7 @@ const OtherSetting = () => {
|
|||||||
{/* />*/}
|
{/* />*/}
|
||||||
{/* </Modal.Actions>*/}
|
{/* </Modal.Actions>*/}
|
||||||
{/*</Modal>*/}
|
{/*</Modal>*/}
|
||||||
</Row>
|
</Row>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -1,12 +1,12 @@
|
|||||||
import React, { useEffect, useState } from 'react';
|
import React, { useEffect, useState } from 'react';
|
||||||
import { Button, Form, Grid, Header, Image, Segment } from 'semantic-ui-react';
|
import { Button, Form, Grid, Header, Image, Segment } from 'semantic-ui-react';
|
||||||
import { API, copy, showError, showInfo, showNotice, showSuccess } from '../helpers';
|
import { API, copy, showError, showNotice } from '../helpers';
|
||||||
import { useSearchParams } from 'react-router-dom';
|
import { useSearchParams } from 'react-router-dom';
|
||||||
|
|
||||||
const PasswordResetConfirm = () => {
|
const PasswordResetConfirm = () => {
|
||||||
const [inputs, setInputs] = useState({
|
const [inputs, setInputs] = useState({
|
||||||
email: '',
|
email: '',
|
||||||
token: '',
|
token: ''
|
||||||
});
|
});
|
||||||
const { email, token } = inputs;
|
const { email, token } = inputs;
|
||||||
|
|
||||||
@ -23,7 +23,7 @@ const PasswordResetConfirm = () => {
|
|||||||
let email = searchParams.get('email');
|
let email = searchParams.get('email');
|
||||||
setInputs({
|
setInputs({
|
||||||
token,
|
token,
|
||||||
email,
|
email
|
||||||
});
|
});
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
@ -37,7 +37,7 @@ const PasswordResetConfirm = () => {
|
|||||||
setDisableButton(false);
|
setDisableButton(false);
|
||||||
setCountdown(30);
|
setCountdown(30);
|
||||||
}
|
}
|
||||||
return () => clearInterval(countdownInterval);
|
return () => clearInterval(countdownInterval);
|
||||||
}, [disableButton, countdown]);
|
}, [disableButton, countdown]);
|
||||||
|
|
||||||
async function handleSubmit(e) {
|
async function handleSubmit(e) {
|
||||||
@ -46,7 +46,7 @@ const PasswordResetConfirm = () => {
|
|||||||
setLoading(true);
|
setLoading(true);
|
||||||
const res = await API.post(`/api/user/reset`, {
|
const res = await API.post(`/api/user/reset`, {
|
||||||
email,
|
email,
|
||||||
token,
|
token
|
||||||
});
|
});
|
||||||
const { success, message } = res.data;
|
const { success, message } = res.data;
|
||||||
if (success) {
|
if (success) {
|
||||||
@ -59,44 +59,44 @@ const PasswordResetConfirm = () => {
|
|||||||
}
|
}
|
||||||
setLoading(false);
|
setLoading(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Grid textAlign='center' style={{ marginTop: '48px' }}>
|
<Grid textAlign="center" style={{ marginTop: '48px' }}>
|
||||||
<Grid.Column style={{ maxWidth: 450 }}>
|
<Grid.Column style={{ maxWidth: 450 }}>
|
||||||
<Header as='h2' color='' textAlign='center'>
|
<Header as="h2" color="" textAlign="center">
|
||||||
<Image src='/logo.png' /> 密码重置确认
|
<Image src="/logo.png" /> 密码重置确认
|
||||||
</Header>
|
</Header>
|
||||||
<Form size='large'>
|
<Form size="large">
|
||||||
<Segment>
|
<Segment>
|
||||||
<Form.Input
|
<Form.Input
|
||||||
fluid
|
fluid
|
||||||
icon='mail'
|
icon="mail"
|
||||||
iconPosition='left'
|
iconPosition="left"
|
||||||
placeholder='邮箱地址'
|
placeholder="邮箱地址"
|
||||||
name='email'
|
name="email"
|
||||||
value={email}
|
value={email}
|
||||||
readOnly
|
readOnly
|
||||||
/>
|
/>
|
||||||
{newPassword && (
|
{newPassword && (
|
||||||
<Form.Input
|
<Form.Input
|
||||||
fluid
|
fluid
|
||||||
icon='lock'
|
icon="lock"
|
||||||
iconPosition='left'
|
iconPosition="left"
|
||||||
placeholder='新密码'
|
placeholder="新密码"
|
||||||
name='newPassword'
|
name="newPassword"
|
||||||
value={newPassword}
|
value={newPassword}
|
||||||
readOnly
|
readOnly
|
||||||
onClick={(e) => {
|
onClick={(e) => {
|
||||||
e.target.select();
|
e.target.select();
|
||||||
navigator.clipboard.writeText(newPassword);
|
navigator.clipboard.writeText(newPassword);
|
||||||
showNotice(`密码已复制到剪贴板:${newPassword}`);
|
showNotice(`密码已复制到剪贴板:${newPassword}`);
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
<Button
|
<Button
|
||||||
color='green'
|
color="green"
|
||||||
fluid
|
fluid
|
||||||
size='large'
|
size="large"
|
||||||
onClick={handleSubmit}
|
onClick={handleSubmit}
|
||||||
loading={loading}
|
loading={loading}
|
||||||
disabled={disableButton}
|
disabled={disableButton}
|
||||||
@ -107,7 +107,7 @@ const PasswordResetConfirm = () => {
|
|||||||
</Form>
|
</Form>
|
||||||
</Grid.Column>
|
</Grid.Column>
|
||||||
</Grid>
|
</Grid>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export default PasswordResetConfirm;
|
export default PasswordResetConfirm;
|
||||||
|
@ -56,19 +56,19 @@ const PasswordResetForm = () => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Grid textAlign='center' style={{ marginTop: '48px' }}>
|
<Grid textAlign="center" style={{ marginTop: '48px' }}>
|
||||||
<Grid.Column style={{ maxWidth: 450 }}>
|
<Grid.Column style={{ maxWidth: 450 }}>
|
||||||
<Header as='h2' color='' textAlign='center'>
|
<Header as="h2" color="" textAlign="center">
|
||||||
<Image src='/logo.png' /> 密码重置
|
<Image src="/logo.png" /> 密码重置
|
||||||
</Header>
|
</Header>
|
||||||
<Form size='large'>
|
<Form size="large">
|
||||||
<Segment>
|
<Segment>
|
||||||
<Form.Input
|
<Form.Input
|
||||||
fluid
|
fluid
|
||||||
icon='mail'
|
icon="mail"
|
||||||
iconPosition='left'
|
iconPosition="left"
|
||||||
placeholder='邮箱地址'
|
placeholder="邮箱地址"
|
||||||
name='email'
|
name="email"
|
||||||
value={email}
|
value={email}
|
||||||
onChange={handleChange}
|
onChange={handleChange}
|
||||||
/>
|
/>
|
||||||
@ -83,9 +83,9 @@ const PasswordResetForm = () => {
|
|||||||
<></>
|
<></>
|
||||||
)}
|
)}
|
||||||
<Button
|
<Button
|
||||||
color='green'
|
color="green"
|
||||||
fluid
|
fluid
|
||||||
size='large'
|
size="large"
|
||||||
onClick={handleSubmit}
|
onClick={handleSubmit}
|
||||||
loading={loading}
|
loading={loading}
|
||||||
disabled={disableButton}
|
disabled={disableButton}
|
||||||
|
File diff suppressed because it is too large
Load Diff
@ -5,7 +5,7 @@ import { history } from '../helpers';
|
|||||||
|
|
||||||
function PrivateRoute({ children }) {
|
function PrivateRoute({ children }) {
|
||||||
if (!localStorage.getItem('user')) {
|
if (!localStorage.getItem('user')) {
|
||||||
return <Navigate to='/login' state={{ from: history.location }} />;
|
return <Navigate to="/login" state={{ from: history.location }} />;
|
||||||
}
|
}
|
||||||
return children;
|
return children;
|
||||||
}
|
}
|
||||||
|
@ -1,406 +1,406 @@
|
|||||||
import React, {useEffect, useState} from 'react';
|
import React, { useEffect, useState } from 'react';
|
||||||
import {API, copy, showError, showInfo, showSuccess, showWarning, timestamp2string} from '../helpers';
|
import { API, copy, showError, showSuccess, timestamp2string } from '../helpers';
|
||||||
|
|
||||||
import {ITEMS_PER_PAGE} from '../constants';
|
import { ITEMS_PER_PAGE } from '../constants';
|
||||||
import {renderQuota} from '../helpers/render';
|
import { renderQuota } from '../helpers/render';
|
||||||
import {Button, Modal, Popconfirm, Popover, Table, Tag, Form} from "@douyinfe/semi-ui";
|
import { Button, Form, Modal, Popconfirm, Popover, Table, Tag } from '@douyinfe/semi-ui';
|
||||||
import EditRedemption from "../pages/Redemption/EditRedemption";
|
import EditRedemption from '../pages/Redemption/EditRedemption';
|
||||||
|
|
||||||
function renderTimestamp(timestamp) {
|
function renderTimestamp(timestamp) {
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
{timestamp2string(timestamp)}
|
{timestamp2string(timestamp)}
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
function renderStatus(status) {
|
function renderStatus(status) {
|
||||||
switch (status) {
|
switch (status) {
|
||||||
case 1:
|
case 1:
|
||||||
return <Tag color='green' size='large'>未使用</Tag>;
|
return <Tag color="green" size="large">未使用</Tag>;
|
||||||
case 2:
|
case 2:
|
||||||
return <Tag color='red' size='large'> 已禁用 </Tag>;
|
return <Tag color="red" size="large"> 已禁用 </Tag>;
|
||||||
case 3:
|
case 3:
|
||||||
return <Tag color='grey' size='large'> 已使用 </Tag>;
|
return <Tag color="grey" size="large"> 已使用 </Tag>;
|
||||||
default:
|
default:
|
||||||
return <Tag color='black' size='large'> 未知状态 </Tag>;
|
return <Tag color="black" size="large"> 未知状态 </Tag>;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const RedemptionsTable = () => {
|
const RedemptionsTable = () => {
|
||||||
const columns = [
|
const columns = [
|
||||||
{
|
{
|
||||||
title: 'ID',
|
title: 'ID',
|
||||||
dataIndex: 'id',
|
dataIndex: 'id'
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: '名称',
|
title: '名称',
|
||||||
dataIndex: 'name',
|
dataIndex: 'name'
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: '状态',
|
title: '状态',
|
||||||
dataIndex: 'status',
|
dataIndex: 'status',
|
||||||
key: 'status',
|
key: 'status',
|
||||||
render: (text, record, index) => {
|
render: (text, record, index) => {
|
||||||
return (
|
return (
|
||||||
<div>
|
<div>
|
||||||
{renderStatus(text)}
|
{renderStatus(text)}
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
},
|
}
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: '额度',
|
title: '额度',
|
||||||
dataIndex: 'quota',
|
dataIndex: 'quota',
|
||||||
render: (text, record, index) => {
|
render: (text, record, index) => {
|
||||||
return (
|
return (
|
||||||
<div>
|
<div>
|
||||||
{renderQuota(parseInt(text))}
|
{renderQuota(parseInt(text))}
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
},
|
}
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: '创建时间',
|
title: '创建时间',
|
||||||
dataIndex: 'created_time',
|
dataIndex: 'created_time',
|
||||||
render: (text, record, index) => {
|
render: (text, record, index) => {
|
||||||
return (
|
return (
|
||||||
<div>
|
<div>
|
||||||
{renderTimestamp(text)}
|
{renderTimestamp(text)}
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
},
|
}
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: '兑换人ID',
|
title: '兑换人ID',
|
||||||
dataIndex: 'used_user_id',
|
dataIndex: 'used_user_id',
|
||||||
render: (text, record, index) => {
|
render: (text, record, index) => {
|
||||||
return (
|
return (
|
||||||
<div>
|
<div>
|
||||||
{text === 0 ? '无' : text}
|
{text === 0 ? '无' : text}
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
},
|
}
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: '',
|
title: '',
|
||||||
dataIndex: 'operate',
|
dataIndex: 'operate',
|
||||||
render: (text, record, index) => (
|
render: (text, record, index) => (
|
||||||
<div>
|
<div>
|
||||||
<Popover
|
<Popover
|
||||||
content={
|
content={
|
||||||
record.key
|
record.key
|
||||||
}
|
|
||||||
style={{padding: 20}}
|
|
||||||
position="top"
|
|
||||||
>
|
|
||||||
<Button theme='light' type='tertiary' style={{marginRight: 1}}>查看</Button>
|
|
||||||
</Popover>
|
|
||||||
<Button theme='light' type='secondary' style={{marginRight: 1}}
|
|
||||||
onClick={async (text) => {
|
|
||||||
await copyText(record.key)
|
|
||||||
}}
|
|
||||||
>复制</Button>
|
|
||||||
<Popconfirm
|
|
||||||
title="确定是否要删除此兑换码?"
|
|
||||||
content="此修改将不可逆"
|
|
||||||
okType={'danger'}
|
|
||||||
position={'left'}
|
|
||||||
onConfirm={() => {
|
|
||||||
manageRedemption(record.id, 'delete', record).then(
|
|
||||||
() => {
|
|
||||||
removeRecord(record.key);
|
|
||||||
}
|
|
||||||
)
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<Button theme='light' type='danger' style={{marginRight: 1}}>删除</Button>
|
|
||||||
</Popconfirm>
|
|
||||||
{
|
|
||||||
record.status === 1 ?
|
|
||||||
<Button theme='light' type='warning' style={{marginRight: 1}} onClick={
|
|
||||||
async () => {
|
|
||||||
manageRedemption(
|
|
||||||
record.id,
|
|
||||||
'disable',
|
|
||||||
record
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}>禁用</Button> :
|
|
||||||
<Button theme='light' type='secondary' style={{marginRight: 1}} onClick={
|
|
||||||
async () => {
|
|
||||||
manageRedemption(
|
|
||||||
record.id,
|
|
||||||
'enable',
|
|
||||||
record
|
|
||||||
);
|
|
||||||
}
|
|
||||||
} disabled={record.status === 3}>启用</Button>
|
|
||||||
}
|
|
||||||
<Button theme='light' type='tertiary' style={{marginRight: 1}} onClick={
|
|
||||||
() => {
|
|
||||||
setEditingRedemption(record);
|
|
||||||
setShowEdit(true);
|
|
||||||
}
|
|
||||||
} disabled={record.status !== 1}>编辑</Button>
|
|
||||||
</div>
|
|
||||||
),
|
|
||||||
},
|
|
||||||
];
|
|
||||||
|
|
||||||
const [redemptions, setRedemptions] = useState([]);
|
|
||||||
const [loading, setLoading] = useState(true);
|
|
||||||
const [activePage, setActivePage] = useState(1);
|
|
||||||
const [searchKeyword, setSearchKeyword] = useState('');
|
|
||||||
const [searching, setSearching] = useState(false);
|
|
||||||
const [tokenCount, setTokenCount] = useState(ITEMS_PER_PAGE);
|
|
||||||
const [selectedKeys, setSelectedKeys] = useState([]);
|
|
||||||
const [editingRedemption, setEditingRedemption] = useState({
|
|
||||||
id: undefined,
|
|
||||||
});
|
|
||||||
const [showEdit, setShowEdit] = useState(false);
|
|
||||||
|
|
||||||
const closeEdit = () => {
|
|
||||||
setShowEdit(false);
|
|
||||||
}
|
|
||||||
|
|
||||||
// const setCount = (data) => {
|
|
||||||
// if (data.length >= (activePage) * ITEMS_PER_PAGE) {
|
|
||||||
// setTokenCount(data.length + 1);
|
|
||||||
// } else {
|
|
||||||
// setTokenCount(data.length);
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
|
|
||||||
const setRedemptionFormat = (redeptions) => {
|
|
||||||
// for (let i = 0; i < redeptions.length; i++) {
|
|
||||||
// redeptions[i].key = '' + redeptions[i].id;
|
|
||||||
// }
|
|
||||||
// data.key = '' + data.id
|
|
||||||
setRedemptions(redeptions);
|
|
||||||
if (redeptions.length >= (activePage) * ITEMS_PER_PAGE) {
|
|
||||||
setTokenCount(redeptions.length + 1);
|
|
||||||
} else {
|
|
||||||
setTokenCount(redeptions.length);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const loadRedemptions = async (startIdx) => {
|
|
||||||
const res = await API.get(`/api/redemption/?p=${startIdx}`);
|
|
||||||
const {success, message, data} = res.data;
|
|
||||||
if (success) {
|
|
||||||
if (startIdx === 0) {
|
|
||||||
setRedemptionFormat(data);
|
|
||||||
} else {
|
|
||||||
let newRedemptions = redemptions;
|
|
||||||
newRedemptions.push(...data);
|
|
||||||
setRedemptionFormat(newRedemptions);
|
|
||||||
}
|
}
|
||||||
} else {
|
style={{ padding: 20 }}
|
||||||
showError(message);
|
position="top"
|
||||||
}
|
>
|
||||||
setLoading(false);
|
<Button theme="light" type="tertiary" style={{ marginRight: 1 }}>查看</Button>
|
||||||
};
|
</Popover>
|
||||||
|
<Button theme="light" type="secondary" style={{ marginRight: 1 }}
|
||||||
const removeRecord = key => {
|
onClick={async (text) => {
|
||||||
let newDataSource = [...redemptions];
|
await copyText(record.key);
|
||||||
if (key != null) {
|
}}
|
||||||
let idx = newDataSource.findIndex(data => data.key === key);
|
>复制</Button>
|
||||||
|
<Popconfirm
|
||||||
if (idx > -1) {
|
title="确定是否要删除此兑换码?"
|
||||||
newDataSource.splice(idx, 1);
|
content="此修改将不可逆"
|
||||||
setRedemptions(newDataSource);
|
okType={'danger'}
|
||||||
}
|
position={'left'}
|
||||||
}
|
onConfirm={() => {
|
||||||
};
|
manageRedemption(record.id, 'delete', record).then(
|
||||||
|
|
||||||
const copyText = async (text) => {
|
|
||||||
if (await copy(text)) {
|
|
||||||
showSuccess('已复制到剪贴板!');
|
|
||||||
} else {
|
|
||||||
// setSearchKeyword(text);
|
|
||||||
Modal.error({title: '无法复制到剪贴板,请手动复制', content: text});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
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 refresh = async () => {
|
|
||||||
await loadRedemptions(activePage - 1);
|
|
||||||
};
|
|
||||||
|
|
||||||
const manageRedemption = async (id, action, record) => {
|
|
||||||
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') {
|
|
||||||
|
|
||||||
} else {
|
|
||||||
record.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 (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);
|
|
||||||
};
|
|
||||||
|
|
||||||
const handlePageChange = page => {
|
|
||||||
setActivePage(page);
|
|
||||||
if (page === Math.ceil(redemptions.length / ITEMS_PER_PAGE) + 1) {
|
|
||||||
// In this case we have to load more data and then append them.
|
|
||||||
loadRedemptions(page - 1).then(r => {
|
|
||||||
});
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
let pageData = redemptions.slice((activePage - 1) * ITEMS_PER_PAGE, activePage * ITEMS_PER_PAGE);
|
|
||||||
const rowSelection = {
|
|
||||||
onSelect: (record, selected) => {
|
|
||||||
},
|
|
||||||
onSelectAll: (selected, selectedRows) => {
|
|
||||||
},
|
|
||||||
onChange: (selectedRowKeys, selectedRows) => {
|
|
||||||
setSelectedKeys(selectedRows);
|
|
||||||
},
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleRow = (record, index) => {
|
|
||||||
if (record.status !== 1) {
|
|
||||||
return {
|
|
||||||
style: {
|
|
||||||
background: 'var(--semi-color-disabled-border)',
|
|
||||||
},
|
|
||||||
};
|
|
||||||
} else {
|
|
||||||
return {};
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
|
||||||
<>
|
|
||||||
<EditRedemption refresh={refresh} editingRedemption={editingRedemption} visiable={showEdit}
|
|
||||||
handleClose={closeEdit}></EditRedemption>
|
|
||||||
<Form onSubmit={searchRedemptions}>
|
|
||||||
<Form.Input
|
|
||||||
label='搜索关键字'
|
|
||||||
field='keyword'
|
|
||||||
icon='search'
|
|
||||||
iconPosition='left'
|
|
||||||
placeholder='关键字(id或者名称)'
|
|
||||||
value={searchKeyword}
|
|
||||||
loading={searching}
|
|
||||||
onChange={handleKeywordChange}
|
|
||||||
/>
|
|
||||||
</Form>
|
|
||||||
|
|
||||||
<Table style={{marginTop: 20}} columns={columns} dataSource={pageData} pagination={{
|
|
||||||
currentPage: activePage,
|
|
||||||
pageSize: ITEMS_PER_PAGE,
|
|
||||||
total: tokenCount,
|
|
||||||
// showSizeChanger: true,
|
|
||||||
// pageSizeOptions: [10, 20, 50, 100],
|
|
||||||
formatPageText: (page) => `第 ${page.currentStart} - ${page.currentEnd} 条,共 ${redemptions.length} 条`,
|
|
||||||
// onPageSizeChange: (size) => {
|
|
||||||
// setPageSize(size);
|
|
||||||
// setActivePage(1);
|
|
||||||
// },
|
|
||||||
onPageChange: handlePageChange,
|
|
||||||
}} loading={loading} rowSelection={rowSelection} onRow={handleRow}>
|
|
||||||
</Table>
|
|
||||||
<Button theme='light' type='primary' style={{marginRight: 8}} onClick={
|
|
||||||
() => {
|
() => {
|
||||||
setEditingRedemption({
|
removeRecord(record.key);
|
||||||
id: undefined,
|
|
||||||
});
|
|
||||||
setShowEdit(true);
|
|
||||||
}
|
}
|
||||||
}>添加兑换码</Button>
|
);
|
||||||
<Button label='复制所选兑换码' type="warning" onClick={
|
}}
|
||||||
|
>
|
||||||
|
<Button theme="light" type="danger" style={{ marginRight: 1 }}>删除</Button>
|
||||||
|
</Popconfirm>
|
||||||
|
{
|
||||||
|
record.status === 1 ?
|
||||||
|
<Button theme="light" type="warning" style={{ marginRight: 1 }} onClick={
|
||||||
async () => {
|
async () => {
|
||||||
if (selectedKeys.length === 0) {
|
manageRedemption(
|
||||||
showError('请至少选择一个兑换码!');
|
record.id,
|
||||||
return;
|
'disable',
|
||||||
}
|
record
|
||||||
let keys = "";
|
);
|
||||||
for (let i = 0; i < selectedKeys.length; i++) {
|
|
||||||
keys += selectedKeys[i].name + " " + selectedKeys[i].key + "\n";
|
|
||||||
}
|
|
||||||
await copyText(keys);
|
|
||||||
}
|
}
|
||||||
}>复制所选兑换码到剪贴板</Button>
|
}>禁用</Button> :
|
||||||
</>
|
<Button theme="light" type="secondary" style={{ marginRight: 1 }} onClick={
|
||||||
);
|
async () => {
|
||||||
|
manageRedemption(
|
||||||
|
record.id,
|
||||||
|
'enable',
|
||||||
|
record
|
||||||
|
);
|
||||||
|
}
|
||||||
|
} disabled={record.status === 3}>启用</Button>
|
||||||
|
}
|
||||||
|
<Button theme="light" type="tertiary" style={{ marginRight: 1 }} onClick={
|
||||||
|
() => {
|
||||||
|
setEditingRedemption(record);
|
||||||
|
setShowEdit(true);
|
||||||
|
}
|
||||||
|
} disabled={record.status !== 1}>编辑</Button>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
];
|
||||||
|
|
||||||
|
const [redemptions, setRedemptions] = useState([]);
|
||||||
|
const [loading, setLoading] = useState(true);
|
||||||
|
const [activePage, setActivePage] = useState(1);
|
||||||
|
const [searchKeyword, setSearchKeyword] = useState('');
|
||||||
|
const [searching, setSearching] = useState(false);
|
||||||
|
const [tokenCount, setTokenCount] = useState(ITEMS_PER_PAGE);
|
||||||
|
const [selectedKeys, setSelectedKeys] = useState([]);
|
||||||
|
const [editingRedemption, setEditingRedemption] = useState({
|
||||||
|
id: undefined
|
||||||
|
});
|
||||||
|
const [showEdit, setShowEdit] = useState(false);
|
||||||
|
|
||||||
|
const closeEdit = () => {
|
||||||
|
setShowEdit(false);
|
||||||
|
};
|
||||||
|
|
||||||
|
// const setCount = (data) => {
|
||||||
|
// if (data.length >= (activePage) * ITEMS_PER_PAGE) {
|
||||||
|
// setTokenCount(data.length + 1);
|
||||||
|
// } else {
|
||||||
|
// setTokenCount(data.length);
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
|
||||||
|
const setRedemptionFormat = (redeptions) => {
|
||||||
|
// for (let i = 0; i < redeptions.length; i++) {
|
||||||
|
// redeptions[i].key = '' + redeptions[i].id;
|
||||||
|
// }
|
||||||
|
// data.key = '' + data.id
|
||||||
|
setRedemptions(redeptions);
|
||||||
|
if (redeptions.length >= (activePage) * ITEMS_PER_PAGE) {
|
||||||
|
setTokenCount(redeptions.length + 1);
|
||||||
|
} else {
|
||||||
|
setTokenCount(redeptions.length);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const loadRedemptions = async (startIdx) => {
|
||||||
|
const res = await API.get(`/api/redemption/?p=${startIdx}`);
|
||||||
|
const { success, message, data } = res.data;
|
||||||
|
if (success) {
|
||||||
|
if (startIdx === 0) {
|
||||||
|
setRedemptionFormat(data);
|
||||||
|
} else {
|
||||||
|
let newRedemptions = redemptions;
|
||||||
|
newRedemptions.push(...data);
|
||||||
|
setRedemptionFormat(newRedemptions);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
showError(message);
|
||||||
|
}
|
||||||
|
setLoading(false);
|
||||||
|
};
|
||||||
|
|
||||||
|
const removeRecord = key => {
|
||||||
|
let newDataSource = [...redemptions];
|
||||||
|
if (key != null) {
|
||||||
|
let idx = newDataSource.findIndex(data => data.key === key);
|
||||||
|
|
||||||
|
if (idx > -1) {
|
||||||
|
newDataSource.splice(idx, 1);
|
||||||
|
setRedemptions(newDataSource);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const copyText = async (text) => {
|
||||||
|
if (await copy(text)) {
|
||||||
|
showSuccess('已复制到剪贴板!');
|
||||||
|
} else {
|
||||||
|
// setSearchKeyword(text);
|
||||||
|
Modal.error({ title: '无法复制到剪贴板,请手动复制', content: text });
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
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 refresh = async () => {
|
||||||
|
await loadRedemptions(activePage - 1);
|
||||||
|
};
|
||||||
|
|
||||||
|
const manageRedemption = async (id, action, record) => {
|
||||||
|
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') {
|
||||||
|
|
||||||
|
} else {
|
||||||
|
record.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 (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);
|
||||||
|
};
|
||||||
|
|
||||||
|
const handlePageChange = page => {
|
||||||
|
setActivePage(page);
|
||||||
|
if (page === Math.ceil(redemptions.length / ITEMS_PER_PAGE) + 1) {
|
||||||
|
// In this case we have to load more data and then append them.
|
||||||
|
loadRedemptions(page - 1).then(r => {
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
let pageData = redemptions.slice((activePage - 1) * ITEMS_PER_PAGE, activePage * ITEMS_PER_PAGE);
|
||||||
|
const rowSelection = {
|
||||||
|
onSelect: (record, selected) => {
|
||||||
|
},
|
||||||
|
onSelectAll: (selected, selectedRows) => {
|
||||||
|
},
|
||||||
|
onChange: (selectedRowKeys, selectedRows) => {
|
||||||
|
setSelectedKeys(selectedRows);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleRow = (record, index) => {
|
||||||
|
if (record.status !== 1) {
|
||||||
|
return {
|
||||||
|
style: {
|
||||||
|
background: 'var(--semi-color-disabled-border)'
|
||||||
|
}
|
||||||
|
};
|
||||||
|
} else {
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<EditRedemption refresh={refresh} editingRedemption={editingRedemption} visiable={showEdit}
|
||||||
|
handleClose={closeEdit}></EditRedemption>
|
||||||
|
<Form onSubmit={searchRedemptions}>
|
||||||
|
<Form.Input
|
||||||
|
label="搜索关键字"
|
||||||
|
field="keyword"
|
||||||
|
icon="search"
|
||||||
|
iconPosition="left"
|
||||||
|
placeholder="关键字(id或者名称)"
|
||||||
|
value={searchKeyword}
|
||||||
|
loading={searching}
|
||||||
|
onChange={handleKeywordChange}
|
||||||
|
/>
|
||||||
|
</Form>
|
||||||
|
|
||||||
|
<Table style={{ marginTop: 20 }} columns={columns} dataSource={pageData} pagination={{
|
||||||
|
currentPage: activePage,
|
||||||
|
pageSize: ITEMS_PER_PAGE,
|
||||||
|
total: tokenCount,
|
||||||
|
// showSizeChanger: true,
|
||||||
|
// pageSizeOptions: [10, 20, 50, 100],
|
||||||
|
formatPageText: (page) => `第 ${page.currentStart} - ${page.currentEnd} 条,共 ${redemptions.length} 条`,
|
||||||
|
// onPageSizeChange: (size) => {
|
||||||
|
// setPageSize(size);
|
||||||
|
// setActivePage(1);
|
||||||
|
// },
|
||||||
|
onPageChange: handlePageChange
|
||||||
|
}} loading={loading} rowSelection={rowSelection} onRow={handleRow}>
|
||||||
|
</Table>
|
||||||
|
<Button theme="light" type="primary" style={{ marginRight: 8 }} onClick={
|
||||||
|
() => {
|
||||||
|
setEditingRedemption({
|
||||||
|
id: undefined
|
||||||
|
});
|
||||||
|
setShowEdit(true);
|
||||||
|
}
|
||||||
|
}>添加兑换码</Button>
|
||||||
|
<Button label="复制所选兑换码" type="warning" onClick={
|
||||||
|
async () => {
|
||||||
|
if (selectedKeys.length === 0) {
|
||||||
|
showError('请至少选择一个兑换码!');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
let keys = '';
|
||||||
|
for (let i = 0; i < selectedKeys.length; i++) {
|
||||||
|
keys += selectedKeys[i].name + ' ' + selectedKeys[i].key + '\n';
|
||||||
|
}
|
||||||
|
await copyText(keys);
|
||||||
|
}
|
||||||
|
}>复制所选兑换码到剪贴板</Button>
|
||||||
|
</>
|
||||||
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export default RedemptionsTable;
|
export default RedemptionsTable;
|
||||||
|
@ -98,49 +98,49 @@ const RegisterForm = () => {
|
|||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Grid textAlign='center' style={{ marginTop: '48px' }}>
|
<Grid textAlign="center" style={{ marginTop: '48px' }}>
|
||||||
<Grid.Column style={{ maxWidth: 450 }}>
|
<Grid.Column style={{ maxWidth: 450 }}>
|
||||||
<Header as='h2' color='' textAlign='center'>
|
<Header as="h2" color="" textAlign="center">
|
||||||
<Image src={logo} /> 新用户注册
|
<Image src={logo} /> 新用户注册
|
||||||
</Header>
|
</Header>
|
||||||
<Form size='large'>
|
<Form size="large">
|
||||||
<Segment>
|
<Segment>
|
||||||
<Form.Input
|
<Form.Input
|
||||||
fluid
|
fluid
|
||||||
icon='user'
|
icon="user"
|
||||||
iconPosition='left'
|
iconPosition="left"
|
||||||
placeholder='输入用户名,最长 12 位'
|
placeholder="输入用户名,最长 12 位"
|
||||||
onChange={handleChange}
|
onChange={handleChange}
|
||||||
name='username'
|
name="username"
|
||||||
/>
|
/>
|
||||||
<Form.Input
|
<Form.Input
|
||||||
fluid
|
fluid
|
||||||
icon='lock'
|
icon="lock"
|
||||||
iconPosition='left'
|
iconPosition="left"
|
||||||
placeholder='输入密码,最短 8 位,最长 20 位'
|
placeholder="输入密码,最短 8 位,最长 20 位"
|
||||||
onChange={handleChange}
|
onChange={handleChange}
|
||||||
name='password'
|
name="password"
|
||||||
type='password'
|
type="password"
|
||||||
/>
|
/>
|
||||||
<Form.Input
|
<Form.Input
|
||||||
fluid
|
fluid
|
||||||
icon='lock'
|
icon="lock"
|
||||||
iconPosition='left'
|
iconPosition="left"
|
||||||
placeholder='输入密码,最短 8 位,最长 20 位'
|
placeholder="输入密码,最短 8 位,最长 20 位"
|
||||||
onChange={handleChange}
|
onChange={handleChange}
|
||||||
name='password2'
|
name="password2"
|
||||||
type='password'
|
type="password"
|
||||||
/>
|
/>
|
||||||
{showEmailVerification ? (
|
{showEmailVerification ? (
|
||||||
<>
|
<>
|
||||||
<Form.Input
|
<Form.Input
|
||||||
fluid
|
fluid
|
||||||
icon='mail'
|
icon="mail"
|
||||||
iconPosition='left'
|
iconPosition="left"
|
||||||
placeholder='输入邮箱地址'
|
placeholder="输入邮箱地址"
|
||||||
onChange={handleChange}
|
onChange={handleChange}
|
||||||
name='email'
|
name="email"
|
||||||
type='email'
|
type="email"
|
||||||
action={
|
action={
|
||||||
<Button onClick={sendVerificationCode} disabled={loading}>
|
<Button onClick={sendVerificationCode} disabled={loading}>
|
||||||
获取验证码
|
获取验证码
|
||||||
@ -149,11 +149,11 @@ const RegisterForm = () => {
|
|||||||
/>
|
/>
|
||||||
<Form.Input
|
<Form.Input
|
||||||
fluid
|
fluid
|
||||||
icon='lock'
|
icon="lock"
|
||||||
iconPosition='left'
|
iconPosition="left"
|
||||||
placeholder='输入验证码'
|
placeholder="输入验证码"
|
||||||
onChange={handleChange}
|
onChange={handleChange}
|
||||||
name='verification_code'
|
name="verification_code"
|
||||||
/>
|
/>
|
||||||
</>
|
</>
|
||||||
) : (
|
) : (
|
||||||
@ -170,9 +170,9 @@ const RegisterForm = () => {
|
|||||||
<></>
|
<></>
|
||||||
)}
|
)}
|
||||||
<Button
|
<Button
|
||||||
color='green'
|
color="green"
|
||||||
fluid
|
fluid
|
||||||
size='large'
|
size="large"
|
||||||
onClick={handleSubmit}
|
onClick={handleSubmit}
|
||||||
loading={loading}
|
loading={loading}
|
||||||
>
|
>
|
||||||
@ -182,7 +182,7 @@ const RegisterForm = () => {
|
|||||||
</Form>
|
</Form>
|
||||||
<Message>
|
<Message>
|
||||||
已有账户?
|
已有账户?
|
||||||
<Link to='/login' className='btn btn-link'>
|
<Link to="/login" className="btn btn-link">
|
||||||
点击登录
|
点击登录
|
||||||
</Link>
|
</Link>
|
||||||
</Message>
|
</Message>
|
||||||
|
@ -1,213 +1,220 @@
|
|||||||
import React, { useContext, useEffect, useLayoutEffect, useMemo, useState } from 'react';
|
import React, { useContext, useEffect, useMemo, useState } from 'react';
|
||||||
import {Link, useNavigate} from 'react-router-dom';
|
import { Link, useNavigate } from 'react-router-dom';
|
||||||
import {UserContext} from '../context/User';
|
import { UserContext } from '../context/User';
|
||||||
import { StatusContext } from '../context/Status';
|
import { StatusContext } from '../context/Status';
|
||||||
|
|
||||||
import { API, getLogo, getSystemName, isAdmin, isMobile, showError, showSuccess } from '../helpers';
|
import { API, getLogo, getSystemName, isAdmin, isMobile, showError } from '../helpers';
|
||||||
import '../index.css';
|
import '../index.css';
|
||||||
|
|
||||||
import {
|
import {
|
||||||
IconCalendarClock,
|
IconCalendarClock,
|
||||||
IconHistogram,
|
IconComment,
|
||||||
IconGift,
|
IconCreditCard,
|
||||||
IconKey,
|
IconGift,
|
||||||
IconUser,
|
IconHistogram,
|
||||||
IconLayers,
|
IconHome,
|
||||||
IconSetting,
|
IconImage,
|
||||||
IconCreditCard,
|
IconKey,
|
||||||
IconComment,
|
IconLayers,
|
||||||
IconHome,
|
IconSetting,
|
||||||
IconImage
|
IconUser
|
||||||
} from '@douyinfe/semi-icons';
|
} from '@douyinfe/semi-icons';
|
||||||
import {Nav, Avatar, Dropdown, Layout} from '@douyinfe/semi-ui';
|
import { Layout, Nav } from '@douyinfe/semi-ui';
|
||||||
|
|
||||||
// HeaderBar Buttons
|
// HeaderBar Buttons
|
||||||
|
|
||||||
const SiderBar = () => {
|
const SiderBar = () => {
|
||||||
const [userState, userDispatch] = useContext(UserContext);
|
const [userState, userDispatch] = useContext(UserContext);
|
||||||
const [statusState, statusDispatch] = useContext(StatusContext);
|
const [statusState, statusDispatch] = useContext(StatusContext);
|
||||||
const defaultIsCollapsed = isMobile() || localStorage.getItem('default_collapse_sidebar') === 'true';
|
const defaultIsCollapsed = isMobile() || localStorage.getItem('default_collapse_sidebar') === 'true';
|
||||||
|
|
||||||
let navigate = useNavigate();
|
let navigate = useNavigate();
|
||||||
const [selectedKeys, setSelectedKeys] = useState(['home']);
|
const [selectedKeys, setSelectedKeys] = useState(['home']);
|
||||||
const systemName = getSystemName();
|
const systemName = getSystemName();
|
||||||
const logo = getLogo();
|
const logo = getLogo();
|
||||||
const [isCollapsed, setIsCollapsed] = useState(defaultIsCollapsed);
|
const [isCollapsed, setIsCollapsed] = useState(defaultIsCollapsed);
|
||||||
|
|
||||||
const headerButtons = useMemo(() => [
|
const routerMap = {
|
||||||
{
|
home: '/',
|
||||||
text: '首页',
|
channel: '/channel',
|
||||||
itemKey: 'home',
|
token: '/token',
|
||||||
to: '/',
|
redemption: '/redemption',
|
||||||
icon: <IconHome/>
|
topup: '/topup',
|
||||||
},
|
user: '/user',
|
||||||
{
|
log: '/log',
|
||||||
text: '渠道',
|
midjourney: '/midjourney',
|
||||||
itemKey: 'channel',
|
setting: '/setting',
|
||||||
to: '/channel',
|
about: '/about',
|
||||||
icon: <IconLayers/>,
|
chat: '/chat',
|
||||||
className: isAdmin()?'semi-navigation-item-normal':'tableHiddle',
|
detail: '/detail'
|
||||||
},
|
};
|
||||||
{
|
|
||||||
text: '聊天',
|
|
||||||
itemKey: 'chat',
|
|
||||||
to: '/chat',
|
|
||||||
icon: <IconComment />,
|
|
||||||
className: localStorage.getItem('chat_link')?'semi-navigation-item-normal':'tableHiddle',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
text: '令牌',
|
|
||||||
itemKey: 'token',
|
|
||||||
to: '/token',
|
|
||||||
icon: <IconKey/>
|
|
||||||
},
|
|
||||||
{
|
|
||||||
text: '兑换码',
|
|
||||||
itemKey: 'redemption',
|
|
||||||
to: '/redemption',
|
|
||||||
icon: <IconGift/>,
|
|
||||||
className: isAdmin()?'semi-navigation-item-normal':'tableHiddle',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
text: '钱包',
|
|
||||||
itemKey: 'topup',
|
|
||||||
to: '/topup',
|
|
||||||
icon: <IconCreditCard/>
|
|
||||||
},
|
|
||||||
{
|
|
||||||
text: '用户管理',
|
|
||||||
itemKey: 'user',
|
|
||||||
to: '/user',
|
|
||||||
icon: <IconUser/>,
|
|
||||||
className: isAdmin()?'semi-navigation-item-normal':'tableHiddle',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
text: '日志',
|
|
||||||
itemKey: 'log',
|
|
||||||
to: '/log',
|
|
||||||
icon: <IconHistogram/>
|
|
||||||
},
|
|
||||||
{
|
|
||||||
text: '数据看板',
|
|
||||||
itemKey: 'detail',
|
|
||||||
to: '/detail',
|
|
||||||
icon: <IconCalendarClock />,
|
|
||||||
className: localStorage.getItem('enable_data_export') === 'true'?'semi-navigation-item-normal':'tableHiddle',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
text: '绘图',
|
|
||||||
itemKey: 'midjourney',
|
|
||||||
to: '/midjourney',
|
|
||||||
icon: <IconImage/>,
|
|
||||||
className: localStorage.getItem('enable_drawing') === 'true'?'semi-navigation-item-normal':'tableHiddle',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
text: '设置',
|
|
||||||
itemKey: 'setting',
|
|
||||||
to: '/setting',
|
|
||||||
icon: <IconSetting/>
|
|
||||||
},
|
|
||||||
// {
|
|
||||||
// text: '关于',
|
|
||||||
// itemKey: 'about',
|
|
||||||
// to: '/about',
|
|
||||||
// icon: <IconAt/>
|
|
||||||
// }
|
|
||||||
], [localStorage.getItem('enable_data_export'), localStorage.getItem('enable_drawing'), localStorage.getItem('chat_link'), isAdmin()]);
|
|
||||||
|
|
||||||
const loadStatus = async () => {
|
const headerButtons = useMemo(() => [
|
||||||
const res = await API.get('/api/status');
|
{
|
||||||
const { success, data } = res.data;
|
text: '首页',
|
||||||
if (success) {
|
itemKey: 'home',
|
||||||
localStorage.setItem('status', JSON.stringify(data));
|
to: '/',
|
||||||
statusDispatch({ type: 'set', payload: data });
|
icon: <IconHome />
|
||||||
localStorage.setItem('system_name', data.system_name);
|
},
|
||||||
localStorage.setItem('logo', data.logo);
|
{
|
||||||
localStorage.setItem('footer_html', data.footer_html);
|
text: '渠道',
|
||||||
localStorage.setItem('quota_per_unit', data.quota_per_unit);
|
itemKey: 'channel',
|
||||||
localStorage.setItem('display_in_currency', data.display_in_currency);
|
to: '/channel',
|
||||||
localStorage.setItem('enable_drawing', data.enable_drawing);
|
icon: <IconLayers />,
|
||||||
localStorage.setItem('enable_data_export', data.enable_data_export);
|
className: isAdmin() ? 'semi-navigation-item-normal' : 'tableHiddle'
|
||||||
localStorage.setItem('data_export_default_time', data.data_export_default_time);
|
},
|
||||||
localStorage.setItem('default_collapse_sidebar', data.default_collapse_sidebar);
|
{
|
||||||
if (data.chat_link) {
|
text: '聊天',
|
||||||
localStorage.setItem('chat_link', data.chat_link);
|
itemKey: 'chat',
|
||||||
} else {
|
to: '/chat',
|
||||||
localStorage.removeItem('chat_link');
|
icon: <IconComment />,
|
||||||
}
|
className: localStorage.getItem('chat_link') ? 'semi-navigation-item-normal' : 'tableHiddle'
|
||||||
if (data.chat_link2) {
|
},
|
||||||
localStorage.setItem('chat_link2', data.chat_link2);
|
{
|
||||||
} else {
|
text: '令牌',
|
||||||
localStorage.removeItem('chat_link2');
|
itemKey: 'token',
|
||||||
}
|
to: '/token',
|
||||||
} else {
|
icon: <IconKey />
|
||||||
showError('无法正常连接至服务器!');
|
},
|
||||||
}
|
{
|
||||||
};
|
text: '兑换码',
|
||||||
|
itemKey: 'redemption',
|
||||||
|
to: '/redemption',
|
||||||
|
icon: <IconGift />,
|
||||||
|
className: isAdmin() ? 'semi-navigation-item-normal' : 'tableHiddle'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
text: '钱包',
|
||||||
|
itemKey: 'topup',
|
||||||
|
to: '/topup',
|
||||||
|
icon: <IconCreditCard />
|
||||||
|
},
|
||||||
|
{
|
||||||
|
text: '用户管理',
|
||||||
|
itemKey: 'user',
|
||||||
|
to: '/user',
|
||||||
|
icon: <IconUser />,
|
||||||
|
className: isAdmin() ? 'semi-navigation-item-normal' : 'tableHiddle'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
text: '日志',
|
||||||
|
itemKey: 'log',
|
||||||
|
to: '/log',
|
||||||
|
icon: <IconHistogram />
|
||||||
|
},
|
||||||
|
{
|
||||||
|
text: '数据看板',
|
||||||
|
itemKey: 'detail',
|
||||||
|
to: '/detail',
|
||||||
|
icon: <IconCalendarClock />,
|
||||||
|
className: localStorage.getItem('enable_data_export') === 'true' ? 'semi-navigation-item-normal' : 'tableHiddle'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
text: '绘图',
|
||||||
|
itemKey: 'midjourney',
|
||||||
|
to: '/midjourney',
|
||||||
|
icon: <IconImage />,
|
||||||
|
className: localStorage.getItem('enable_drawing') === 'true' ? 'semi-navigation-item-normal' : 'tableHiddle'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
text: '设置',
|
||||||
|
itemKey: 'setting',
|
||||||
|
to: '/setting',
|
||||||
|
icon: <IconSetting />
|
||||||
|
}
|
||||||
|
// {
|
||||||
|
// text: '关于',
|
||||||
|
// itemKey: 'about',
|
||||||
|
// to: '/about',
|
||||||
|
// icon: <IconAt/>
|
||||||
|
// }
|
||||||
|
], [localStorage.getItem('enable_data_export'), localStorage.getItem('enable_drawing'), localStorage.getItem('chat_link'), isAdmin()]);
|
||||||
|
|
||||||
useEffect(() => {
|
const loadStatus = async () => {
|
||||||
loadStatus().then(() => {
|
const res = await API.get('/api/status');
|
||||||
setIsCollapsed(isMobile() || localStorage.getItem('default_collapse_sidebar') === 'true');
|
const { success, data } = res.data;
|
||||||
});
|
if (success) {
|
||||||
},[])
|
localStorage.setItem('status', JSON.stringify(data));
|
||||||
|
statusDispatch({ type: 'set', payload: data });
|
||||||
|
localStorage.setItem('system_name', data.system_name);
|
||||||
|
localStorage.setItem('logo', data.logo);
|
||||||
|
localStorage.setItem('footer_html', data.footer_html);
|
||||||
|
localStorage.setItem('quota_per_unit', data.quota_per_unit);
|
||||||
|
localStorage.setItem('display_in_currency', data.display_in_currency);
|
||||||
|
localStorage.setItem('enable_drawing', data.enable_drawing);
|
||||||
|
localStorage.setItem('enable_data_export', data.enable_data_export);
|
||||||
|
localStorage.setItem('data_export_default_time', data.data_export_default_time);
|
||||||
|
localStorage.setItem('default_collapse_sidebar', data.default_collapse_sidebar);
|
||||||
|
localStorage.setItem('mj_notify_enabled', data.mj_notify_enabled);
|
||||||
|
if (data.chat_link) {
|
||||||
|
localStorage.setItem('chat_link', data.chat_link);
|
||||||
|
} else {
|
||||||
|
localStorage.removeItem('chat_link');
|
||||||
|
}
|
||||||
|
if (data.chat_link2) {
|
||||||
|
localStorage.setItem('chat_link2', data.chat_link2);
|
||||||
|
} else {
|
||||||
|
localStorage.removeItem('chat_link2');
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
showError('无法正常连接至服务器!');
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
useEffect(() => {
|
||||||
<>
|
loadStatus().then(() => {
|
||||||
<Layout>
|
setIsCollapsed(isMobile() || localStorage.getItem('default_collapse_sidebar') === 'true');
|
||||||
<div style={{height: '100%'}}>
|
});
|
||||||
<Nav
|
let localKey = window.location.pathname.split('/')[1]
|
||||||
// bodyStyle={{ maxWidth: 200 }}
|
if (localKey === '') {
|
||||||
style={{ maxWidth: 200 }}
|
localKey = 'home'
|
||||||
defaultIsCollapsed={isMobile() || localStorage.getItem('default_collapse_sidebar') === 'true'}
|
}
|
||||||
isCollapsed={isCollapsed}
|
setSelectedKeys([localKey]);
|
||||||
onCollapseChange={collapsed => {
|
}, []);
|
||||||
setIsCollapsed(collapsed);
|
|
||||||
}}
|
|
||||||
selectedKeys={selectedKeys}
|
|
||||||
renderWrapper={({itemElement, isSubNav, isInSubNav, props}) => {
|
|
||||||
const routerMap = {
|
|
||||||
home: "/",
|
|
||||||
channel: "/channel",
|
|
||||||
token: "/token",
|
|
||||||
redemption: "/redemption",
|
|
||||||
topup: "/topup",
|
|
||||||
user: "/user",
|
|
||||||
log: "/log",
|
|
||||||
midjourney: "/midjourney",
|
|
||||||
setting: "/setting",
|
|
||||||
about: "/about",
|
|
||||||
chat: "/chat",
|
|
||||||
detail: "/detail",
|
|
||||||
};
|
|
||||||
return (
|
|
||||||
<Link
|
|
||||||
style={{textDecoration: "none"}}
|
|
||||||
to={routerMap[props.itemKey]}
|
|
||||||
>
|
|
||||||
{itemElement}
|
|
||||||
</Link>
|
|
||||||
);
|
|
||||||
}}
|
|
||||||
items={headerButtons}
|
|
||||||
onSelect={key => {
|
|
||||||
setSelectedKeys([key.itemKey]);
|
|
||||||
}}
|
|
||||||
header={{
|
|
||||||
logo: <img src={logo} alt='logo' style={{marginRight: '0.75em'}}/>,
|
|
||||||
text: systemName,
|
|
||||||
}}
|
|
||||||
// footer={{
|
|
||||||
// text: '© 2021 NekoAPI',
|
|
||||||
// }}
|
|
||||||
>
|
|
||||||
|
|
||||||
<Nav.Footer collapseButton={true}>
|
return (
|
||||||
</Nav.Footer>
|
<>
|
||||||
</Nav>
|
<Layout>
|
||||||
</div>
|
<div style={{ height: '100%' }}>
|
||||||
</Layout>
|
<Nav
|
||||||
</>
|
// bodyStyle={{ maxWidth: 200 }}
|
||||||
);
|
style={{ maxWidth: 200 }}
|
||||||
|
defaultIsCollapsed={isMobile() || localStorage.getItem('default_collapse_sidebar') === 'true'}
|
||||||
|
isCollapsed={isCollapsed}
|
||||||
|
onCollapseChange={collapsed => {
|
||||||
|
setIsCollapsed(collapsed);
|
||||||
|
}}
|
||||||
|
selectedKeys={selectedKeys}
|
||||||
|
renderWrapper={({ itemElement, isSubNav, isInSubNav, props }) => {
|
||||||
|
return (
|
||||||
|
<Link
|
||||||
|
style={{ textDecoration: 'none' }}
|
||||||
|
to={routerMap[props.itemKey]}
|
||||||
|
>
|
||||||
|
{itemElement}
|
||||||
|
</Link>
|
||||||
|
);
|
||||||
|
}}
|
||||||
|
items={headerButtons}
|
||||||
|
onSelect={key => {
|
||||||
|
setSelectedKeys([key.itemKey]);
|
||||||
|
}}
|
||||||
|
header={{
|
||||||
|
logo: <img src={logo} alt="logo" style={{ marginRight: '0.75em' }} />,
|
||||||
|
text: systemName
|
||||||
|
}}
|
||||||
|
// footer={{
|
||||||
|
// text: '© 2021 NekoAPI',
|
||||||
|
// }}
|
||||||
|
>
|
||||||
|
|
||||||
|
<Nav.Footer collapseButton={true}>
|
||||||
|
</Nav.Footer>
|
||||||
|
</Nav>
|
||||||
|
</div>
|
||||||
|
</Layout>
|
||||||
|
</>
|
||||||
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export default SiderBar;
|
export default SiderBar;
|
||||||
|
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
@ -1,337 +1,338 @@
|
|||||||
import React, {useEffect, useState} from 'react';
|
import React, { useEffect, useState } from 'react';
|
||||||
import {API, isAdmin, showError, showSuccess} from '../helpers';
|
import { API, showError, showSuccess } from '../helpers';
|
||||||
import {Button, Modal, Popconfirm, Popover, Table, Tag, Form, Tooltip, Space} from "@douyinfe/semi-ui";
|
import { Button, Form, Popconfirm, Space, Table, Tag, Tooltip } from '@douyinfe/semi-ui';
|
||||||
import {ITEMS_PER_PAGE} from '../constants';
|
import { ITEMS_PER_PAGE } from '../constants';
|
||||||
import {renderGroup, renderNumber, renderQuota, renderText, stringToColor} from '../helpers/render';
|
import { renderGroup, renderNumber, renderQuota } from '../helpers/render';
|
||||||
import AddUser from "../pages/User/AddUser";
|
import AddUser from '../pages/User/AddUser';
|
||||||
import EditUser from "../pages/User/EditUser";
|
import EditUser from '../pages/User/EditUser';
|
||||||
|
|
||||||
function renderRole(role) {
|
function renderRole(role) {
|
||||||
switch (role) {
|
switch (role) {
|
||||||
case 1:
|
case 1:
|
||||||
return <Tag size='large'>普通用户</Tag>;
|
return <Tag size="large">普通用户</Tag>;
|
||||||
case 10:
|
case 10:
|
||||||
return <Tag color='yellow' size='large'>管理员</Tag>;
|
return <Tag color="yellow" size="large">管理员</Tag>;
|
||||||
case 100:
|
case 100:
|
||||||
return <Tag color='orange' size='large'>超级管理员</Tag>;
|
return <Tag color="orange" size="large">超级管理员</Tag>;
|
||||||
default:
|
default:
|
||||||
return <Tag color='red' size='large'>未知身份</Tag>;
|
return <Tag color="red" size="large">未知身份</Tag>;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const UsersTable = () => {
|
const UsersTable = () => {
|
||||||
const columns = [{
|
const columns = [{
|
||||||
title: 'ID', dataIndex: 'id',
|
title: 'ID', dataIndex: 'id'
|
||||||
}, {
|
}, {
|
||||||
title: '用户名', dataIndex: 'username',
|
title: '用户名', dataIndex: 'username'
|
||||||
}, {
|
}, {
|
||||||
title: '分组', dataIndex: 'group', render: (text, record, index) => {
|
title: '分组', dataIndex: 'group', render: (text, record, index) => {
|
||||||
return (<div>
|
return (<div>
|
||||||
{renderGroup(text)}
|
{renderGroup(text)}
|
||||||
</div>);
|
</div>);
|
||||||
},
|
}
|
||||||
}, {
|
}, {
|
||||||
title: '统计信息', dataIndex: 'info', render: (text, record, index) => {
|
title: '统计信息', dataIndex: 'info', render: (text, record, index) => {
|
||||||
return (<div>
|
return (<div>
|
||||||
<Space spacing={1}>
|
<Space spacing={1}>
|
||||||
<Tooltip content={'剩余额度'}>
|
<Tooltip content={'剩余额度'}>
|
||||||
<Tag color='white' size='large'>{renderQuota(record.quota)}</Tag>
|
<Tag color="white" size="large">{renderQuota(record.quota)}</Tag>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
<Tooltip content={'已用额度'}>
|
<Tooltip content={'已用额度'}>
|
||||||
<Tag color='white' size='large'>{renderQuota(record.used_quota)}</Tag>
|
<Tag color="white" size="large">{renderQuota(record.used_quota)}</Tag>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
<Tooltip content={'调用次数'}>
|
<Tooltip content={'调用次数'}>
|
||||||
<Tag color='white' size='large'>{renderNumber(record.request_count)}</Tag>
|
<Tag color="white" size="large">{renderNumber(record.request_count)}</Tag>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
</Space>
|
</Space>
|
||||||
</div>);
|
</div>);
|
||||||
}
|
}
|
||||||
}, {
|
}, {
|
||||||
title: '邀请信息', dataIndex: 'invite', render: (text, record, index) => {
|
title: '邀请信息', dataIndex: 'invite', render: (text, record, index) => {
|
||||||
return (<div>
|
return (<div>
|
||||||
<Space spacing={1}>
|
<Space spacing={1}>
|
||||||
<Tooltip content={'邀请人数'}>
|
<Tooltip content={'邀请人数'}>
|
||||||
<Tag color='white' size='large'>{renderNumber(record.aff_count)}</Tag>
|
<Tag color="white" size="large">{renderNumber(record.aff_count)}</Tag>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
<Tooltip content={'邀请总收益'}>
|
<Tooltip content={'邀请总收益'}>
|
||||||
<Tag color='white' size='large'>{renderQuota(record.aff_history_quota)}</Tag>
|
<Tag color="white" size="large">{renderQuota(record.aff_history_quota)}</Tag>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
<Tooltip content={'邀请人ID'}>
|
<Tooltip content={'邀请人ID'}>
|
||||||
{record.inviter_id === 0 ? <Tag color='white' size='large'>无</Tag> :
|
{record.inviter_id === 0 ? <Tag color="white" size="large">无</Tag> :
|
||||||
<Tag color='white' size='large'>{record.inviter_id}</Tag>}
|
<Tag color="white" size="large">{record.inviter_id}</Tag>}
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
</Space>
|
</Space>
|
||||||
</div>);
|
</div>);
|
||||||
}
|
}
|
||||||
}, {
|
}, {
|
||||||
title: '角色', dataIndex: 'role', render: (text, record, index) => {
|
title: '角色', dataIndex: 'role', render: (text, record, index) => {
|
||||||
return (<div>
|
return (<div>
|
||||||
{renderRole(text)}
|
{renderRole(text)}
|
||||||
</div>);
|
</div>);
|
||||||
},
|
}
|
||||||
}, {
|
}, {
|
||||||
title: '状态', dataIndex: 'status', render: (text, record, index) => {
|
title: '状态', dataIndex: 'status', render: (text, record, index) => {
|
||||||
return (<div>
|
return (<div>
|
||||||
{record.DeletedAt !== null? <Tag color='red'>已注销</Tag> : renderStatus(text)}
|
{record.DeletedAt !== null ? <Tag color="red">已注销</Tag> : renderStatus(text)}
|
||||||
</div>);
|
</div>);
|
||||||
},
|
}
|
||||||
}, {
|
}, {
|
||||||
title: '', dataIndex: 'operate', render: (text, record, index) => (<div>
|
title: '', dataIndex: 'operate', render: (text, record, index) => (<div>
|
||||||
{
|
{
|
||||||
record.DeletedAt !== null ? <></>:
|
record.DeletedAt !== null ? <></> :
|
||||||
<>
|
<>
|
||||||
<Popconfirm
|
|
||||||
title="确定?"
|
|
||||||
okType={'warning'}
|
|
||||||
onConfirm={() => {
|
|
||||||
manageUser(record.username, 'promote', record)
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<Button theme='light' type='warning' style={{marginRight: 1}}>提升</Button>
|
|
||||||
</Popconfirm>
|
|
||||||
<Popconfirm
|
|
||||||
title="确定?"
|
|
||||||
okType={'warning'}
|
|
||||||
onConfirm={() => {
|
|
||||||
manageUser(record.username, 'demote', record)
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<Button theme='light' type='secondary' style={{marginRight: 1}}>降级</Button>
|
|
||||||
</Popconfirm>
|
|
||||||
{record.status === 1 ?
|
|
||||||
<Button theme='light' type='warning' style={{marginRight: 1}} onClick={async () => {
|
|
||||||
manageUser(record.username, 'disable', record)
|
|
||||||
}}>禁用</Button> :
|
|
||||||
<Button theme='light' type='secondary' style={{marginRight: 1}} onClick={async () => {
|
|
||||||
manageUser(record.username, 'enable', record);
|
|
||||||
}} disabled={record.status === 3}>启用</Button>}
|
|
||||||
<Button theme='light' type='tertiary' style={{marginRight: 1}} onClick={() => {
|
|
||||||
setEditingUser(record);
|
|
||||||
setShowEditUser(true);
|
|
||||||
}}>编辑</Button>
|
|
||||||
</>
|
|
||||||
|
|
||||||
}
|
|
||||||
<Popconfirm
|
<Popconfirm
|
||||||
title="确定是否要删除此用户?"
|
title="确定?"
|
||||||
content="硬删除,此修改将不可逆"
|
okType={'warning'}
|
||||||
okType={'danger'}
|
onConfirm={() => {
|
||||||
position={'left'}
|
manageUser(record.username, 'promote', record);
|
||||||
onConfirm={() => {
|
}}
|
||||||
manageUser(record.username, 'delete', record).then(() => {
|
|
||||||
removeRecord(record.id);
|
|
||||||
})
|
|
||||||
}}
|
|
||||||
>
|
>
|
||||||
<Button theme='light' type='danger' style={{marginRight: 1}}>删除</Button>
|
<Button theme="light" type="warning" style={{ marginRight: 1 }}>提升</Button>
|
||||||
</Popconfirm>
|
</Popconfirm>
|
||||||
</div>),
|
<Popconfirm
|
||||||
},];
|
title="确定?"
|
||||||
|
okType={'warning'}
|
||||||
|
onConfirm={() => {
|
||||||
|
manageUser(record.username, 'demote', record);
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Button theme="light" type="secondary" style={{ marginRight: 1 }}>降级</Button>
|
||||||
|
</Popconfirm>
|
||||||
|
{record.status === 1 ?
|
||||||
|
<Button theme="light" type="warning" style={{ marginRight: 1 }} onClick={async () => {
|
||||||
|
manageUser(record.username, 'disable', record);
|
||||||
|
}}>禁用</Button> :
|
||||||
|
<Button theme="light" type="secondary" style={{ marginRight: 1 }} onClick={async () => {
|
||||||
|
manageUser(record.username, 'enable', record);
|
||||||
|
}} disabled={record.status === 3}>启用</Button>}
|
||||||
|
<Button theme="light" type="tertiary" style={{ marginRight: 1 }} onClick={() => {
|
||||||
|
setEditingUser(record);
|
||||||
|
setShowEditUser(true);
|
||||||
|
}}>编辑</Button>
|
||||||
|
</>
|
||||||
|
|
||||||
const [users, setUsers] = useState([]);
|
}
|
||||||
const [loading, setLoading] = useState(true);
|
<Popconfirm
|
||||||
const [activePage, setActivePage] = useState(1);
|
title="确定是否要删除此用户?"
|
||||||
const [searchKeyword, setSearchKeyword] = useState('');
|
content="硬删除,此修改将不可逆"
|
||||||
const [searching, setSearching] = useState(false);
|
okType={'danger'}
|
||||||
const [userCount, setUserCount] = useState(ITEMS_PER_PAGE);
|
position={'left'}
|
||||||
const [showAddUser, setShowAddUser] = useState(false);
|
onConfirm={() => {
|
||||||
const [showEditUser, setShowEditUser] = useState(false);
|
manageUser(record.username, 'delete', record).then(() => {
|
||||||
const [editingUser, setEditingUser] = useState({
|
removeRecord(record.id);
|
||||||
id: undefined,
|
});
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Button theme="light" type="danger" style={{ marginRight: 1 }}>删除</Button>
|
||||||
|
</Popconfirm>
|
||||||
|
</div>)
|
||||||
|
}];
|
||||||
|
|
||||||
|
const [users, setUsers] = useState([]);
|
||||||
|
const [loading, setLoading] = useState(true);
|
||||||
|
const [activePage, setActivePage] = useState(1);
|
||||||
|
const [searchKeyword, setSearchKeyword] = useState('');
|
||||||
|
const [searching, setSearching] = useState(false);
|
||||||
|
const [userCount, setUserCount] = useState(ITEMS_PER_PAGE);
|
||||||
|
const [showAddUser, setShowAddUser] = useState(false);
|
||||||
|
const [showEditUser, setShowEditUser] = useState(false);
|
||||||
|
const [editingUser, setEditingUser] = useState({
|
||||||
|
id: undefined
|
||||||
|
});
|
||||||
|
|
||||||
|
const setCount = (data) => {
|
||||||
|
if (data.length >= (activePage) * ITEMS_PER_PAGE) {
|
||||||
|
setUserCount(data.length + 1);
|
||||||
|
} else {
|
||||||
|
setUserCount(data.length);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const removeRecord = key => {
|
||||||
|
console.log(key);
|
||||||
|
let newDataSource = [...users];
|
||||||
|
if (key != null) {
|
||||||
|
let idx = newDataSource.findIndex(data => data.id === key);
|
||||||
|
|
||||||
|
if (idx > -1) {
|
||||||
|
newDataSource.splice(idx, 1);
|
||||||
|
setUsers(newDataSource);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const loadUsers = async (startIdx) => {
|
||||||
|
const res = await API.get(`/api/user/?p=${startIdx}`);
|
||||||
|
const { success, message, data } = res.data;
|
||||||
|
if (success) {
|
||||||
|
if (startIdx === 0) {
|
||||||
|
setUsers(data);
|
||||||
|
setCount(data);
|
||||||
|
} else {
|
||||||
|
let newUsers = users;
|
||||||
|
newUsers.push(...data);
|
||||||
|
setUsers(newUsers);
|
||||||
|
setCount(newUsers);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
showError(message);
|
||||||
|
}
|
||||||
|
setLoading(false);
|
||||||
|
};
|
||||||
|
|
||||||
|
const onPaginationChange = (e, { activePage }) => {
|
||||||
|
(async () => {
|
||||||
|
if (activePage === Math.ceil(users.length / ITEMS_PER_PAGE) + 1) {
|
||||||
|
// In this case we have to load more data and then append them.
|
||||||
|
await loadUsers(activePage - 1);
|
||||||
|
}
|
||||||
|
setActivePage(activePage);
|
||||||
|
})();
|
||||||
|
};
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
loadUsers(0)
|
||||||
|
.then()
|
||||||
|
.catch((reason) => {
|
||||||
|
showError(reason);
|
||||||
|
});
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const manageUser = async (username, action, record) => {
|
||||||
|
const res = await API.post('/api/user/manage', {
|
||||||
|
username, action
|
||||||
});
|
});
|
||||||
|
const { success, message } = res.data;
|
||||||
|
if (success) {
|
||||||
|
showSuccess('操作成功完成!');
|
||||||
|
let user = res.data.data;
|
||||||
|
let newUsers = [...users];
|
||||||
|
if (action === 'delete') {
|
||||||
|
|
||||||
const setCount = (data) => {
|
} else {
|
||||||
if (data.length >= (activePage) * ITEMS_PER_PAGE) {
|
record.status = user.status;
|
||||||
setUserCount(data.length + 1);
|
record.role = user.role;
|
||||||
} else {
|
}
|
||||||
setUserCount(data.length);
|
setUsers(newUsers);
|
||||||
}
|
} else {
|
||||||
|
showError(message);
|
||||||
}
|
}
|
||||||
|
};
|
||||||
|
|
||||||
const removeRecord = key => {
|
const renderStatus = (status) => {
|
||||||
console.log(key);
|
switch (status) {
|
||||||
let newDataSource = [...users];
|
case 1:
|
||||||
if (key != null) {
|
return <Tag size="large">已激活</Tag>;
|
||||||
let idx = newDataSource.findIndex(data => data.id === key);
|
case 2:
|
||||||
|
return (<Tag size="large" color="red">
|
||||||
if (idx > -1) {
|
已封禁
|
||||||
newDataSource.splice(idx, 1);
|
</Tag>);
|
||||||
setUsers(newDataSource);
|
default:
|
||||||
}
|
return (<Tag size="large" color="grey">
|
||||||
}
|
未知状态
|
||||||
};
|
</Tag>);
|
||||||
|
|
||||||
const loadUsers = async (startIdx) => {
|
|
||||||
const res = await API.get(`/api/user/?p=${startIdx}`);
|
|
||||||
const {success, message, data} = res.data;
|
|
||||||
if (success) {
|
|
||||||
if (startIdx === 0) {
|
|
||||||
setUsers(data);
|
|
||||||
setCount(data);
|
|
||||||
} else {
|
|
||||||
let newUsers = users;
|
|
||||||
newUsers.push(...data);
|
|
||||||
setUsers(newUsers);
|
|
||||||
setCount(newUsers);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
showError(message);
|
|
||||||
}
|
|
||||||
setLoading(false);
|
|
||||||
};
|
|
||||||
|
|
||||||
const onPaginationChange = (e, {activePage}) => {
|
|
||||||
(async () => {
|
|
||||||
if (activePage === Math.ceil(users.length / ITEMS_PER_PAGE) + 1) {
|
|
||||||
// In this case we have to load more data and then append them.
|
|
||||||
await loadUsers(activePage - 1);
|
|
||||||
}
|
|
||||||
setActivePage(activePage);
|
|
||||||
})();
|
|
||||||
};
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
loadUsers(0)
|
|
||||||
.then()
|
|
||||||
.catch((reason) => {
|
|
||||||
showError(reason);
|
|
||||||
});
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
const manageUser = async (username, action, record) => {
|
|
||||||
const res = await API.post('/api/user/manage', {
|
|
||||||
username, action
|
|
||||||
});
|
|
||||||
const {success, message} = res.data;
|
|
||||||
if (success) {
|
|
||||||
showSuccess('操作成功完成!');
|
|
||||||
let user = res.data.data;
|
|
||||||
let newUsers = [...users];
|
|
||||||
if (action === 'delete') {
|
|
||||||
|
|
||||||
} else {
|
|
||||||
record.status = user.status;
|
|
||||||
record.role = user.role;
|
|
||||||
}
|
|
||||||
setUsers(newUsers);
|
|
||||||
} else {
|
|
||||||
showError(message);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const renderStatus = (status) => {
|
|
||||||
switch (status) {
|
|
||||||
case 1:
|
|
||||||
return <Tag size='large'>已激活</Tag>;
|
|
||||||
case 2:
|
|
||||||
return (<Tag size='large' color='red'>
|
|
||||||
已封禁
|
|
||||||
</Tag>);
|
|
||||||
default:
|
|
||||||
return (<Tag size='large' color='grey'>
|
|
||||||
未知状态
|
|
||||||
</Tag>);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const searchUsers = async () => {
|
|
||||||
if (searchKeyword === '') {
|
|
||||||
// if keyword is blank, load files instead.
|
|
||||||
await loadUsers(0);
|
|
||||||
setActivePage(1);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
setSearching(true);
|
|
||||||
const res = await API.get(`/api/user/search?keyword=${searchKeyword}`);
|
|
||||||
const {success, message, data} = res.data;
|
|
||||||
if (success) {
|
|
||||||
setUsers(data);
|
|
||||||
setActivePage(1);
|
|
||||||
} else {
|
|
||||||
showError(message);
|
|
||||||
}
|
|
||||||
setSearching(false);
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleKeywordChange = async (value) => {
|
|
||||||
setSearchKeyword(value.trim());
|
|
||||||
};
|
|
||||||
|
|
||||||
const sortUser = (key) => {
|
|
||||||
if (users.length === 0) return;
|
|
||||||
setLoading(true);
|
|
||||||
let sortedUsers = [...users];
|
|
||||||
sortedUsers.sort((a, b) => {
|
|
||||||
return ('' + a[key]).localeCompare(b[key]);
|
|
||||||
});
|
|
||||||
if (sortedUsers[0].id === users[0].id) {
|
|
||||||
sortedUsers.reverse();
|
|
||||||
}
|
|
||||||
setUsers(sortedUsers);
|
|
||||||
setLoading(false);
|
|
||||||
};
|
|
||||||
|
|
||||||
const handlePageChange = page => {
|
|
||||||
setActivePage(page);
|
|
||||||
if (page === Math.ceil(users.length / ITEMS_PER_PAGE) + 1) {
|
|
||||||
// In this case we have to load more data and then append them.
|
|
||||||
loadUsers(page - 1).then(r => {
|
|
||||||
});
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const pageData = users.slice((activePage - 1) * ITEMS_PER_PAGE, activePage * ITEMS_PER_PAGE);
|
|
||||||
|
|
||||||
const closeAddUser = () => {
|
|
||||||
setShowAddUser(false);
|
|
||||||
}
|
}
|
||||||
|
};
|
||||||
|
|
||||||
const closeEditUser = () => {
|
const searchUsers = async () => {
|
||||||
setShowEditUser(false);
|
if (searchKeyword === '') {
|
||||||
setEditingUser({
|
// if keyword is blank, load files instead.
|
||||||
id: undefined,
|
await loadUsers(0);
|
||||||
});
|
setActivePage(1);
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
setSearching(true);
|
||||||
|
const res = await API.get(`/api/user/search?keyword=${searchKeyword}`);
|
||||||
|
const { success, message, data } = res.data;
|
||||||
|
if (success) {
|
||||||
|
setUsers(data);
|
||||||
|
setActivePage(1);
|
||||||
|
} else {
|
||||||
|
showError(message);
|
||||||
|
}
|
||||||
|
setSearching(false);
|
||||||
|
};
|
||||||
|
|
||||||
const refresh = async () => {
|
const handleKeywordChange = async (value) => {
|
||||||
if (searchKeyword === '') {
|
setSearchKeyword(value.trim());
|
||||||
await loadUsers(activePage - 1);
|
};
|
||||||
} else {
|
|
||||||
await searchUsers();
|
const sortUser = (key) => {
|
||||||
|
if (users.length === 0) return;
|
||||||
|
setLoading(true);
|
||||||
|
let sortedUsers = [...users];
|
||||||
|
sortedUsers.sort((a, b) => {
|
||||||
|
return ('' + a[key]).localeCompare(b[key]);
|
||||||
|
});
|
||||||
|
if (sortedUsers[0].id === users[0].id) {
|
||||||
|
sortedUsers.reverse();
|
||||||
|
}
|
||||||
|
setUsers(sortedUsers);
|
||||||
|
setLoading(false);
|
||||||
|
};
|
||||||
|
|
||||||
|
const handlePageChange = page => {
|
||||||
|
setActivePage(page);
|
||||||
|
if (page === Math.ceil(users.length / ITEMS_PER_PAGE) + 1) {
|
||||||
|
// In this case we have to load more data and then append them.
|
||||||
|
loadUsers(page - 1).then(r => {
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const pageData = users.slice((activePage - 1) * ITEMS_PER_PAGE, activePage * ITEMS_PER_PAGE);
|
||||||
|
|
||||||
|
const closeAddUser = () => {
|
||||||
|
setShowAddUser(false);
|
||||||
|
};
|
||||||
|
|
||||||
|
const closeEditUser = () => {
|
||||||
|
setShowEditUser(false);
|
||||||
|
setEditingUser({
|
||||||
|
id: undefined
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
const refresh = async () => {
|
||||||
|
if (searchKeyword === '') {
|
||||||
|
await loadUsers(activePage - 1);
|
||||||
|
} else {
|
||||||
|
await searchUsers();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<AddUser refresh={refresh} visible={showAddUser} handleClose={closeAddUser}></AddUser>
|
||||||
|
<EditUser refresh={refresh} visible={showEditUser} handleClose={closeEditUser}
|
||||||
|
editingUser={editingUser}></EditUser>
|
||||||
|
<Form onSubmit={searchUsers}>
|
||||||
|
<Form.Input
|
||||||
|
label="搜索关键字"
|
||||||
|
icon="search"
|
||||||
|
field="keyword"
|
||||||
|
iconPosition="left"
|
||||||
|
placeholder="搜索用户的 ID,用户名,显示名称,以及邮箱地址 ..."
|
||||||
|
value={searchKeyword}
|
||||||
|
loading={searching}
|
||||||
|
onChange={value => handleKeywordChange(value)}
|
||||||
|
/>
|
||||||
|
</Form>
|
||||||
|
|
||||||
|
<Table columns={columns} dataSource={pageData} pagination={{
|
||||||
|
currentPage: activePage,
|
||||||
|
pageSize: ITEMS_PER_PAGE,
|
||||||
|
total: userCount,
|
||||||
|
pageSizeOpts: [10, 20, 50, 100],
|
||||||
|
onPageChange: handlePageChange
|
||||||
|
}} loading={loading} />
|
||||||
|
<Button theme="light" type="primary" style={{ marginRight: 8 }} onClick={
|
||||||
|
() => {
|
||||||
|
setShowAddUser(true);
|
||||||
}
|
}
|
||||||
};
|
}>添加用户</Button>
|
||||||
|
</>
|
||||||
return (
|
);
|
||||||
<>
|
|
||||||
<AddUser refresh={refresh} visible={showAddUser} handleClose={closeAddUser}></AddUser>
|
|
||||||
<EditUser refresh={refresh} visible={showEditUser} handleClose={closeEditUser} editingUser={editingUser}></EditUser>
|
|
||||||
<Form onSubmit={searchUsers}>
|
|
||||||
<Form.Input
|
|
||||||
label='搜索关键字'
|
|
||||||
icon='search'
|
|
||||||
field='keyword'
|
|
||||||
iconPosition='left'
|
|
||||||
placeholder='搜索用户的 ID,用户名,显示名称,以及邮箱地址 ...'
|
|
||||||
value={searchKeyword}
|
|
||||||
loading={searching}
|
|
||||||
onChange={value => handleKeywordChange(value)}
|
|
||||||
/>
|
|
||||||
</Form>
|
|
||||||
|
|
||||||
<Table columns={columns} dataSource={pageData} pagination={{
|
|
||||||
currentPage: activePage,
|
|
||||||
pageSize: ITEMS_PER_PAGE,
|
|
||||||
total: userCount,
|
|
||||||
pageSizeOpts: [10, 20, 50, 100],
|
|
||||||
onPageChange: handlePageChange,
|
|
||||||
}} loading={loading}/>
|
|
||||||
<Button theme='light' type='primary' style={{marginRight: 8}} onClick={
|
|
||||||
() => {
|
|
||||||
setShowAddUser(true);
|
|
||||||
}
|
|
||||||
}>添加用户</Button>
|
|
||||||
</>
|
|
||||||
);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
export default UsersTable;
|
export default UsersTable;
|
||||||
|
@ -3,14 +3,14 @@ import { Icon } from '@douyinfe/semi-ui';
|
|||||||
|
|
||||||
const WeChatIcon = () => {
|
const WeChatIcon = () => {
|
||||||
function CustomIcon() {
|
function CustomIcon() {
|
||||||
return <svg t='1709714447384' className='icon' viewBox='0 0 1024 1024' version='1.1'
|
return <svg t="1709714447384" className="icon" viewBox="0 0 1024 1024" version="1.1"
|
||||||
xmlns='http://www.w3.org/2000/svg' p-id='5091' width='16' height='16'>
|
xmlns="http://www.w3.org/2000/svg" p-id="5091" width="16" height="16">
|
||||||
<path
|
<path
|
||||||
d='M690.1 377.4c5.9 0 11.8 0.2 17.6 0.5-24.4-128.7-158.3-227.1-319.9-227.1C209 150.8 64 271.4 64 420.2c0 81.1 43.6 154.2 111.9 203.6 5.5 3.9 9.1 10.3 9.1 17.6 0 2.4-0.5 4.6-1.1 6.9-5.5 20.3-14.2 52.8-14.6 54.3-0.7 2.6-1.7 5.2-1.7 7.9 0 5.9 4.8 10.8 10.8 10.8 2.3 0 4.2-0.9 6.2-2l70.9-40.9c5.3-3.1 11-5 17.2-5 3.2 0 6.4 0.5 9.5 1.4 33.1 9.5 68.8 14.8 105.7 14.8 6 0 11.9-0.1 17.8-0.4-7.1-21-10.9-43.1-10.9-66 0-135.8 132.2-245.8 295.3-245.8z m-194.3-86.5c23.8 0 43.2 19.3 43.2 43.1s-19.3 43.1-43.2 43.1c-23.8 0-43.2-19.3-43.2-43.1s19.4-43.1 43.2-43.1z m-215.9 86.2c-23.8 0-43.2-19.3-43.2-43.1s19.3-43.1 43.2-43.1 43.2 19.3 43.2 43.1-19.4 43.1-43.2 43.1z'
|
d="M690.1 377.4c5.9 0 11.8 0.2 17.6 0.5-24.4-128.7-158.3-227.1-319.9-227.1C209 150.8 64 271.4 64 420.2c0 81.1 43.6 154.2 111.9 203.6 5.5 3.9 9.1 10.3 9.1 17.6 0 2.4-0.5 4.6-1.1 6.9-5.5 20.3-14.2 52.8-14.6 54.3-0.7 2.6-1.7 5.2-1.7 7.9 0 5.9 4.8 10.8 10.8 10.8 2.3 0 4.2-0.9 6.2-2l70.9-40.9c5.3-3.1 11-5 17.2-5 3.2 0 6.4 0.5 9.5 1.4 33.1 9.5 68.8 14.8 105.7 14.8 6 0 11.9-0.1 17.8-0.4-7.1-21-10.9-43.1-10.9-66 0-135.8 132.2-245.8 295.3-245.8z m-194.3-86.5c23.8 0 43.2 19.3 43.2 43.1s-19.3 43.1-43.2 43.1c-23.8 0-43.2-19.3-43.2-43.1s19.4-43.1 43.2-43.1z m-215.9 86.2c-23.8 0-43.2-19.3-43.2-43.1s19.3-43.1 43.2-43.1 43.2 19.3 43.2 43.1-19.4 43.1-43.2 43.1z"
|
||||||
p-id='5092'></path>
|
p-id="5092"></path>
|
||||||
<path
|
<path
|
||||||
d='M866.7 792.7c56.9-41.2 93.2-102 93.2-169.7 0-124-120.8-224.5-269.9-224.5-149 0-269.9 100.5-269.9 224.5S540.9 847.5 690 847.5c30.8 0 60.6-4.4 88.1-12.3 2.6-0.8 5.2-1.2 7.9-1.2 5.2 0 9.9 1.6 14.3 4.1l59.1 34c1.7 1 3.3 1.7 5.2 1.7 2.4 0 4.7-0.9 6.4-2.6 1.7-1.7 2.6-4 2.6-6.4 0-2.2-0.9-4.4-1.4-6.6-0.3-1.2-7.6-28.3-12.2-45.3-0.5-1.9-0.9-3.8-0.9-5.7 0.1-5.9 3.1-11.2 7.6-14.5zM600.2 587.2c-19.9 0-36-16.1-36-35.9 0-19.8 16.1-35.9 36-35.9s36 16.1 36 35.9c0 19.8-16.2 35.9-36 35.9z m179.9 0c-19.9 0-36-16.1-36-35.9 0-19.8 16.1-35.9 36-35.9s36 16.1 36 35.9c-0.1 19.8-16.2 35.9-36 35.9z'
|
d="M866.7 792.7c56.9-41.2 93.2-102 93.2-169.7 0-124-120.8-224.5-269.9-224.5-149 0-269.9 100.5-269.9 224.5S540.9 847.5 690 847.5c30.8 0 60.6-4.4 88.1-12.3 2.6-0.8 5.2-1.2 7.9-1.2 5.2 0 9.9 1.6 14.3 4.1l59.1 34c1.7 1 3.3 1.7 5.2 1.7 2.4 0 4.7-0.9 6.4-2.6 1.7-1.7 2.6-4 2.6-6.4 0-2.2-0.9-4.4-1.4-6.6-0.3-1.2-7.6-28.3-12.2-45.3-0.5-1.9-0.9-3.8-0.9-5.7 0.1-5.9 3.1-11.2 7.6-14.5zM600.2 587.2c-19.9 0-36-16.1-36-35.9 0-19.8 16.1-35.9 36-35.9s36 16.1 36 35.9c0 19.8-16.2 35.9-36 35.9z m179.9 0c-19.9 0-36-16.1-36-35.9 0-19.8 16.1-35.9 36-35.9s36 16.1 36 35.9c-0.1 19.8-16.2 35.9-36 35.9z"
|
||||||
p-id='5093'></path>
|
p-id="5093"></path>
|
||||||
</svg>;
|
</svg>;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,17 +1,17 @@
|
|||||||
import React, { useEffect, useState } from 'react';
|
import React, { useEffect, useState } from 'react';
|
||||||
import { useParams, useNavigate } from 'react-router-dom';
|
import { useNavigate, useParams } from 'react-router-dom';
|
||||||
import {API, downloadTextAsFile, isMobile, showError, showSuccess} from '../../helpers';
|
import { API, downloadTextAsFile, isMobile, showError, showSuccess } from '../../helpers';
|
||||||
import { renderQuota, renderQuotaWithPrompt } from '../../helpers/render';
|
import { renderQuotaWithPrompt } from '../../helpers/render';
|
||||||
import {SideSheet, Space, Spin, Button, Input, Typography, AutoComplete, Modal} from "@douyinfe/semi-ui";
|
import { AutoComplete, Button, Input, Modal, SideSheet, Space, Spin, Typography } from '@douyinfe/semi-ui';
|
||||||
import Title from "@douyinfe/semi-ui/lib/es/typography/title";
|
import Title from '@douyinfe/semi-ui/lib/es/typography/title';
|
||||||
import {Divider} from "semantic-ui-react";
|
import { Divider } from 'semantic-ui-react';
|
||||||
|
|
||||||
const EditRedemption = (props) => {
|
const EditRedemption = (props) => {
|
||||||
const isEdit = props.editingRedemption.id !== undefined;
|
const isEdit = props.editingRedemption.id !== undefined;
|
||||||
const [loading, setLoading] = useState(isEdit);
|
const [loading, setLoading] = useState(isEdit);
|
||||||
|
|
||||||
const params = useParams();
|
const params = useParams();
|
||||||
const navigate = useNavigate()
|
const navigate = useNavigate();
|
||||||
const originInputs = {
|
const originInputs = {
|
||||||
name: '',
|
name: '',
|
||||||
quota: 100000,
|
quota: 100000,
|
||||||
@ -22,8 +22,8 @@ const EditRedemption = (props) => {
|
|||||||
|
|
||||||
const handleCancel = () => {
|
const handleCancel = () => {
|
||||||
props.handleClose();
|
props.handleClose();
|
||||||
}
|
};
|
||||||
|
|
||||||
const handleInputChange = (name, value) => {
|
const handleInputChange = (name, value) => {
|
||||||
setInputs((inputs) => ({ ...inputs, [name]: value }));
|
setInputs((inputs) => ({ ...inputs, [name]: value }));
|
||||||
};
|
};
|
||||||
@ -43,9 +43,9 @@ const EditRedemption = (props) => {
|
|||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (isEdit) {
|
if (isEdit) {
|
||||||
loadRedemption().then(
|
loadRedemption().then(
|
||||||
() => {
|
() => {
|
||||||
// console.log(inputs);
|
// console.log(inputs);
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
setInputs(originInputs);
|
setInputs(originInputs);
|
||||||
@ -82,21 +82,21 @@ const EditRedemption = (props) => {
|
|||||||
showError(message);
|
showError(message);
|
||||||
}
|
}
|
||||||
if (!isEdit && data) {
|
if (!isEdit && data) {
|
||||||
let text = "";
|
let text = '';
|
||||||
for (let i = 0; i < data.length; i++) {
|
for (let i = 0; i < data.length; i++) {
|
||||||
text += data[i] + "\n";
|
text += data[i] + '\n';
|
||||||
}
|
}
|
||||||
// downloadTextAsFile(text, `${inputs.name}.txt`);
|
// downloadTextAsFile(text, `${inputs.name}.txt`);
|
||||||
Modal.confirm({
|
Modal.confirm({
|
||||||
title: '兑换码创建成功',
|
title: '兑换码创建成功',
|
||||||
content: (
|
content: (
|
||||||
<div>
|
<div>
|
||||||
<p>兑换码创建成功,是否下载兑换码?</p>
|
<p>兑换码创建成功,是否下载兑换码?</p>
|
||||||
<p>兑换码将以文本文件的形式下载,文件名为兑换码的名称。</p>
|
<p>兑换码将以文本文件的形式下载,文件名为兑换码的名称。</p>
|
||||||
</div>
|
</div>
|
||||||
),
|
),
|
||||||
onOk: () => {
|
onOk: () => {
|
||||||
downloadTextAsFile(text, `${inputs.name}.txt`);
|
downloadTextAsFile(text, `${inputs.name}.txt`);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@ -106,71 +106,71 @@ const EditRedemption = (props) => {
|
|||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<SideSheet
|
<SideSheet
|
||||||
placement={isEdit ? 'right' : 'left'}
|
placement={isEdit ? 'right' : 'left'}
|
||||||
title={<Title level={3}>{isEdit ? '更新兑换码信息' : '创建新的兑换码'}</Title>}
|
title={<Title level={3}>{isEdit ? '更新兑换码信息' : '创建新的兑换码'}</Title>}
|
||||||
headerStyle={{borderBottom: '1px solid var(--semi-color-border)'}}
|
headerStyle={{ borderBottom: '1px solid var(--semi-color-border)' }}
|
||||||
bodyStyle={{borderBottom: '1px solid var(--semi-color-border)'}}
|
bodyStyle={{ borderBottom: '1px solid var(--semi-color-border)' }}
|
||||||
visible={props.visiable}
|
visible={props.visiable}
|
||||||
footer={
|
footer={
|
||||||
<div style={{display: 'flex', justifyContent: 'flex-end'}}>
|
<div style={{ display: 'flex', justifyContent: 'flex-end' }}>
|
||||||
<Space>
|
<Space>
|
||||||
<Button theme='solid' size={'large'} onClick={submit}>提交</Button>
|
<Button theme="solid" size={'large'} onClick={submit}>提交</Button>
|
||||||
<Button theme='solid' size={'large'} type={'tertiary'} onClick={handleCancel}>取消</Button>
|
<Button theme="solid" size={'large'} type={'tertiary'} onClick={handleCancel}>取消</Button>
|
||||||
</Space>
|
</Space>
|
||||||
</div>
|
</div>
|
||||||
}
|
}
|
||||||
closeIcon={null}
|
closeIcon={null}
|
||||||
onCancel={() => handleCancel()}
|
onCancel={() => handleCancel()}
|
||||||
width={isMobile() ? '100%' : 600}
|
width={isMobile() ? '100%' : 600}
|
||||||
>
|
>
|
||||||
<Spin spinning={loading}>
|
<Spin spinning={loading}>
|
||||||
<Input
|
<Input
|
||||||
style={{ marginTop: 20 }}
|
style={{ marginTop: 20 }}
|
||||||
label='名称'
|
label="名称"
|
||||||
name='name'
|
name="name"
|
||||||
placeholder={'请输入名称'}
|
placeholder={'请输入名称'}
|
||||||
onChange={value => handleInputChange('name', value)}
|
onChange={value => handleInputChange('name', value)}
|
||||||
value={name}
|
value={name}
|
||||||
autoComplete='new-password'
|
autoComplete="new-password"
|
||||||
required={!isEdit}
|
required={!isEdit}
|
||||||
/>
|
/>
|
||||||
<Divider/>
|
<Divider />
|
||||||
<div style={{ marginTop: 20 }}>
|
<div style={{ marginTop: 20 }}>
|
||||||
<Typography.Text>{`额度${renderQuotaWithPrompt(quota)}`}</Typography.Text>
|
<Typography.Text>{`额度${renderQuotaWithPrompt(quota)}`}</Typography.Text>
|
||||||
</div>
|
</div>
|
||||||
<AutoComplete
|
<AutoComplete
|
||||||
style={{ marginTop: 8 }}
|
style={{ marginTop: 8 }}
|
||||||
name='quota'
|
name="quota"
|
||||||
placeholder={'请输入额度'}
|
placeholder={'请输入额度'}
|
||||||
onChange={(value) => handleInputChange('quota', value)}
|
onChange={(value) => handleInputChange('quota', value)}
|
||||||
value={quota}
|
value={quota}
|
||||||
autoComplete='new-password'
|
autoComplete="new-password"
|
||||||
type='number'
|
type="number"
|
||||||
position={'bottom'}
|
position={'bottom'}
|
||||||
data={[
|
data={[
|
||||||
{value: 500000, label: '1$'},
|
{ value: 500000, label: '1$' },
|
||||||
{value: 5000000, label: '10$'},
|
{ value: 5000000, label: '10$' },
|
||||||
{value: 25000000, label: '50$'},
|
{ value: 25000000, label: '50$' },
|
||||||
{value: 50000000, label: '100$'},
|
{ value: 50000000, label: '100$' },
|
||||||
{value: 250000000, label: '500$'},
|
{ value: 250000000, label: '500$' },
|
||||||
{value: 500000000, label: '1000$'},
|
{ value: 500000000, label: '1000$' }
|
||||||
]}
|
]}
|
||||||
/>
|
/>
|
||||||
{
|
{
|
||||||
!isEdit && <>
|
!isEdit && <>
|
||||||
<Divider/>
|
<Divider />
|
||||||
<Typography.Text>生成数量</Typography.Text>
|
<Typography.Text>生成数量</Typography.Text>
|
||||||
<Input
|
<Input
|
||||||
style={{ marginTop: 8 }}
|
style={{ marginTop: 8 }}
|
||||||
label='生成数量'
|
label="生成数量"
|
||||||
name='count'
|
name="count"
|
||||||
placeholder={'请输入生成数量'}
|
placeholder={'请输入生成数量'}
|
||||||
onChange={value => handleInputChange('count', value)}
|
onChange={value => handleInputChange('count', value)}
|
||||||
value={count}
|
value={count}
|
||||||
autoComplete='new-password'
|
autoComplete="new-password"
|
||||||
type='number'
|
type="number"
|
||||||
/>
|
/>
|
||||||
</>
|
</>
|
||||||
}
|
}
|
||||||
</Spin>
|
</Spin>
|
||||||
</SideSheet>
|
</SideSheet>
|
||||||
|
@ -1,352 +1,351 @@
|
|||||||
import React, {useEffect, useRef, useState} from 'react';
|
import React, { useEffect, useState } from 'react';
|
||||||
import {useParams, useNavigate} from 'react-router-dom';
|
import { useNavigate } from 'react-router-dom';
|
||||||
import {API, isMobile, showError, showSuccess, timestamp2string} from '../../helpers';
|
import { API, isMobile, showError, showSuccess, timestamp2string } from '../../helpers';
|
||||||
import {renderQuota, renderQuotaWithPrompt} from '../../helpers/render';
|
import { renderQuotaWithPrompt } from '../../helpers/render';
|
||||||
import {
|
import {
|
||||||
Layout,
|
AutoComplete,
|
||||||
SideSheet,
|
Banner,
|
||||||
Button,
|
Button,
|
||||||
|
Checkbox,
|
||||||
|
DatePicker,
|
||||||
|
Input,
|
||||||
|
Select,
|
||||||
|
SideSheet,
|
||||||
Space,
|
Space,
|
||||||
Spin,
|
Spin,
|
||||||
Banner,
|
Typography
|
||||||
Input,
|
} from '@douyinfe/semi-ui';
|
||||||
DatePicker,
|
import Title from '@douyinfe/semi-ui/lib/es/typography/title';
|
||||||
AutoComplete,
|
import { Divider } from 'semantic-ui-react';
|
||||||
Typography,
|
|
||||||
Checkbox, Select
|
|
||||||
} from "@douyinfe/semi-ui";
|
|
||||||
import Title from "@douyinfe/semi-ui/lib/es/typography/title";
|
|
||||||
import {Divider} from "semantic-ui-react";
|
|
||||||
|
|
||||||
const EditToken = (props) => {
|
const EditToken = (props) => {
|
||||||
const [isEdit, setIsEdit] = useState(false);
|
const [isEdit, setIsEdit] = useState(false);
|
||||||
const [loading, setLoading] = useState(isEdit);
|
const [loading, setLoading] = useState(isEdit);
|
||||||
const originInputs = {
|
const originInputs = {
|
||||||
name: '',
|
name: '',
|
||||||
remain_quota: isEdit ? 0 : 500000,
|
remain_quota: isEdit ? 0 : 500000,
|
||||||
expired_time: -1,
|
expired_time: -1,
|
||||||
unlimited_quota: false,
|
unlimited_quota: false,
|
||||||
model_limits_enabled: false,
|
model_limits_enabled: false,
|
||||||
model_limits: [],
|
model_limits: []
|
||||||
};
|
};
|
||||||
const [inputs, setInputs] = useState(originInputs);
|
const [inputs, setInputs] = useState(originInputs);
|
||||||
const {name, remain_quota, expired_time, unlimited_quota, model_limits_enabled, model_limits} = inputs;
|
const { name, remain_quota, expired_time, unlimited_quota, model_limits_enabled, model_limits } = inputs;
|
||||||
// const [visible, setVisible] = useState(false);
|
// const [visible, setVisible] = useState(false);
|
||||||
const [models, setModels] = useState({});
|
const [models, setModels] = useState({});
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
const handleInputChange = (name, value) => {
|
const handleInputChange = (name, value) => {
|
||||||
setInputs((inputs) => ({...inputs, [name]: value}));
|
setInputs((inputs) => ({ ...inputs, [name]: value }));
|
||||||
};
|
};
|
||||||
const handleCancel = () => {
|
const handleCancel = () => {
|
||||||
|
props.handleClose();
|
||||||
|
};
|
||||||
|
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 setUnlimitedQuota = () => {
|
||||||
|
setInputs({ ...inputs, unlimited_quota: !unlimited_quota });
|
||||||
|
};
|
||||||
|
|
||||||
|
const loadModels = async () => {
|
||||||
|
let res = await API.get(`/api/user/models`);
|
||||||
|
const { success, message, data } = res.data;
|
||||||
|
if (success) {
|
||||||
|
let localModelOptions = data.map((model) => ({
|
||||||
|
label: model,
|
||||||
|
value: model
|
||||||
|
}));
|
||||||
|
setModels(localModelOptions);
|
||||||
|
} else {
|
||||||
|
showError(message);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const loadToken = async () => {
|
||||||
|
setLoading(true);
|
||||||
|
let res = await API.get(`/api/token/${props.editingToken.id}`);
|
||||||
|
const { success, message, data } = res.data;
|
||||||
|
if (success) {
|
||||||
|
if (data.expired_time !== -1) {
|
||||||
|
data.expired_time = timestamp2string(data.expired_time);
|
||||||
|
}
|
||||||
|
if (data.model_limits !== '') {
|
||||||
|
data.model_limits = data.model_limits.split(',');
|
||||||
|
} else {
|
||||||
|
data.model_limits = [];
|
||||||
|
}
|
||||||
|
setInputs(data);
|
||||||
|
} else {
|
||||||
|
showError(message);
|
||||||
|
}
|
||||||
|
setLoading(false);
|
||||||
|
};
|
||||||
|
useEffect(() => {
|
||||||
|
setIsEdit(props.editingToken.id !== undefined);
|
||||||
|
}, [props.editingToken.id]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (!isEdit) {
|
||||||
|
setInputs(originInputs);
|
||||||
|
} else {
|
||||||
|
loadToken().then(
|
||||||
|
() => {
|
||||||
|
// console.log(inputs);
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
loadModels();
|
||||||
|
}, [isEdit]);
|
||||||
|
|
||||||
|
// 新增 state 变量 tokenCount 来记录用户想要创建的令牌数量,默认为 1
|
||||||
|
const [tokenCount, setTokenCount] = useState(1);
|
||||||
|
|
||||||
|
// 新增处理 tokenCount 变化的函数
|
||||||
|
const handleTokenCountChange = (value) => {
|
||||||
|
// 确保用户输入的是正整数
|
||||||
|
const count = parseInt(value, 10);
|
||||||
|
if (!isNaN(count) && count > 0) {
|
||||||
|
setTokenCount(count);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// 生成一个随机的四位字母数字字符串
|
||||||
|
const generateRandomSuffix = () => {
|
||||||
|
const characters = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789';
|
||||||
|
let result = '';
|
||||||
|
for (let i = 0; i < 6; i++) {
|
||||||
|
result += characters.charAt(Math.floor(Math.random() * characters.length));
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
};
|
||||||
|
|
||||||
|
const submit = async () => {
|
||||||
|
setLoading(true);
|
||||||
|
if (isEdit) {
|
||||||
|
// 编辑令牌的逻辑保持不变
|
||||||
|
let localInputs = { ...inputs };
|
||||||
|
localInputs.remain_quota = parseInt(localInputs.remain_quota);
|
||||||
|
if (localInputs.expired_time !== -1) {
|
||||||
|
let time = Date.parse(localInputs.expired_time);
|
||||||
|
if (isNaN(time)) {
|
||||||
|
showError('过期时间格式错误!');
|
||||||
|
setLoading(false);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
localInputs.expired_time = Math.ceil(time / 1000);
|
||||||
|
}
|
||||||
|
localInputs.model_limits = localInputs.model_limits.join(',');
|
||||||
|
let res = await API.put(`/api/token/`, { ...localInputs, id: parseInt(props.editingToken.id) });
|
||||||
|
const { success, message } = res.data;
|
||||||
|
if (success) {
|
||||||
|
showSuccess('令牌更新成功!');
|
||||||
|
props.refresh();
|
||||||
props.handleClose();
|
props.handleClose();
|
||||||
}
|
} else {
|
||||||
const setExpiredTime = (month, day, hour, minute) => {
|
showError(message);
|
||||||
let now = new Date();
|
}
|
||||||
let timestamp = now.getTime() / 1000;
|
} else {
|
||||||
let seconds = month * 30 * 24 * 60 * 60;
|
// 处理新增多个令牌的情况
|
||||||
seconds += day * 24 * 60 * 60;
|
let successCount = 0; // 记录成功创建的令牌数量
|
||||||
seconds += hour * 60 * 60;
|
for (let i = 0; i < tokenCount; i++) {
|
||||||
seconds += minute * 60;
|
let localInputs = { ...inputs };
|
||||||
if (seconds !== 0) {
|
if (i !== 0) {
|
||||||
timestamp += seconds;
|
// 如果用户想要创建多个令牌,则给每个令牌一个序号后缀
|
||||||
setInputs({...inputs, expired_time: timestamp2string(timestamp)});
|
localInputs.name = `${inputs.name}-${generateRandomSuffix()}`;
|
||||||
} else {
|
|
||||||
setInputs({...inputs, expired_time: -1});
|
|
||||||
}
|
}
|
||||||
};
|
localInputs.remain_quota = parseInt(localInputs.remain_quota);
|
||||||
|
|
||||||
const setUnlimitedQuota = () => {
|
if (localInputs.expired_time !== -1) {
|
||||||
setInputs({...inputs, unlimited_quota: !unlimited_quota});
|
let time = Date.parse(localInputs.expired_time);
|
||||||
};
|
if (isNaN(time)) {
|
||||||
|
showError('过期时间格式错误!');
|
||||||
|
setLoading(false);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
localInputs.expired_time = Math.ceil(time / 1000);
|
||||||
|
}
|
||||||
|
localInputs.model_limits = localInputs.model_limits.join(',');
|
||||||
|
let res = await API.post(`/api/token/`, localInputs);
|
||||||
|
const { success, message } = res.data;
|
||||||
|
|
||||||
const loadModels = async () => {
|
|
||||||
let res = await API.get(`/api/user/models`);
|
|
||||||
const {success, message, data} = res.data;
|
|
||||||
if (success) {
|
if (success) {
|
||||||
let localModelOptions = data.map((model) => ({
|
successCount++;
|
||||||
label: model,
|
|
||||||
value: model
|
|
||||||
}));
|
|
||||||
setModels(localModelOptions);
|
|
||||||
} else {
|
} else {
|
||||||
showError(message);
|
showError(message);
|
||||||
|
break; // 如果创建失败,终止循环
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (successCount > 0) {
|
||||||
|
showSuccess(`${successCount}个令牌创建成功,请在列表页面点击复制获取令牌!`);
|
||||||
|
props.refresh();
|
||||||
|
props.handleClose();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
setLoading(false);
|
||||||
|
setInputs(originInputs); // 重置表单
|
||||||
|
setTokenCount(1); // 重置数量为默认值
|
||||||
|
};
|
||||||
|
|
||||||
const loadToken = async () => {
|
|
||||||
setLoading(true);
|
return (
|
||||||
let res = await API.get(`/api/token/${props.editingToken.id}`);
|
<>
|
||||||
const {success, message, data} = res.data;
|
<SideSheet
|
||||||
if (success) {
|
placement={isEdit ? 'right' : 'left'}
|
||||||
if (data.expired_time !== -1) {
|
title={<Title level={3}>{isEdit ? '更新令牌信息' : '创建新的令牌'}</Title>}
|
||||||
data.expired_time = timestamp2string(data.expired_time);
|
headerStyle={{ borderBottom: '1px solid var(--semi-color-border)' }}
|
||||||
}
|
bodyStyle={{ borderBottom: '1px solid var(--semi-color-border)' }}
|
||||||
if (data.model_limits !== '') {
|
visible={props.visiable}
|
||||||
data.model_limits = data.model_limits.split(',');
|
footer={
|
||||||
} else {
|
<div style={{ display: 'flex', justifyContent: 'flex-end' }}>
|
||||||
data.model_limits = [];
|
<Space>
|
||||||
}
|
<Button theme="solid" size={'large'} onClick={submit}>提交</Button>
|
||||||
setInputs(data);
|
<Button theme="solid" size={'large'} type={'tertiary'} onClick={handleCancel}>取消</Button>
|
||||||
} else {
|
</Space>
|
||||||
showError(message);
|
</div>
|
||||||
}
|
}
|
||||||
setLoading(false);
|
closeIcon={null}
|
||||||
};
|
onCancel={() => handleCancel()}
|
||||||
useEffect(() => {
|
width={isMobile() ? '100%' : 600}
|
||||||
setIsEdit(props.editingToken.id !== undefined);
|
>
|
||||||
}, [props.editingToken.id]);
|
<Spin spinning={loading}>
|
||||||
|
<Input
|
||||||
|
style={{ marginTop: 20 }}
|
||||||
|
label="名称"
|
||||||
|
name="name"
|
||||||
|
placeholder={'请输入名称'}
|
||||||
|
onChange={(value) => handleInputChange('name', value)}
|
||||||
|
value={name}
|
||||||
|
autoComplete="new-password"
|
||||||
|
required={!isEdit}
|
||||||
|
/>
|
||||||
|
<Divider />
|
||||||
|
<DatePicker
|
||||||
|
label="过期时间"
|
||||||
|
name="expired_time"
|
||||||
|
placeholder={'请选择过期时间'}
|
||||||
|
onChange={(value) => handleInputChange('expired_time', value)}
|
||||||
|
value={expired_time}
|
||||||
|
autoComplete="new-password"
|
||||||
|
type="dateTime"
|
||||||
|
/>
|
||||||
|
<div style={{ marginTop: 20 }}>
|
||||||
|
<Space>
|
||||||
|
<Button type={'tertiary'} onClick={() => {
|
||||||
|
setExpiredTime(0, 0, 0, 0);
|
||||||
|
}}>永不过期</Button>
|
||||||
|
<Button type={'tertiary'} onClick={() => {
|
||||||
|
setExpiredTime(0, 0, 1, 0);
|
||||||
|
}}>一小时</Button>
|
||||||
|
<Button type={'tertiary'} onClick={() => {
|
||||||
|
setExpiredTime(1, 0, 0, 0);
|
||||||
|
}}>一个月</Button>
|
||||||
|
<Button type={'tertiary'} onClick={() => {
|
||||||
|
setExpiredTime(0, 1, 0, 0);
|
||||||
|
}}>一天</Button>
|
||||||
|
</Space>
|
||||||
|
</div>
|
||||||
|
|
||||||
useEffect(() => {
|
<Divider />
|
||||||
if (!isEdit) {
|
<Banner type={'warning'}
|
||||||
setInputs(originInputs);
|
description={'注意,令牌的额度仅用于限制令牌本身的最大额度使用量,实际的使用受到账户的剩余额度限制。'}></Banner>
|
||||||
} else {
|
<div style={{ marginTop: 20 }}>
|
||||||
loadToken().then(
|
<Typography.Text>{`额度${renderQuotaWithPrompt(remain_quota)}`}</Typography.Text>
|
||||||
() => {
|
</div>
|
||||||
// console.log(inputs);
|
<AutoComplete
|
||||||
}
|
style={{ marginTop: 8 }}
|
||||||
);
|
name="remain_quota"
|
||||||
}
|
placeholder={'请输入额度'}
|
||||||
loadModels();
|
onChange={(value) => handleInputChange('remain_quota', value)}
|
||||||
}, [isEdit]);
|
value={remain_quota}
|
||||||
|
autoComplete="new-password"
|
||||||
|
type="number"
|
||||||
|
// position={'top'}
|
||||||
|
data={[
|
||||||
|
{ value: 500000, label: '1$' },
|
||||||
|
{ value: 5000000, label: '10$' },
|
||||||
|
{ value: 25000000, label: '50$' },
|
||||||
|
{ value: 50000000, label: '100$' },
|
||||||
|
{ value: 250000000, label: '500$' },
|
||||||
|
{ value: 500000000, label: '1000$' }
|
||||||
|
]}
|
||||||
|
disabled={unlimited_quota}
|
||||||
|
/>
|
||||||
|
|
||||||
// 新增 state 变量 tokenCount 来记录用户想要创建的令牌数量,默认为 1
|
{!isEdit && (
|
||||||
const [tokenCount, setTokenCount] = useState(1);
|
<>
|
||||||
|
<div style={{ marginTop: 20 }}>
|
||||||
|
<Typography.Text>新建数量</Typography.Text>
|
||||||
|
</div>
|
||||||
|
<AutoComplete
|
||||||
|
style={{ marginTop: 8 }}
|
||||||
|
label="数量"
|
||||||
|
placeholder={'请选择或输入创建令牌的数量'}
|
||||||
|
onChange={(value) => handleTokenCountChange(value)}
|
||||||
|
onSelect={(value) => handleTokenCountChange(value)}
|
||||||
|
value={tokenCount.toString()}
|
||||||
|
autoComplete="off"
|
||||||
|
type="number"
|
||||||
|
data={[
|
||||||
|
{ value: 10, label: '10个' },
|
||||||
|
{ value: 20, label: '20个' },
|
||||||
|
{ value: 30, label: '30个' },
|
||||||
|
{ value: 100, label: '100个' }
|
||||||
|
]}
|
||||||
|
disabled={unlimited_quota}
|
||||||
|
/>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
|
||||||
// 新增处理 tokenCount 变化的函数
|
<div>
|
||||||
const handleTokenCountChange = (value) => {
|
<Button style={{ marginTop: 8 }} type={'warning'} onClick={() => {
|
||||||
// 确保用户输入的是正整数
|
setUnlimitedQuota();
|
||||||
const count = parseInt(value, 10);
|
}}>{unlimited_quota ? '取消无限额度' : '设为无限额度'}</Button>
|
||||||
if (!isNaN(count) && count > 0) {
|
</div>
|
||||||
setTokenCount(count);
|
<Divider />
|
||||||
}
|
<div style={{ marginTop: 10, display: 'flex' }}>
|
||||||
};
|
<Space>
|
||||||
|
<Checkbox
|
||||||
|
name="model_limits_enabled"
|
||||||
|
checked={model_limits_enabled}
|
||||||
|
onChange={(e) => handleInputChange('model_limits_enabled', e.target.checked)}
|
||||||
|
>
|
||||||
|
</Checkbox>
|
||||||
|
<Typography.Text>启用模型限制(非必要,不建议启用)</Typography.Text>
|
||||||
|
</Space>
|
||||||
|
</div>
|
||||||
|
|
||||||
// 生成一个随机的四位字母数字字符串
|
<Select
|
||||||
const generateRandomSuffix = () => {
|
style={{ marginTop: 8 }}
|
||||||
const characters = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789';
|
placeholder={'请选择该渠道所支持的模型'}
|
||||||
let result = '';
|
name="models"
|
||||||
for (let i = 0; i < 6; i++) {
|
required
|
||||||
result += characters.charAt(Math.floor(Math.random() * characters.length));
|
multiple
|
||||||
}
|
selection
|
||||||
return result;
|
onChange={value => {
|
||||||
};
|
handleInputChange('model_limits', value);
|
||||||
|
}}
|
||||||
const submit = async () => {
|
value={inputs.model_limits}
|
||||||
setLoading(true);
|
autoComplete="new-password"
|
||||||
if (isEdit) {
|
optionList={models}
|
||||||
// 编辑令牌的逻辑保持不变
|
disabled={!model_limits_enabled}
|
||||||
let localInputs = {...inputs};
|
/>
|
||||||
localInputs.remain_quota = parseInt(localInputs.remain_quota);
|
</Spin>
|
||||||
if (localInputs.expired_time !== -1) {
|
</SideSheet>
|
||||||
let time = Date.parse(localInputs.expired_time);
|
</>
|
||||||
if (isNaN(time)) {
|
);
|
||||||
showError('过期时间格式错误!');
|
|
||||||
setLoading(false);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
localInputs.expired_time = Math.ceil(time / 1000);
|
|
||||||
}
|
|
||||||
localInputs.model_limits = localInputs.model_limits.join(',');
|
|
||||||
let res = await API.put(`/api/token/`, {...localInputs, id: parseInt(props.editingToken.id)});
|
|
||||||
const {success, message} = res.data;
|
|
||||||
if (success) {
|
|
||||||
showSuccess('令牌更新成功!');
|
|
||||||
props.refresh();
|
|
||||||
props.handleClose();
|
|
||||||
} else {
|
|
||||||
showError(message);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// 处理新增多个令牌的情况
|
|
||||||
let successCount = 0; // 记录成功创建的令牌数量
|
|
||||||
for (let i = 0; i < tokenCount; i++) {
|
|
||||||
let localInputs = {...inputs};
|
|
||||||
if (i !== 0) {
|
|
||||||
// 如果用户想要创建多个令牌,则给每个令牌一个序号后缀
|
|
||||||
localInputs.name = `${inputs.name}-${generateRandomSuffix()}`;
|
|
||||||
}
|
|
||||||
localInputs.remain_quota = parseInt(localInputs.remain_quota);
|
|
||||||
|
|
||||||
if (localInputs.expired_time !== -1) {
|
|
||||||
let time = Date.parse(localInputs.expired_time);
|
|
||||||
if (isNaN(time)) {
|
|
||||||
showError('过期时间格式错误!');
|
|
||||||
setLoading(false);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
localInputs.expired_time = Math.ceil(time / 1000);
|
|
||||||
}
|
|
||||||
localInputs.model_limits = localInputs.model_limits.join(',');
|
|
||||||
let res = await API.post(`/api/token/`, localInputs);
|
|
||||||
const {success, message} = res.data;
|
|
||||||
|
|
||||||
if (success) {
|
|
||||||
successCount++;
|
|
||||||
} else {
|
|
||||||
showError(message);
|
|
||||||
break; // 如果创建失败,终止循环
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (successCount > 0) {
|
|
||||||
showSuccess(`${successCount}个令牌创建成功,请在列表页面点击复制获取令牌!`);
|
|
||||||
props.refresh();
|
|
||||||
props.handleClose();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
setLoading(false);
|
|
||||||
setInputs(originInputs); // 重置表单
|
|
||||||
setTokenCount(1); // 重置数量为默认值
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
return (
|
|
||||||
<>
|
|
||||||
<SideSheet
|
|
||||||
placement={isEdit ? 'right' : 'left'}
|
|
||||||
title={<Title level={3}>{isEdit ? '更新令牌信息' : '创建新的令牌'}</Title>}
|
|
||||||
headerStyle={{borderBottom: '1px solid var(--semi-color-border)'}}
|
|
||||||
bodyStyle={{borderBottom: '1px solid var(--semi-color-border)'}}
|
|
||||||
visible={props.visiable}
|
|
||||||
footer={
|
|
||||||
<div style={{display: 'flex', justifyContent: 'flex-end'}}>
|
|
||||||
<Space>
|
|
||||||
<Button theme='solid' size={'large'} onClick={submit}>提交</Button>
|
|
||||||
<Button theme='solid' size={'large'} type={'tertiary'} onClick={handleCancel}>取消</Button>
|
|
||||||
</Space>
|
|
||||||
</div>
|
|
||||||
}
|
|
||||||
closeIcon={null}
|
|
||||||
onCancel={() => handleCancel()}
|
|
||||||
width={isMobile() ? '100%' : 600}
|
|
||||||
>
|
|
||||||
<Spin spinning={loading}>
|
|
||||||
<Input
|
|
||||||
style={{marginTop: 20}}
|
|
||||||
label='名称'
|
|
||||||
name='name'
|
|
||||||
placeholder={'请输入名称'}
|
|
||||||
onChange={(value) => handleInputChange('name', value)}
|
|
||||||
value={name}
|
|
||||||
autoComplete='new-password'
|
|
||||||
required={!isEdit}
|
|
||||||
/>
|
|
||||||
<Divider/>
|
|
||||||
<DatePicker
|
|
||||||
label='过期时间'
|
|
||||||
name='expired_time'
|
|
||||||
placeholder={'请选择过期时间'}
|
|
||||||
onChange={(value) => handleInputChange('expired_time', value)}
|
|
||||||
value={expired_time}
|
|
||||||
autoComplete='new-password'
|
|
||||||
type='dateTime'
|
|
||||||
/>
|
|
||||||
<div style={{marginTop: 20}}>
|
|
||||||
<Space>
|
|
||||||
<Button type={'tertiary'} onClick={() => {
|
|
||||||
setExpiredTime(0, 0, 0, 0);
|
|
||||||
}}>永不过期</Button>
|
|
||||||
<Button type={'tertiary'} onClick={() => {
|
|
||||||
setExpiredTime(0, 0, 1, 0);
|
|
||||||
}}>一小时</Button>
|
|
||||||
<Button type={'tertiary'} onClick={() => {
|
|
||||||
setExpiredTime(1, 0, 0, 0);
|
|
||||||
}}>一个月</Button>
|
|
||||||
<Button type={'tertiary'} onClick={() => {
|
|
||||||
setExpiredTime(0, 1, 0, 0);
|
|
||||||
}}>一天</Button>
|
|
||||||
</Space>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<Divider/>
|
|
||||||
<Banner type={'warning'}
|
|
||||||
description={'注意,令牌的额度仅用于限制令牌本身的最大额度使用量,实际的使用受到账户的剩余额度限制。'}></Banner>
|
|
||||||
<div style={{marginTop: 20}}>
|
|
||||||
<Typography.Text>{`额度${renderQuotaWithPrompt(remain_quota)}`}</Typography.Text>
|
|
||||||
</div>
|
|
||||||
<AutoComplete
|
|
||||||
style={{marginTop: 8}}
|
|
||||||
name='remain_quota'
|
|
||||||
placeholder={'请输入额度'}
|
|
||||||
onChange={(value) => handleInputChange('remain_quota', value)}
|
|
||||||
value={remain_quota}
|
|
||||||
autoComplete='new-password'
|
|
||||||
type='number'
|
|
||||||
// position={'top'}
|
|
||||||
data={[
|
|
||||||
{value: 500000, label: '1$'},
|
|
||||||
{value: 5000000, label: '10$'},
|
|
||||||
{value: 25000000, label: '50$'},
|
|
||||||
{value: 50000000, label: '100$'},
|
|
||||||
{value: 250000000, label: '500$'},
|
|
||||||
{value: 500000000, label: '1000$'},
|
|
||||||
]}
|
|
||||||
disabled={unlimited_quota}
|
|
||||||
/>
|
|
||||||
|
|
||||||
{!isEdit && (
|
|
||||||
<>
|
|
||||||
<div style={{marginTop: 20}}>
|
|
||||||
<Typography.Text>新建数量</Typography.Text>
|
|
||||||
</div>
|
|
||||||
<AutoComplete
|
|
||||||
style={{ marginTop: 8 }}
|
|
||||||
label='数量'
|
|
||||||
placeholder={'请选择或输入创建令牌的数量'}
|
|
||||||
onChange={(value) => handleTokenCountChange(value)}
|
|
||||||
onSelect={(value) => handleTokenCountChange(value)}
|
|
||||||
value={tokenCount.toString()}
|
|
||||||
autoComplete='off'
|
|
||||||
type='number'
|
|
||||||
data={[
|
|
||||||
{ value: 10, label: '10个' },
|
|
||||||
{ value: 20, label: '20个' },
|
|
||||||
{ value: 30, label: '30个' },
|
|
||||||
{ value: 100, label: '100个' },
|
|
||||||
]}
|
|
||||||
disabled={unlimited_quota}
|
|
||||||
/>
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
|
|
||||||
<div>
|
|
||||||
<Button style={{marginTop: 8}} type={'warning'} onClick={() => {
|
|
||||||
setUnlimitedQuota();
|
|
||||||
}}>{unlimited_quota ? '取消无限额度' : '设为无限额度'}</Button>
|
|
||||||
</div>
|
|
||||||
<Divider/>
|
|
||||||
<div style={{marginTop: 10, display: 'flex'}}>
|
|
||||||
<Space>
|
|
||||||
<Checkbox
|
|
||||||
name='model_limits_enabled'
|
|
||||||
checked={model_limits_enabled}
|
|
||||||
onChange={(e) => handleInputChange('model_limits_enabled', e.target.checked)}
|
|
||||||
>
|
|
||||||
</Checkbox>
|
|
||||||
<Typography.Text>启用模型限制(非必要,不建议启用)</Typography.Text>
|
|
||||||
</Space>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<Select
|
|
||||||
style={{marginTop: 8}}
|
|
||||||
placeholder={'请选择该渠道所支持的模型'}
|
|
||||||
name='models'
|
|
||||||
required
|
|
||||||
multiple
|
|
||||||
selection
|
|
||||||
onChange={value => {
|
|
||||||
handleInputChange('model_limits', value);
|
|
||||||
}}
|
|
||||||
value={inputs.model_limits}
|
|
||||||
autoComplete='new-password'
|
|
||||||
optionList={models}
|
|
||||||
disabled={!model_limits_enabled}
|
|
||||||
/>
|
|
||||||
</Spin>
|
|
||||||
</SideSheet>
|
|
||||||
</>
|
|
||||||
);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
export default EditToken;
|
export default EditToken;
|
||||||
|
@ -1,98 +1,98 @@
|
|||||||
import React, {useState} from 'react';
|
import React, { useState } from 'react';
|
||||||
import {API, isMobile, showError, showSuccess} from '../../helpers';
|
import { API, isMobile, showError, showSuccess } from '../../helpers';
|
||||||
import Title from "@douyinfe/semi-ui/lib/es/typography/title";
|
import Title from '@douyinfe/semi-ui/lib/es/typography/title';
|
||||||
import {Button, SideSheet, Space, Input, Spin} from "@douyinfe/semi-ui";
|
import { Button, Input, SideSheet, Space, Spin } from '@douyinfe/semi-ui';
|
||||||
|
|
||||||
const AddUser = (props) => {
|
const AddUser = (props) => {
|
||||||
const originInputs = {
|
const originInputs = {
|
||||||
username: '',
|
username: '',
|
||||||
display_name: '',
|
display_name: '',
|
||||||
password: '',
|
password: ''
|
||||||
};
|
};
|
||||||
const [inputs, setInputs] = useState(originInputs);
|
const [inputs, setInputs] = useState(originInputs);
|
||||||
const [loading, setLoading] = useState(false);
|
const [loading, setLoading] = useState(false);
|
||||||
const {username, display_name, password} = inputs;
|
const { username, display_name, password } = inputs;
|
||||||
|
|
||||||
const handleInputChange = (name, value) => {
|
const handleInputChange = (name, value) => {
|
||||||
setInputs((inputs) => ({...inputs, [name]: value}));
|
setInputs((inputs) => ({ ...inputs, [name]: value }));
|
||||||
};
|
};
|
||||||
|
|
||||||
const submit = async () => {
|
const submit = async () => {
|
||||||
setLoading(true);
|
setLoading(true);
|
||||||
if (inputs.username === '' || inputs.password === '') return;
|
if (inputs.username === '' || inputs.password === '') return;
|
||||||
const res = await API.post(`/api/user/`, inputs);
|
const res = await API.post(`/api/user/`, inputs);
|
||||||
const {success, message} = res.data;
|
const { success, message } = res.data;
|
||||||
if (success) {
|
if (success) {
|
||||||
showSuccess('用户账户创建成功!');
|
showSuccess('用户账户创建成功!');
|
||||||
setInputs(originInputs);
|
setInputs(originInputs);
|
||||||
props.refresh();
|
props.refresh();
|
||||||
props.handleClose();
|
props.handleClose();
|
||||||
} else {
|
} else {
|
||||||
showError(message);
|
showError(message);
|
||||||
}
|
|
||||||
setLoading(false);
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleCancel = () => {
|
|
||||||
props.handleClose();
|
|
||||||
}
|
}
|
||||||
|
setLoading(false);
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
const handleCancel = () => {
|
||||||
<>
|
props.handleClose();
|
||||||
<SideSheet
|
};
|
||||||
placement={'left'}
|
|
||||||
title={<Title level={3}>{'添加用户'}</Title>}
|
return (
|
||||||
headerStyle={{borderBottom: '1px solid var(--semi-color-border)'}}
|
<>
|
||||||
bodyStyle={{borderBottom: '1px solid var(--semi-color-border)'}}
|
<SideSheet
|
||||||
visible={props.visible}
|
placement={'left'}
|
||||||
footer={
|
title={<Title level={3}>{'添加用户'}</Title>}
|
||||||
<div style={{display: 'flex', justifyContent: 'flex-end'}}>
|
headerStyle={{ borderBottom: '1px solid var(--semi-color-border)' }}
|
||||||
<Space>
|
bodyStyle={{ borderBottom: '1px solid var(--semi-color-border)' }}
|
||||||
<Button theme='solid' size={'large'} onClick={submit}>提交</Button>
|
visible={props.visible}
|
||||||
<Button theme='solid' size={'large'} type={'tertiary'} onClick={handleCancel}>取消</Button>
|
footer={
|
||||||
</Space>
|
<div style={{ display: 'flex', justifyContent: 'flex-end' }}>
|
||||||
</div>
|
<Space>
|
||||||
}
|
<Button theme="solid" size={'large'} onClick={submit}>提交</Button>
|
||||||
closeIcon={null}
|
<Button theme="solid" size={'large'} type={'tertiary'} onClick={handleCancel}>取消</Button>
|
||||||
onCancel={() => handleCancel()}
|
</Space>
|
||||||
width={isMobile() ? '100%' : 600}
|
</div>
|
||||||
>
|
}
|
||||||
<Spin spinning={loading}>
|
closeIcon={null}
|
||||||
<Input
|
onCancel={() => handleCancel()}
|
||||||
style={{marginTop: 20}}
|
width={isMobile() ? '100%' : 600}
|
||||||
label="用户名"
|
>
|
||||||
name="username"
|
<Spin spinning={loading}>
|
||||||
addonBefore={'用户名'}
|
<Input
|
||||||
placeholder={'请输入用户名'}
|
style={{ marginTop: 20 }}
|
||||||
onChange={value => handleInputChange('username', value)}
|
label="用户名"
|
||||||
value={username}
|
name="username"
|
||||||
autoComplete="off"
|
addonBefore={'用户名'}
|
||||||
/>
|
placeholder={'请输入用户名'}
|
||||||
<Input
|
onChange={value => handleInputChange('username', value)}
|
||||||
style={{marginTop: 20}}
|
value={username}
|
||||||
addonBefore={'显示名'}
|
autoComplete="off"
|
||||||
label="显示名称"
|
/>
|
||||||
name="display_name"
|
<Input
|
||||||
autoComplete="off"
|
style={{ marginTop: 20 }}
|
||||||
placeholder={'请输入显示名称'}
|
addonBefore={'显示名'}
|
||||||
onChange={value => handleInputChange('display_name', value)}
|
label="显示名称"
|
||||||
value={display_name}
|
name="display_name"
|
||||||
/>
|
autoComplete="off"
|
||||||
<Input
|
placeholder={'请输入显示名称'}
|
||||||
style={{marginTop: 20}}
|
onChange={value => handleInputChange('display_name', value)}
|
||||||
label="密 码"
|
value={display_name}
|
||||||
name="password"
|
/>
|
||||||
type={'password'}
|
<Input
|
||||||
addonBefore={'密码'}
|
style={{ marginTop: 20 }}
|
||||||
placeholder={'请输入密码'}
|
label="密 码"
|
||||||
onChange={value => handleInputChange('password', value)}
|
name="password"
|
||||||
value={password}
|
type={'password'}
|
||||||
autoComplete="off"
|
addonBefore={'密码'}
|
||||||
/>
|
placeholder={'请输入密码'}
|
||||||
</Spin>
|
onChange={value => handleInputChange('password', value)}
|
||||||
</SideSheet>
|
value={password}
|
||||||
</>
|
autoComplete="off"
|
||||||
);
|
/>
|
||||||
|
</Spin>
|
||||||
|
</SideSheet>
|
||||||
|
</>
|
||||||
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export default AddUser;
|
export default AddUser;
|
||||||
|
@ -1,9 +1,9 @@
|
|||||||
import React, { useEffect, useState } from 'react';
|
import React, { useEffect, useState } from 'react';
|
||||||
import { useParams, useNavigate } from 'react-router-dom';
|
import { useNavigate } from 'react-router-dom';
|
||||||
import { API, isMobile, showError, showSuccess } from '../../helpers';
|
import { API, isMobile, showError, showSuccess } from '../../helpers';
|
||||||
import { renderQuota, renderQuotaWithPrompt } from '../../helpers/render';
|
import { renderQuotaWithPrompt } from '../../helpers/render';
|
||||||
import Title from "@douyinfe/semi-ui/lib/es/typography/title";
|
import Title from '@douyinfe/semi-ui/lib/es/typography/title';
|
||||||
import { SideSheet, Space, Button, Spin, Input, Typography, Select, Divider } from "@douyinfe/semi-ui";
|
import { Button, Divider, Input, Select, SideSheet, Space, Spin, Typography } from '@douyinfe/semi-ui';
|
||||||
|
|
||||||
const EditUser = (props) => {
|
const EditUser = (props) => {
|
||||||
const userId = props.editingUser.id;
|
const userId = props.editingUser.id;
|
||||||
@ -30,7 +30,7 @@ const EditUser = (props) => {
|
|||||||
let res = await API.get(`/api/group/`);
|
let res = await API.get(`/api/group/`);
|
||||||
setGroupOptions(res.data.data.map((group) => ({
|
setGroupOptions(res.data.data.map((group) => ({
|
||||||
label: group,
|
label: group,
|
||||||
value: group,
|
value: group
|
||||||
})));
|
})));
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
showError(error.message);
|
showError(error.message);
|
||||||
@ -39,7 +39,7 @@ const EditUser = (props) => {
|
|||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
const handleCancel = () => {
|
const handleCancel = () => {
|
||||||
props.handleClose();
|
props.handleClose();
|
||||||
}
|
};
|
||||||
const loadUser = async () => {
|
const loadUser = async () => {
|
||||||
setLoading(true);
|
setLoading(true);
|
||||||
let res = undefined;
|
let res = undefined;
|
||||||
@ -99,8 +99,8 @@ const EditUser = (props) => {
|
|||||||
footer={
|
footer={
|
||||||
<div style={{ display: 'flex', justifyContent: 'flex-end' }}>
|
<div style={{ display: 'flex', justifyContent: 'flex-end' }}>
|
||||||
<Space>
|
<Space>
|
||||||
<Button theme='solid' size={'large'} onClick={submit}>提交</Button>
|
<Button theme="solid" size={'large'} onClick={submit}>提交</Button>
|
||||||
<Button theme='solid' size={'large'} type={'tertiary'} onClick={handleCancel}>取消</Button>
|
<Button theme="solid" size={'large'} type={'tertiary'} onClick={handleCancel}>取消</Button>
|
||||||
</Space>
|
</Space>
|
||||||
</div>
|
</div>
|
||||||
}
|
}
|
||||||
@ -113,35 +113,35 @@ const EditUser = (props) => {
|
|||||||
<Typography.Text>用户名</Typography.Text>
|
<Typography.Text>用户名</Typography.Text>
|
||||||
</div>
|
</div>
|
||||||
<Input
|
<Input
|
||||||
label='用户名'
|
label="用户名"
|
||||||
name='username'
|
name="username"
|
||||||
placeholder={'请输入新的用户名'}
|
placeholder={'请输入新的用户名'}
|
||||||
onChange={value => handleInputChange('username', value)}
|
onChange={value => handleInputChange('username', value)}
|
||||||
value={username}
|
value={username}
|
||||||
autoComplete='new-password'
|
autoComplete="new-password"
|
||||||
/>
|
/>
|
||||||
<div style={{ marginTop: 20 }}>
|
<div style={{ marginTop: 20 }}>
|
||||||
<Typography.Text>密码</Typography.Text>
|
<Typography.Text>密码</Typography.Text>
|
||||||
</div>
|
</div>
|
||||||
<Input
|
<Input
|
||||||
label='密码'
|
label="密码"
|
||||||
name='password'
|
name="password"
|
||||||
type={'password'}
|
type={'password'}
|
||||||
placeholder={'请输入新的密码,最短 8 位'}
|
placeholder={'请输入新的密码,最短 8 位'}
|
||||||
onChange={value => handleInputChange('password', value)}
|
onChange={value => handleInputChange('password', value)}
|
||||||
value={password}
|
value={password}
|
||||||
autoComplete='new-password'
|
autoComplete="new-password"
|
||||||
/>
|
/>
|
||||||
<div style={{ marginTop: 20 }}>
|
<div style={{ marginTop: 20 }}>
|
||||||
<Typography.Text>显示名称</Typography.Text>
|
<Typography.Text>显示名称</Typography.Text>
|
||||||
</div>
|
</div>
|
||||||
<Input
|
<Input
|
||||||
label='显示名称'
|
label="显示名称"
|
||||||
name='display_name'
|
name="display_name"
|
||||||
placeholder={'请输入新的显示名称'}
|
placeholder={'请输入新的显示名称'}
|
||||||
onChange={value => handleInputChange('display_name', value)}
|
onChange={value => handleInputChange('display_name', value)}
|
||||||
value={display_name}
|
value={display_name}
|
||||||
autoComplete='new-password'
|
autoComplete="new-password"
|
||||||
/>
|
/>
|
||||||
{
|
{
|
||||||
userId && <>
|
userId && <>
|
||||||
@ -150,7 +150,7 @@ const EditUser = (props) => {
|
|||||||
</div>
|
</div>
|
||||||
<Select
|
<Select
|
||||||
placeholder={'请选择分组'}
|
placeholder={'请选择分组'}
|
||||||
name='group'
|
name="group"
|
||||||
fluid
|
fluid
|
||||||
search
|
search
|
||||||
selection
|
selection
|
||||||
@ -158,19 +158,19 @@ const EditUser = (props) => {
|
|||||||
additionLabel={'请在系统设置页面编辑分组倍率以添加新的分组:'}
|
additionLabel={'请在系统设置页面编辑分组倍率以添加新的分组:'}
|
||||||
onChange={value => handleInputChange('group', value)}
|
onChange={value => handleInputChange('group', value)}
|
||||||
value={inputs.group}
|
value={inputs.group}
|
||||||
autoComplete='new-password'
|
autoComplete="new-password"
|
||||||
optionList={groupOptions}
|
optionList={groupOptions}
|
||||||
/>
|
/>
|
||||||
<div style={{ marginTop: 20 }}>
|
<div style={{ marginTop: 20 }}>
|
||||||
<Typography.Text>{`剩余额度${renderQuotaWithPrompt(quota)}`}</Typography.Text>
|
<Typography.Text>{`剩余额度${renderQuotaWithPrompt(quota)}`}</Typography.Text>
|
||||||
</div>
|
</div>
|
||||||
<Input
|
<Input
|
||||||
name='quota'
|
name="quota"
|
||||||
placeholder={'请输入新的剩余额度'}
|
placeholder={'请输入新的剩余额度'}
|
||||||
onChange={value => handleInputChange('quota', value)}
|
onChange={value => handleInputChange('quota', value)}
|
||||||
value={quota}
|
value={quota}
|
||||||
type={'number'}
|
type={'number'}
|
||||||
autoComplete='new-password'
|
autoComplete="new-password"
|
||||||
/>
|
/>
|
||||||
</>
|
</>
|
||||||
}
|
}
|
||||||
@ -179,10 +179,10 @@ const EditUser = (props) => {
|
|||||||
<Typography.Text>已绑定的 GitHub 账户</Typography.Text>
|
<Typography.Text>已绑定的 GitHub 账户</Typography.Text>
|
||||||
</div>
|
</div>
|
||||||
<Input
|
<Input
|
||||||
name='github_id'
|
name="github_id"
|
||||||
value={github_id}
|
value={github_id}
|
||||||
autoComplete='new-password'
|
autoComplete="new-password"
|
||||||
placeholder='此项只读,需要用户通过个人设置页面的相关绑定按钮进行绑定,不可直接修改'
|
placeholder="此项只读,需要用户通过个人设置页面的相关绑定按钮进行绑定,不可直接修改"
|
||||||
readonly
|
readonly
|
||||||
/>
|
/>
|
||||||
<div style={{ marginTop: 20 }}>
|
<div style={{ marginTop: 20 }}>
|
||||||
@ -199,30 +199,30 @@ const EditUser = (props) => {
|
|||||||
<Typography.Text>已绑定的微信账户</Typography.Text>
|
<Typography.Text>已绑定的微信账户</Typography.Text>
|
||||||
</div>
|
</div>
|
||||||
<Input
|
<Input
|
||||||
name='wechat_id'
|
name="wechat_id"
|
||||||
value={wechat_id}
|
value={wechat_id}
|
||||||
autoComplete='new-password'
|
autoComplete="new-password"
|
||||||
placeholder='此项只读,需要用户通过个人设置页面的相关绑定按钮进行绑定,不可直接修改'
|
placeholder="此项只读,需要用户通过个人设置页面的相关绑定按钮进行绑定,不可直接修改"
|
||||||
readonly
|
readonly
|
||||||
/>
|
/>
|
||||||
<div style={{ marginTop: 20 }}>
|
<div style={{ marginTop: 20 }}>
|
||||||
<Typography.Text>已绑定的 Telegram 账户</Typography.Text>
|
<Typography.Text>已绑定的 Telegram 账户</Typography.Text>
|
||||||
</div>
|
</div>
|
||||||
<Input
|
<Input
|
||||||
name='telegram_id'
|
name="telegram_id"
|
||||||
value={telegram_id}
|
value={telegram_id}
|
||||||
autoComplete='new-password'
|
autoComplete="new-password"
|
||||||
placeholder='此项只读,需要用户通过个人设置页面的相关绑定按钮进行绑定,不可直接修改'
|
placeholder="此项只读,需要用户通过个人设置页面的相关绑定按钮进行绑定,不可直接修改"
|
||||||
readonly
|
readonly
|
||||||
/>
|
/>
|
||||||
<div style={{ marginTop: 20 }}>
|
<div style={{ marginTop: 20 }}>
|
||||||
<Typography.Text>已绑定的邮箱账户</Typography.Text>
|
<Typography.Text>已绑定的邮箱账户</Typography.Text>
|
||||||
</div>
|
</div>
|
||||||
<Input
|
<Input
|
||||||
name='email'
|
name="email"
|
||||||
value={email}
|
value={email}
|
||||||
autoComplete='new-password'
|
autoComplete="new-password"
|
||||||
placeholder='此项只读,需要用户通过个人设置页面的相关绑定按钮进行绑定,不可直接修改'
|
placeholder="此项只读,需要用户通过个人设置页面的相关绑定按钮进行绑定,不可直接修改"
|
||||||
readonly
|
readonly
|
||||||
/>
|
/>
|
||||||
</Spin>
|
</Spin>
|
||||||
|
Loading…
Reference in New Issue
Block a user