Handler 改进: - 添加 GET /api/v1/admin/gemini/oauth/capabilities 接口 - 简化 GenerateAuthURL,redirect_uri 由服务层决定 Repository 改进: - ExchangeCode/RefreshToken 根据 oauthType 选择正确的 OAuth 客户端 - Code Assist 始终使用内置客户端,AI Studio 使用用户配置的客户端
136 lines
4.2 KiB
Go
136 lines
4.2 KiB
Go
package admin
|
|
|
|
import (
|
|
"fmt"
|
|
"strings"
|
|
|
|
"github.com/Wei-Shaw/sub2api/internal/pkg/response"
|
|
"github.com/Wei-Shaw/sub2api/internal/service"
|
|
|
|
"github.com/gin-gonic/gin"
|
|
)
|
|
|
|
type GeminiOAuthHandler struct {
|
|
geminiOAuthService *service.GeminiOAuthService
|
|
}
|
|
|
|
func NewGeminiOAuthHandler(geminiOAuthService *service.GeminiOAuthService) *GeminiOAuthHandler {
|
|
return &GeminiOAuthHandler{geminiOAuthService: geminiOAuthService}
|
|
}
|
|
|
|
// GET /api/v1/admin/gemini/oauth/capabilities
|
|
func (h *GeminiOAuthHandler) GetCapabilities(c *gin.Context) {
|
|
cfg := h.geminiOAuthService.GetOAuthConfig()
|
|
response.Success(c, cfg)
|
|
}
|
|
|
|
type GeminiGenerateAuthURLRequest struct {
|
|
ProxyID *int64 `json:"proxy_id"`
|
|
ProjectID string `json:"project_id"`
|
|
// OAuth 类型: "code_assist" (需要 project_id) 或 "ai_studio" (不需要 project_id)
|
|
// 默认为 "code_assist" 以保持向后兼容
|
|
OAuthType string `json:"oauth_type"`
|
|
}
|
|
|
|
// GenerateAuthURL generates Google OAuth authorization URL for Gemini.
|
|
// POST /api/v1/admin/gemini/oauth/auth-url
|
|
func (h *GeminiOAuthHandler) GenerateAuthURL(c *gin.Context) {
|
|
var req GeminiGenerateAuthURLRequest
|
|
if err := c.ShouldBindJSON(&req); err != nil {
|
|
response.BadRequest(c, "Invalid request: "+err.Error())
|
|
return
|
|
}
|
|
|
|
// 默认使用 code_assist 以保持向后兼容
|
|
oauthType := strings.TrimSpace(req.OAuthType)
|
|
if oauthType == "" {
|
|
oauthType = "code_assist"
|
|
}
|
|
if oauthType != "code_assist" && oauthType != "ai_studio" {
|
|
response.BadRequest(c, "Invalid oauth_type: must be 'code_assist' or 'ai_studio'")
|
|
return
|
|
}
|
|
|
|
// Always pass the "hosted" callback URI; the OAuth service may override it depending on
|
|
// oauth_type and whether the built-in Gemini CLI OAuth client is used.
|
|
redirectURI := deriveGeminiRedirectURI(c)
|
|
result, err := h.geminiOAuthService.GenerateAuthURL(c.Request.Context(), req.ProxyID, redirectURI, req.ProjectID, oauthType)
|
|
if err != nil {
|
|
msg := err.Error()
|
|
// Treat missing/invalid OAuth client configuration as a user/config error.
|
|
if strings.Contains(msg, "OAuth client not configured") || strings.Contains(msg, "requires your own OAuth Client") {
|
|
response.BadRequest(c, "Failed to generate auth URL: "+msg)
|
|
return
|
|
}
|
|
response.InternalError(c, "Failed to generate auth URL: "+msg)
|
|
return
|
|
}
|
|
|
|
response.Success(c, result)
|
|
}
|
|
|
|
type GeminiExchangeCodeRequest struct {
|
|
SessionID string `json:"session_id" binding:"required"`
|
|
State string `json:"state" binding:"required"`
|
|
Code string `json:"code" binding:"required"`
|
|
ProxyID *int64 `json:"proxy_id"`
|
|
// OAuth 类型: "code_assist" 或 "ai_studio",需要与 GenerateAuthURL 时的类型一致
|
|
OAuthType string `json:"oauth_type"`
|
|
}
|
|
|
|
// ExchangeCode exchanges authorization code for tokens.
|
|
// POST /api/v1/admin/gemini/oauth/exchange-code
|
|
func (h *GeminiOAuthHandler) ExchangeCode(c *gin.Context) {
|
|
var req GeminiExchangeCodeRequest
|
|
if err := c.ShouldBindJSON(&req); err != nil {
|
|
response.BadRequest(c, "Invalid request: "+err.Error())
|
|
return
|
|
}
|
|
|
|
// 默认使用 code_assist 以保持向后兼容
|
|
oauthType := strings.TrimSpace(req.OAuthType)
|
|
if oauthType == "" {
|
|
oauthType = "code_assist"
|
|
}
|
|
if oauthType != "code_assist" && oauthType != "ai_studio" {
|
|
response.BadRequest(c, "Invalid oauth_type: must be 'code_assist' or 'ai_studio'")
|
|
return
|
|
}
|
|
|
|
tokenInfo, err := h.geminiOAuthService.ExchangeCode(c.Request.Context(), &service.GeminiExchangeCodeInput{
|
|
SessionID: req.SessionID,
|
|
State: req.State,
|
|
Code: req.Code,
|
|
ProxyID: req.ProxyID,
|
|
OAuthType: oauthType,
|
|
})
|
|
if err != nil {
|
|
response.BadRequest(c, "Failed to exchange code: "+err.Error())
|
|
return
|
|
}
|
|
|
|
response.Success(c, tokenInfo)
|
|
}
|
|
|
|
func deriveGeminiRedirectURI(c *gin.Context) string {
|
|
origin := strings.TrimSpace(c.GetHeader("Origin"))
|
|
if origin != "" {
|
|
return strings.TrimRight(origin, "/") + "/auth/callback"
|
|
}
|
|
|
|
scheme := "http"
|
|
if c.Request.TLS != nil {
|
|
scheme = "https"
|
|
}
|
|
if xfProto := strings.TrimSpace(c.GetHeader("X-Forwarded-Proto")); xfProto != "" {
|
|
scheme = strings.TrimSpace(strings.Split(xfProto, ",")[0])
|
|
}
|
|
|
|
host := strings.TrimSpace(c.Request.Host)
|
|
if xfHost := strings.TrimSpace(c.GetHeader("X-Forwarded-Host")); xfHost != "" {
|
|
host = strings.TrimSpace(strings.Split(xfHost, ",")[0])
|
|
}
|
|
|
|
return fmt.Sprintf("%s://%s/auth/callback", scheme, host)
|
|
}
|