|
|
package controller |
|
|
|
|
|
import ( |
|
|
"net/http" |
|
|
"strconv" |
|
|
"strings" |
|
|
|
|
|
"github.com/QuantumNous/new-api/common" |
|
|
"github.com/QuantumNous/new-api/model" |
|
|
|
|
|
"github.com/gin-gonic/gin" |
|
|
) |
|
|
|
|
|
func GetAllTokens(c *gin.Context) { |
|
|
userId := c.GetInt("id") |
|
|
pageInfo := common.GetPageQuery(c) |
|
|
tokens, err := model.GetAllUserTokens(userId, pageInfo.GetStartIdx(), pageInfo.GetPageSize()) |
|
|
if err != nil { |
|
|
common.ApiError(c, err) |
|
|
return |
|
|
} |
|
|
total, _ := model.CountUserTokens(userId) |
|
|
pageInfo.SetTotal(int(total)) |
|
|
pageInfo.SetItems(tokens) |
|
|
common.ApiSuccess(c, pageInfo) |
|
|
return |
|
|
} |
|
|
|
|
|
func SearchTokens(c *gin.Context) { |
|
|
userId := c.GetInt("id") |
|
|
keyword := c.Query("keyword") |
|
|
token := c.Query("token") |
|
|
tokens, err := model.SearchUserTokens(userId, keyword, token) |
|
|
if err != nil { |
|
|
common.ApiError(c, err) |
|
|
return |
|
|
} |
|
|
c.JSON(http.StatusOK, gin.H{ |
|
|
"success": true, |
|
|
"message": "", |
|
|
"data": tokens, |
|
|
}) |
|
|
return |
|
|
} |
|
|
|
|
|
func GetToken(c *gin.Context) { |
|
|
id, err := strconv.Atoi(c.Param("id")) |
|
|
userId := c.GetInt("id") |
|
|
if err != nil { |
|
|
common.ApiError(c, err) |
|
|
return |
|
|
} |
|
|
token, err := model.GetTokenByIds(id, userId) |
|
|
if err != nil { |
|
|
common.ApiError(c, err) |
|
|
return |
|
|
} |
|
|
c.JSON(http.StatusOK, gin.H{ |
|
|
"success": true, |
|
|
"message": "", |
|
|
"data": token, |
|
|
}) |
|
|
return |
|
|
} |
|
|
|
|
|
func GetTokenStatus(c *gin.Context) { |
|
|
tokenId := c.GetInt("token_id") |
|
|
userId := c.GetInt("id") |
|
|
token, err := model.GetTokenByIds(tokenId, userId) |
|
|
if err != nil { |
|
|
common.ApiError(c, err) |
|
|
return |
|
|
} |
|
|
expiredAt := token.ExpiredTime |
|
|
if expiredAt == -1 { |
|
|
expiredAt = 0 |
|
|
} |
|
|
c.JSON(http.StatusOK, gin.H{ |
|
|
"object": "credit_summary", |
|
|
"total_granted": token.RemainQuota, |
|
|
"total_used": 0, |
|
|
"total_available": token.RemainQuota, |
|
|
"expires_at": expiredAt * 1000, |
|
|
}) |
|
|
} |
|
|
|
|
|
func GetTokenUsage(c *gin.Context) { |
|
|
authHeader := c.GetHeader("Authorization") |
|
|
if authHeader == "" { |
|
|
c.JSON(http.StatusUnauthorized, gin.H{ |
|
|
"success": false, |
|
|
"message": "No Authorization header", |
|
|
}) |
|
|
return |
|
|
} |
|
|
|
|
|
parts := strings.Split(authHeader, " ") |
|
|
if len(parts) != 2 || strings.ToLower(parts[0]) != "bearer" { |
|
|
c.JSON(http.StatusUnauthorized, gin.H{ |
|
|
"success": false, |
|
|
"message": "Invalid Bearer token", |
|
|
}) |
|
|
return |
|
|
} |
|
|
tokenKey := parts[1] |
|
|
|
|
|
token, err := model.GetTokenByKey(strings.TrimPrefix(tokenKey, "sk-"), false) |
|
|
if err != nil { |
|
|
c.JSON(http.StatusOK, gin.H{ |
|
|
"success": false, |
|
|
"message": err.Error(), |
|
|
}) |
|
|
return |
|
|
} |
|
|
|
|
|
expiredAt := token.ExpiredTime |
|
|
if expiredAt == -1 { |
|
|
expiredAt = 0 |
|
|
} |
|
|
|
|
|
c.JSON(http.StatusOK, gin.H{ |
|
|
"code": true, |
|
|
"message": "ok", |
|
|
"data": gin.H{ |
|
|
"object": "token_usage", |
|
|
"name": token.Name, |
|
|
"total_granted": token.RemainQuota + token.UsedQuota, |
|
|
"total_used": token.UsedQuota, |
|
|
"total_available": token.RemainQuota, |
|
|
"unlimited_quota": token.UnlimitedQuota, |
|
|
"model_limits": token.GetModelLimitsMap(), |
|
|
"model_limits_enabled": token.ModelLimitsEnabled, |
|
|
"expires_at": expiredAt, |
|
|
}, |
|
|
}) |
|
|
} |
|
|
|
|
|
func AddToken(c *gin.Context) { |
|
|
token := model.Token{} |
|
|
err := c.ShouldBindJSON(&token) |
|
|
if err != nil { |
|
|
common.ApiError(c, err) |
|
|
return |
|
|
} |
|
|
if len(token.Name) > 30 { |
|
|
c.JSON(http.StatusOK, gin.H{ |
|
|
"success": false, |
|
|
"message": "令牌名称过长", |
|
|
}) |
|
|
return |
|
|
} |
|
|
key, err := common.GenerateKey() |
|
|
if err != nil { |
|
|
c.JSON(http.StatusOK, gin.H{ |
|
|
"success": false, |
|
|
"message": "生成令牌失败", |
|
|
}) |
|
|
common.SysLog("failed to generate token key: " + err.Error()) |
|
|
return |
|
|
} |
|
|
cleanToken := model.Token{ |
|
|
UserId: c.GetInt("id"), |
|
|
Name: token.Name, |
|
|
Key: key, |
|
|
CreatedTime: common.GetTimestamp(), |
|
|
AccessedTime: common.GetTimestamp(), |
|
|
ExpiredTime: token.ExpiredTime, |
|
|
RemainQuota: token.RemainQuota, |
|
|
UnlimitedQuota: token.UnlimitedQuota, |
|
|
ModelLimitsEnabled: token.ModelLimitsEnabled, |
|
|
ModelLimits: token.ModelLimits, |
|
|
AllowIps: token.AllowIps, |
|
|
Group: token.Group, |
|
|
} |
|
|
err = cleanToken.Insert() |
|
|
if err != nil { |
|
|
common.ApiError(c, err) |
|
|
return |
|
|
} |
|
|
c.JSON(http.StatusOK, gin.H{ |
|
|
"success": true, |
|
|
"message": "", |
|
|
}) |
|
|
return |
|
|
} |
|
|
|
|
|
func DeleteToken(c *gin.Context) { |
|
|
id, _ := strconv.Atoi(c.Param("id")) |
|
|
userId := c.GetInt("id") |
|
|
err := model.DeleteTokenById(id, userId) |
|
|
if err != nil { |
|
|
common.ApiError(c, err) |
|
|
return |
|
|
} |
|
|
c.JSON(http.StatusOK, gin.H{ |
|
|
"success": true, |
|
|
"message": "", |
|
|
}) |
|
|
return |
|
|
} |
|
|
|
|
|
func UpdateToken(c *gin.Context) { |
|
|
userId := c.GetInt("id") |
|
|
statusOnly := c.Query("status_only") |
|
|
token := model.Token{} |
|
|
err := c.ShouldBindJSON(&token) |
|
|
if err != nil { |
|
|
common.ApiError(c, err) |
|
|
return |
|
|
} |
|
|
if len(token.Name) > 30 { |
|
|
c.JSON(http.StatusOK, gin.H{ |
|
|
"success": false, |
|
|
"message": "令牌名称过长", |
|
|
}) |
|
|
return |
|
|
} |
|
|
cleanToken, err := model.GetTokenByIds(token.Id, userId) |
|
|
if err != nil { |
|
|
common.ApiError(c, err) |
|
|
return |
|
|
} |
|
|
if token.Status == common.TokenStatusEnabled { |
|
|
if cleanToken.Status == common.TokenStatusExpired && cleanToken.ExpiredTime <= common.GetTimestamp() && cleanToken.ExpiredTime != -1 { |
|
|
c.JSON(http.StatusOK, gin.H{ |
|
|
"success": false, |
|
|
"message": "令牌已过期,无法启用,请先修改令牌过期时间,或者设置为永不过期", |
|
|
}) |
|
|
return |
|
|
} |
|
|
if cleanToken.Status == common.TokenStatusExhausted && cleanToken.RemainQuota <= 0 && !cleanToken.UnlimitedQuota { |
|
|
c.JSON(http.StatusOK, gin.H{ |
|
|
"success": false, |
|
|
"message": "令牌可用额度已用尽,无法启用,请先修改令牌剩余额度,或者设置为无限额度", |
|
|
}) |
|
|
return |
|
|
} |
|
|
} |
|
|
if statusOnly != "" { |
|
|
cleanToken.Status = token.Status |
|
|
} else { |
|
|
|
|
|
cleanToken.Name = token.Name |
|
|
cleanToken.ExpiredTime = token.ExpiredTime |
|
|
cleanToken.RemainQuota = token.RemainQuota |
|
|
cleanToken.UnlimitedQuota = token.UnlimitedQuota |
|
|
cleanToken.ModelLimitsEnabled = token.ModelLimitsEnabled |
|
|
cleanToken.ModelLimits = token.ModelLimits |
|
|
cleanToken.AllowIps = token.AllowIps |
|
|
cleanToken.Group = token.Group |
|
|
} |
|
|
err = cleanToken.Update() |
|
|
if err != nil { |
|
|
common.ApiError(c, err) |
|
|
return |
|
|
} |
|
|
c.JSON(http.StatusOK, gin.H{ |
|
|
"success": true, |
|
|
"message": "", |
|
|
"data": cleanToken, |
|
|
}) |
|
|
return |
|
|
} |
|
|
|
|
|
type TokenBatch struct { |
|
|
Ids []int `json:"ids"` |
|
|
} |
|
|
|
|
|
func DeleteTokenBatch(c *gin.Context) { |
|
|
tokenBatch := TokenBatch{} |
|
|
if err := c.ShouldBindJSON(&tokenBatch); err != nil || len(tokenBatch.Ids) == 0 { |
|
|
c.JSON(http.StatusOK, gin.H{ |
|
|
"success": false, |
|
|
"message": "参数错误", |
|
|
}) |
|
|
return |
|
|
} |
|
|
userId := c.GetInt("id") |
|
|
count, err := model.BatchDeleteTokens(tokenBatch.Ids, userId) |
|
|
if err != nil { |
|
|
common.ApiError(c, err) |
|
|
return |
|
|
} |
|
|
c.JSON(http.StatusOK, gin.H{ |
|
|
"success": true, |
|
|
"message": "", |
|
|
"data": count, |
|
|
}) |
|
|
} |
|
|
|