| package controller
|
|
|
| import (
|
| "fmt"
|
| "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) > 50 {
|
| c.JSON(http.StatusOK, gin.H{
|
| "success": false,
|
| "message": "令牌名称过长",
|
| })
|
| return
|
| }
|
|
|
| if !token.UnlimitedQuota {
|
| if token.RemainQuota < 0 {
|
| c.JSON(http.StatusOK, gin.H{
|
| "success": false,
|
| "message": "额度值不能为负数",
|
| })
|
| return
|
| }
|
| maxQuotaValue := int((1000000000 * common.QuotaPerUnit))
|
| if token.RemainQuota > maxQuotaValue {
|
| c.JSON(http.StatusOK, gin.H{
|
| "success": false,
|
| "message": fmt.Sprintf("额度值超出有效范围,最大值为 %d", maxQuotaValue),
|
| })
|
| 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,
|
| CrossGroupRetry: token.CrossGroupRetry,
|
| }
|
| 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) > 50 {
|
| c.JSON(http.StatusOK, gin.H{
|
| "success": false,
|
| "message": "令牌名称过长",
|
| })
|
| return
|
| }
|
| if !token.UnlimitedQuota {
|
| if token.RemainQuota < 0 {
|
| c.JSON(http.StatusOK, gin.H{
|
| "success": false,
|
| "message": "额度值不能为负数",
|
| })
|
| return
|
| }
|
| maxQuotaValue := int((1000000000 * common.QuotaPerUnit))
|
| if token.RemainQuota > maxQuotaValue {
|
| c.JSON(http.StatusOK, gin.H{
|
| "success": false,
|
| "message": fmt.Sprintf("额度值超出有效范围,最大值为 %d", maxQuotaValue),
|
| })
|
| 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
|
| cleanToken.CrossGroupRetry = token.CrossGroupRetry
|
| }
|
| err = cleanToken.Update()
|
| if err != nil {
|
| common.ApiError(c, err)
|
| return
|
| }
|
| c.JSON(http.StatusOK, gin.H{
|
| "success": true,
|
| "message": "",
|
| "data": cleanToken,
|
| })
|
| }
|
|
|
| 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,
|
| })
|
| }
|
|
|