2026-01-04 19:27:53 +08:00
|
|
|
// Package routes provides HTTP route registration and handlers.
|
2025-12-26 10:42:08 +08:00
|
|
|
package routes
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"github.com/Wei-Shaw/sub2api/internal/handler"
|
|
|
|
|
"github.com/Wei-Shaw/sub2api/internal/server/middleware"
|
|
|
|
|
|
|
|
|
|
"github.com/gin-gonic/gin"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
// RegisterAdminRoutes 注册管理员路由
|
|
|
|
|
func RegisterAdminRoutes(
|
|
|
|
|
v1 *gin.RouterGroup,
|
|
|
|
|
h *handler.Handlers,
|
|
|
|
|
adminAuth middleware.AdminAuthMiddleware,
|
|
|
|
|
) {
|
|
|
|
|
admin := v1.Group("/admin")
|
|
|
|
|
admin.Use(gin.HandlerFunc(adminAuth))
|
|
|
|
|
{
|
|
|
|
|
// 仪表盘
|
|
|
|
|
registerDashboardRoutes(admin, h)
|
|
|
|
|
|
|
|
|
|
// 用户管理
|
|
|
|
|
registerUserManagementRoutes(admin, h)
|
|
|
|
|
|
|
|
|
|
// 分组管理
|
|
|
|
|
registerGroupRoutes(admin, h)
|
|
|
|
|
|
|
|
|
|
// 账号管理
|
|
|
|
|
registerAccountRoutes(admin, h)
|
|
|
|
|
|
|
|
|
|
// OpenAI OAuth
|
|
|
|
|
registerOpenAIOAuthRoutes(admin, h)
|
|
|
|
|
|
2025-12-26 00:17:55 -08:00
|
|
|
// Gemini OAuth
|
|
|
|
|
registerGeminiOAuthRoutes(admin, h)
|
|
|
|
|
|
2025-12-29 00:44:07 +08:00
|
|
|
// Antigravity OAuth
|
|
|
|
|
registerAntigravityOAuthRoutes(admin, h)
|
|
|
|
|
|
2025-12-26 10:42:08 +08:00
|
|
|
// 代理管理
|
|
|
|
|
registerProxyRoutes(admin, h)
|
|
|
|
|
|
|
|
|
|
// 卡密管理
|
|
|
|
|
registerRedeemCodeRoutes(admin, h)
|
|
|
|
|
|
|
|
|
|
// 系统设置
|
|
|
|
|
registerSettingsRoutes(admin, h)
|
|
|
|
|
|
|
|
|
|
// 系统管理
|
|
|
|
|
registerSystemRoutes(admin, h)
|
|
|
|
|
|
|
|
|
|
// 订阅管理
|
|
|
|
|
registerSubscriptionRoutes(admin, h)
|
|
|
|
|
|
|
|
|
|
// 使用记录管理
|
|
|
|
|
registerUsageRoutes(admin, h)
|
2026-01-01 18:58:34 +08:00
|
|
|
|
|
|
|
|
// 用户属性管理
|
|
|
|
|
registerUserAttributeRoutes(admin, h)
|
2025-12-26 10:42:08 +08:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func registerDashboardRoutes(admin *gin.RouterGroup, h *handler.Handlers) {
|
|
|
|
|
dashboard := admin.Group("/dashboard")
|
|
|
|
|
{
|
|
|
|
|
dashboard.GET("/stats", h.Admin.Dashboard.GetStats)
|
|
|
|
|
dashboard.GET("/realtime", h.Admin.Dashboard.GetRealtimeMetrics)
|
|
|
|
|
dashboard.GET("/trend", h.Admin.Dashboard.GetUsageTrend)
|
|
|
|
|
dashboard.GET("/models", h.Admin.Dashboard.GetModelStats)
|
2026-01-04 19:27:53 +08:00
|
|
|
dashboard.GET("/api-keys-trend", h.Admin.Dashboard.GetAPIKeyUsageTrend)
|
2025-12-26 10:42:08 +08:00
|
|
|
dashboard.GET("/users-trend", h.Admin.Dashboard.GetUserUsageTrend)
|
|
|
|
|
dashboard.POST("/users-usage", h.Admin.Dashboard.GetBatchUsersUsage)
|
2026-01-04 19:27:53 +08:00
|
|
|
dashboard.POST("/api-keys-usage", h.Admin.Dashboard.GetBatchAPIKeysUsage)
|
2025-12-26 10:42:08 +08:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func registerUserManagementRoutes(admin *gin.RouterGroup, h *handler.Handlers) {
|
|
|
|
|
users := admin.Group("/users")
|
|
|
|
|
{
|
|
|
|
|
users.GET("", h.Admin.User.List)
|
|
|
|
|
users.GET("/:id", h.Admin.User.GetByID)
|
|
|
|
|
users.POST("", h.Admin.User.Create)
|
|
|
|
|
users.PUT("/:id", h.Admin.User.Update)
|
|
|
|
|
users.DELETE("/:id", h.Admin.User.Delete)
|
|
|
|
|
users.POST("/:id/balance", h.Admin.User.UpdateBalance)
|
|
|
|
|
users.GET("/:id/api-keys", h.Admin.User.GetUserAPIKeys)
|
|
|
|
|
users.GET("/:id/usage", h.Admin.User.GetUserUsage)
|
2026-01-01 18:58:34 +08:00
|
|
|
|
|
|
|
|
// User attribute values
|
|
|
|
|
users.GET("/:id/attributes", h.Admin.UserAttribute.GetUserAttributes)
|
|
|
|
|
users.PUT("/:id/attributes", h.Admin.UserAttribute.UpdateUserAttributes)
|
2025-12-26 10:42:08 +08:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func registerGroupRoutes(admin *gin.RouterGroup, h *handler.Handlers) {
|
|
|
|
|
groups := admin.Group("/groups")
|
|
|
|
|
{
|
|
|
|
|
groups.GET("", h.Admin.Group.List)
|
|
|
|
|
groups.GET("/all", h.Admin.Group.GetAll)
|
|
|
|
|
groups.GET("/:id", h.Admin.Group.GetByID)
|
|
|
|
|
groups.POST("", h.Admin.Group.Create)
|
|
|
|
|
groups.PUT("/:id", h.Admin.Group.Update)
|
|
|
|
|
groups.DELETE("/:id", h.Admin.Group.Delete)
|
|
|
|
|
groups.GET("/:id/stats", h.Admin.Group.GetStats)
|
|
|
|
|
groups.GET("/:id/api-keys", h.Admin.Group.GetGroupAPIKeys)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func registerAccountRoutes(admin *gin.RouterGroup, h *handler.Handlers) {
|
|
|
|
|
accounts := admin.Group("/accounts")
|
|
|
|
|
{
|
|
|
|
|
accounts.GET("", h.Admin.Account.List)
|
|
|
|
|
accounts.GET("/:id", h.Admin.Account.GetByID)
|
|
|
|
|
accounts.POST("", h.Admin.Account.Create)
|
|
|
|
|
accounts.POST("/sync/crs", h.Admin.Account.SyncFromCRS)
|
|
|
|
|
accounts.PUT("/:id", h.Admin.Account.Update)
|
|
|
|
|
accounts.DELETE("/:id", h.Admin.Account.Delete)
|
|
|
|
|
accounts.POST("/:id/test", h.Admin.Account.Test)
|
|
|
|
|
accounts.POST("/:id/refresh", h.Admin.Account.Refresh)
|
feat(gemini): 添加 Google One 存储空间推断 Tier 功能
## 功能概述
通过 Google Drive API 获取存储空间配额来推断 Google One 订阅等级,并优化统一的配额显示系统。
## 后端改动
- 新增 Drive API 客户端 (drive_client.go)
- 支持代理和指数退避重试
- 处理 403/429 错误
- 添加 Tier 推断逻辑 (inferGoogleOneTier)
- 支持 6 种 tier 类型:AI_PREMIUM, GOOGLE_ONE_STANDARD, GOOGLE_ONE_BASIC, FREE, GOOGLE_ONE_UNKNOWN, GOOGLE_ONE_UNLIMITED
- 集成到 OAuth 流程
- ExchangeCode: 授权时自动获取 tier
- RefreshAccountToken: Token 刷新时更新 tier (24小时缓存)
- 新增管理 API 端点
- POST /api/v1/admin/accounts/:id/refresh-tier (单个账号刷新)
- POST /api/v1/admin/accounts/batch-refresh-tier (批量刷新)
## 前端改动
- 更新 AccountQuotaInfo.vue
- 添加 Google One tier 标签映射
- 添加 tier 颜色样式 (紫色/蓝色/绿色/灰色/琥珀色)
- 更新 AccountUsageCell.vue
- 添加 Google One tier 显示逻辑
- 根据 oauth_type 区分显示方式
- 添加国际化翻译 (en.ts, zh.ts)
- aiPremium, standard, basic, free, personal, unlimited
## Tier 推断规则
- >= 2TB: AI Premium
- >= 200GB: Google One Standard
- >= 100GB: Google One Basic
- >= 15GB: Free
- > 100TB: Unlimited (G Suite legacy)
- 其他/失败: Unknown (显示为 Personal)
## 优雅降级
- Drive API 失败时使用 GOOGLE_ONE_UNKNOWN
- 不阻断 OAuth 流程
- 24小时缓存避免频繁调用
## 测试
- ✅ 后端编译成功
- ✅ 前端构建成功
- ✅ 所有代码符合现有规范
2025-12-31 21:45:24 -08:00
|
|
|
accounts.POST("/:id/refresh-tier", h.Admin.Account.RefreshTier)
|
2025-12-26 10:42:08 +08:00
|
|
|
accounts.GET("/:id/stats", h.Admin.Account.GetStats)
|
|
|
|
|
accounts.POST("/:id/clear-error", h.Admin.Account.ClearError)
|
|
|
|
|
accounts.GET("/:id/usage", h.Admin.Account.GetUsage)
|
|
|
|
|
accounts.GET("/:id/today-stats", h.Admin.Account.GetTodayStats)
|
|
|
|
|
accounts.POST("/:id/clear-rate-limit", h.Admin.Account.ClearRateLimit)
|
2026-01-03 06:34:00 -08:00
|
|
|
accounts.GET("/:id/temp-unschedulable", h.Admin.Account.GetTempUnschedulable)
|
|
|
|
|
accounts.DELETE("/:id/temp-unschedulable", h.Admin.Account.ClearTempUnschedulable)
|
2025-12-26 10:42:08 +08:00
|
|
|
accounts.POST("/:id/schedulable", h.Admin.Account.SetSchedulable)
|
|
|
|
|
accounts.GET("/:id/models", h.Admin.Account.GetAvailableModels)
|
|
|
|
|
accounts.POST("/batch", h.Admin.Account.BatchCreate)
|
|
|
|
|
accounts.POST("/batch-update-credentials", h.Admin.Account.BatchUpdateCredentials)
|
feat(gemini): 添加 Google One 存储空间推断 Tier 功能
## 功能概述
通过 Google Drive API 获取存储空间配额来推断 Google One 订阅等级,并优化统一的配额显示系统。
## 后端改动
- 新增 Drive API 客户端 (drive_client.go)
- 支持代理和指数退避重试
- 处理 403/429 错误
- 添加 Tier 推断逻辑 (inferGoogleOneTier)
- 支持 6 种 tier 类型:AI_PREMIUM, GOOGLE_ONE_STANDARD, GOOGLE_ONE_BASIC, FREE, GOOGLE_ONE_UNKNOWN, GOOGLE_ONE_UNLIMITED
- 集成到 OAuth 流程
- ExchangeCode: 授权时自动获取 tier
- RefreshAccountToken: Token 刷新时更新 tier (24小时缓存)
- 新增管理 API 端点
- POST /api/v1/admin/accounts/:id/refresh-tier (单个账号刷新)
- POST /api/v1/admin/accounts/batch-refresh-tier (批量刷新)
## 前端改动
- 更新 AccountQuotaInfo.vue
- 添加 Google One tier 标签映射
- 添加 tier 颜色样式 (紫色/蓝色/绿色/灰色/琥珀色)
- 更新 AccountUsageCell.vue
- 添加 Google One tier 显示逻辑
- 根据 oauth_type 区分显示方式
- 添加国际化翻译 (en.ts, zh.ts)
- aiPremium, standard, basic, free, personal, unlimited
## Tier 推断规则
- >= 2TB: AI Premium
- >= 200GB: Google One Standard
- >= 100GB: Google One Basic
- >= 15GB: Free
- > 100TB: Unlimited (G Suite legacy)
- 其他/失败: Unknown (显示为 Personal)
## 优雅降级
- Drive API 失败时使用 GOOGLE_ONE_UNKNOWN
- 不阻断 OAuth 流程
- 24小时缓存避免频繁调用
## 测试
- ✅ 后端编译成功
- ✅ 前端构建成功
- ✅ 所有代码符合现有规范
2025-12-31 21:45:24 -08:00
|
|
|
accounts.POST("/batch-refresh-tier", h.Admin.Account.BatchRefreshTier)
|
2025-12-26 10:42:08 +08:00
|
|
|
accounts.POST("/bulk-update", h.Admin.Account.BulkUpdate)
|
|
|
|
|
|
|
|
|
|
// Claude OAuth routes
|
|
|
|
|
accounts.POST("/generate-auth-url", h.Admin.OAuth.GenerateAuthURL)
|
|
|
|
|
accounts.POST("/generate-setup-token-url", h.Admin.OAuth.GenerateSetupTokenURL)
|
|
|
|
|
accounts.POST("/exchange-code", h.Admin.OAuth.ExchangeCode)
|
|
|
|
|
accounts.POST("/exchange-setup-token-code", h.Admin.OAuth.ExchangeSetupTokenCode)
|
|
|
|
|
accounts.POST("/cookie-auth", h.Admin.OAuth.CookieAuth)
|
|
|
|
|
accounts.POST("/setup-token-cookie-auth", h.Admin.OAuth.SetupTokenCookieAuth)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func registerOpenAIOAuthRoutes(admin *gin.RouterGroup, h *handler.Handlers) {
|
|
|
|
|
openai := admin.Group("/openai")
|
|
|
|
|
{
|
|
|
|
|
openai.POST("/generate-auth-url", h.Admin.OpenAIOAuth.GenerateAuthURL)
|
|
|
|
|
openai.POST("/exchange-code", h.Admin.OpenAIOAuth.ExchangeCode)
|
|
|
|
|
openai.POST("/refresh-token", h.Admin.OpenAIOAuth.RefreshToken)
|
|
|
|
|
openai.POST("/accounts/:id/refresh", h.Admin.OpenAIOAuth.RefreshAccountToken)
|
|
|
|
|
openai.POST("/create-from-oauth", h.Admin.OpenAIOAuth.CreateAccountFromOAuth)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2025-12-26 00:17:55 -08:00
|
|
|
func registerGeminiOAuthRoutes(admin *gin.RouterGroup, h *handler.Handlers) {
|
|
|
|
|
gemini := admin.Group("/gemini")
|
|
|
|
|
{
|
|
|
|
|
gemini.POST("/oauth/auth-url", h.Admin.GeminiOAuth.GenerateAuthURL)
|
|
|
|
|
gemini.POST("/oauth/exchange-code", h.Admin.GeminiOAuth.ExchangeCode)
|
|
|
|
|
gemini.GET("/oauth/capabilities", h.Admin.GeminiOAuth.GetCapabilities)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2025-12-28 15:54:42 +08:00
|
|
|
func registerAntigravityOAuthRoutes(admin *gin.RouterGroup, h *handler.Handlers) {
|
|
|
|
|
antigravity := admin.Group("/antigravity")
|
|
|
|
|
{
|
|
|
|
|
antigravity.POST("/oauth/auth-url", h.Admin.AntigravityOAuth.GenerateAuthURL)
|
|
|
|
|
antigravity.POST("/oauth/exchange-code", h.Admin.AntigravityOAuth.ExchangeCode)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2025-12-26 10:42:08 +08:00
|
|
|
func registerProxyRoutes(admin *gin.RouterGroup, h *handler.Handlers) {
|
|
|
|
|
proxies := admin.Group("/proxies")
|
|
|
|
|
{
|
|
|
|
|
proxies.GET("", h.Admin.Proxy.List)
|
|
|
|
|
proxies.GET("/all", h.Admin.Proxy.GetAll)
|
|
|
|
|
proxies.GET("/:id", h.Admin.Proxy.GetByID)
|
|
|
|
|
proxies.POST("", h.Admin.Proxy.Create)
|
|
|
|
|
proxies.PUT("/:id", h.Admin.Proxy.Update)
|
|
|
|
|
proxies.DELETE("/:id", h.Admin.Proxy.Delete)
|
|
|
|
|
proxies.POST("/:id/test", h.Admin.Proxy.Test)
|
|
|
|
|
proxies.GET("/:id/stats", h.Admin.Proxy.GetStats)
|
|
|
|
|
proxies.GET("/:id/accounts", h.Admin.Proxy.GetProxyAccounts)
|
|
|
|
|
proxies.POST("/batch", h.Admin.Proxy.BatchCreate)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func registerRedeemCodeRoutes(admin *gin.RouterGroup, h *handler.Handlers) {
|
|
|
|
|
codes := admin.Group("/redeem-codes")
|
|
|
|
|
{
|
|
|
|
|
codes.GET("", h.Admin.Redeem.List)
|
|
|
|
|
codes.GET("/stats", h.Admin.Redeem.GetStats)
|
|
|
|
|
codes.GET("/export", h.Admin.Redeem.Export)
|
|
|
|
|
codes.GET("/:id", h.Admin.Redeem.GetByID)
|
|
|
|
|
codes.POST("/generate", h.Admin.Redeem.Generate)
|
|
|
|
|
codes.DELETE("/:id", h.Admin.Redeem.Delete)
|
|
|
|
|
codes.POST("/batch-delete", h.Admin.Redeem.BatchDelete)
|
|
|
|
|
codes.POST("/:id/expire", h.Admin.Redeem.Expire)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func registerSettingsRoutes(admin *gin.RouterGroup, h *handler.Handlers) {
|
|
|
|
|
adminSettings := admin.Group("/settings")
|
|
|
|
|
{
|
|
|
|
|
adminSettings.GET("", h.Admin.Setting.GetSettings)
|
|
|
|
|
adminSettings.PUT("", h.Admin.Setting.UpdateSettings)
|
2026-01-04 19:27:53 +08:00
|
|
|
adminSettings.POST("/test-smtp", h.Admin.Setting.TestSMTPConnection)
|
2025-12-26 10:42:08 +08:00
|
|
|
adminSettings.POST("/send-test-email", h.Admin.Setting.SendTestEmail)
|
|
|
|
|
// Admin API Key 管理
|
2026-01-04 19:27:53 +08:00
|
|
|
adminSettings.GET("/admin-api-key", h.Admin.Setting.GetAdminAPIKey)
|
|
|
|
|
adminSettings.POST("/admin-api-key/regenerate", h.Admin.Setting.RegenerateAdminAPIKey)
|
|
|
|
|
adminSettings.DELETE("/admin-api-key", h.Admin.Setting.DeleteAdminAPIKey)
|
2025-12-26 10:42:08 +08:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func registerSystemRoutes(admin *gin.RouterGroup, h *handler.Handlers) {
|
|
|
|
|
system := admin.Group("/system")
|
|
|
|
|
{
|
|
|
|
|
system.GET("/version", h.Admin.System.GetVersion)
|
|
|
|
|
system.GET("/check-updates", h.Admin.System.CheckUpdates)
|
|
|
|
|
system.POST("/update", h.Admin.System.PerformUpdate)
|
|
|
|
|
system.POST("/rollback", h.Admin.System.Rollback)
|
|
|
|
|
system.POST("/restart", h.Admin.System.RestartService)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func registerSubscriptionRoutes(admin *gin.RouterGroup, h *handler.Handlers) {
|
|
|
|
|
subscriptions := admin.Group("/subscriptions")
|
|
|
|
|
{
|
|
|
|
|
subscriptions.GET("", h.Admin.Subscription.List)
|
|
|
|
|
subscriptions.GET("/:id", h.Admin.Subscription.GetByID)
|
|
|
|
|
subscriptions.GET("/:id/progress", h.Admin.Subscription.GetProgress)
|
|
|
|
|
subscriptions.POST("/assign", h.Admin.Subscription.Assign)
|
|
|
|
|
subscriptions.POST("/bulk-assign", h.Admin.Subscription.BulkAssign)
|
|
|
|
|
subscriptions.POST("/:id/extend", h.Admin.Subscription.Extend)
|
|
|
|
|
subscriptions.DELETE("/:id", h.Admin.Subscription.Revoke)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 分组下的订阅列表
|
|
|
|
|
admin.GET("/groups/:id/subscriptions", h.Admin.Subscription.ListByGroup)
|
|
|
|
|
|
|
|
|
|
// 用户下的订阅列表
|
|
|
|
|
admin.GET("/users/:id/subscriptions", h.Admin.Subscription.ListByUser)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func registerUsageRoutes(admin *gin.RouterGroup, h *handler.Handlers) {
|
|
|
|
|
usage := admin.Group("/usage")
|
|
|
|
|
{
|
|
|
|
|
usage.GET("", h.Admin.Usage.List)
|
|
|
|
|
usage.GET("/stats", h.Admin.Usage.Stats)
|
|
|
|
|
usage.GET("/search-users", h.Admin.Usage.SearchUsers)
|
2026-01-04 19:27:53 +08:00
|
|
|
usage.GET("/search-api-keys", h.Admin.Usage.SearchAPIKeys)
|
2025-12-26 10:42:08 +08:00
|
|
|
}
|
|
|
|
|
}
|
2026-01-01 18:58:34 +08:00
|
|
|
|
|
|
|
|
func registerUserAttributeRoutes(admin *gin.RouterGroup, h *handler.Handlers) {
|
|
|
|
|
attrs := admin.Group("/user-attributes")
|
|
|
|
|
{
|
|
|
|
|
attrs.GET("", h.Admin.UserAttribute.ListDefinitions)
|
|
|
|
|
attrs.POST("", h.Admin.UserAttribute.CreateDefinition)
|
|
|
|
|
attrs.POST("/batch", h.Admin.UserAttribute.GetBatchUserAttributes)
|
|
|
|
|
attrs.PUT("/reorder", h.Admin.UserAttribute.ReorderDefinitions)
|
|
|
|
|
attrs.PUT("/:id", h.Admin.UserAttribute.UpdateDefinition)
|
|
|
|
|
attrs.DELETE("/:id", h.Admin.UserAttribute.DeleteDefinition)
|
|
|
|
|
}
|
|
|
|
|
}
|