| | 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, |
| | }) |
| | } |
| |
|