196 lines
5.2 KiB
Go
196 lines
5.2 KiB
Go
package ai
|
||
|
||
import (
|
||
"bytes"
|
||
"encoding/json"
|
||
"fmt"
|
||
"io"
|
||
"net/http"
|
||
"strings"
|
||
"time"
|
||
)
|
||
|
||
type GeminiClient struct {
|
||
BaseURL string
|
||
APIKey string
|
||
Model string
|
||
Endpoint string
|
||
HTTPClient *http.Client
|
||
}
|
||
|
||
type GeminiTextRequest struct {
|
||
Contents []GeminiContent `json:"contents"`
|
||
SystemInstruction *GeminiInstruction `json:"systemInstruction,omitempty"`
|
||
}
|
||
|
||
type GeminiContent struct {
|
||
Parts []GeminiPart `json:"parts"`
|
||
Role string `json:"role,omitempty"`
|
||
}
|
||
|
||
type GeminiPart struct {
|
||
Text string `json:"text"`
|
||
}
|
||
|
||
type GeminiInstruction struct {
|
||
Parts []GeminiPart `json:"parts"`
|
||
}
|
||
|
||
type GeminiTextResponse struct {
|
||
Candidates []struct {
|
||
Content struct {
|
||
Parts []struct {
|
||
Text string `json:"text"`
|
||
} `json:"parts"`
|
||
Role string `json:"role"`
|
||
} `json:"content"`
|
||
FinishReason string `json:"finishReason"`
|
||
Index int `json:"index"`
|
||
SafetyRatings []struct {
|
||
Category string `json:"category"`
|
||
Probability string `json:"probability"`
|
||
} `json:"safetyRatings"`
|
||
} `json:"candidates"`
|
||
UsageMetadata struct {
|
||
PromptTokenCount int `json:"promptTokenCount"`
|
||
CandidatesTokenCount int `json:"candidatesTokenCount"`
|
||
TotalTokenCount int `json:"totalTokenCount"`
|
||
} `json:"usageMetadata"`
|
||
}
|
||
|
||
func NewGeminiClient(baseURL, apiKey, model, endpoint string) *GeminiClient {
|
||
if baseURL == "" {
|
||
baseURL = "https://generativelanguage.googleapis.com"
|
||
}
|
||
if endpoint == "" {
|
||
endpoint = "/v1beta/models/{model}:generateContent"
|
||
}
|
||
if model == "" {
|
||
model = "gemini-3-pro"
|
||
}
|
||
return &GeminiClient{
|
||
BaseURL: baseURL,
|
||
APIKey: apiKey,
|
||
Model: model,
|
||
Endpoint: endpoint,
|
||
HTTPClient: &http.Client{
|
||
Timeout: 10 * time.Minute,
|
||
},
|
||
}
|
||
}
|
||
|
||
func (c *GeminiClient) GenerateText(prompt string, systemPrompt string, options ...func(*ChatCompletionRequest)) (string, error) {
|
||
model := c.Model
|
||
|
||
// 构建请求体
|
||
reqBody := GeminiTextRequest{
|
||
Contents: []GeminiContent{
|
||
{
|
||
Parts: []GeminiPart{{Text: prompt}},
|
||
Role: "user",
|
||
},
|
||
},
|
||
}
|
||
|
||
// 使用 systemInstruction 字段处理系统提示
|
||
if systemPrompt != "" {
|
||
reqBody.SystemInstruction = &GeminiInstruction{
|
||
Parts: []GeminiPart{{Text: systemPrompt}},
|
||
}
|
||
}
|
||
|
||
jsonData, err := json.Marshal(reqBody)
|
||
if err != nil {
|
||
fmt.Printf("Gemini: Failed to marshal request: %v\n", err)
|
||
return "", fmt.Errorf("marshal request: %w", err)
|
||
}
|
||
|
||
// 替换端点中的 {model} 占位符
|
||
endpoint := c.BaseURL + c.Endpoint
|
||
endpoint = strings.ReplaceAll(endpoint, "{model}", model)
|
||
url := fmt.Sprintf("%s?key=%s", endpoint, c.APIKey)
|
||
|
||
// 打印请求信息(隐藏 API Key)
|
||
safeURL := strings.Replace(url, c.APIKey, "***", 1)
|
||
fmt.Printf("Gemini: Sending request to: %s\n", safeURL)
|
||
requestPreview := string(jsonData)
|
||
if len(jsonData) > 300 {
|
||
requestPreview = string(jsonData[:300]) + "..."
|
||
}
|
||
fmt.Printf("Gemini: Request body: %s\n", requestPreview)
|
||
|
||
req, err := http.NewRequest("POST", url, bytes.NewBuffer(jsonData))
|
||
if err != nil {
|
||
fmt.Printf("Gemini: Failed to create request: %v\n", err)
|
||
return "", fmt.Errorf("create request: %w", err)
|
||
}
|
||
|
||
req.Header.Set("Content-Type", "application/json")
|
||
|
||
fmt.Printf("Gemini: Executing HTTP request...\n")
|
||
resp, err := c.HTTPClient.Do(req)
|
||
if err != nil {
|
||
fmt.Printf("Gemini: HTTP request failed: %v\n", err)
|
||
return "", fmt.Errorf("send request: %w", err)
|
||
}
|
||
defer resp.Body.Close()
|
||
|
||
fmt.Printf("Gemini: Received response with status: %d\n", resp.StatusCode)
|
||
|
||
body, err := io.ReadAll(resp.Body)
|
||
if err != nil {
|
||
fmt.Printf("Gemini: Failed to read response body: %v\n", err)
|
||
return "", fmt.Errorf("read response: %w", err)
|
||
}
|
||
|
||
if resp.StatusCode != http.StatusOK {
|
||
fmt.Printf("Gemini: API error (status %d): %s\n", resp.StatusCode, string(body))
|
||
return "", fmt.Errorf("API error (status %d): %s", resp.StatusCode, string(body))
|
||
}
|
||
|
||
// 打印响应体用于调试
|
||
bodyPreview := string(body)
|
||
if len(body) > 500 {
|
||
bodyPreview = string(body[:500]) + "..."
|
||
}
|
||
fmt.Printf("Gemini: Response body: %s\n", bodyPreview)
|
||
|
||
var result GeminiTextResponse
|
||
if err := json.Unmarshal(body, &result); err != nil {
|
||
errorPreview := string(body)
|
||
if len(body) > 200 {
|
||
errorPreview = string(body[:200])
|
||
}
|
||
fmt.Printf("Gemini: Failed to parse response: %v\n", err)
|
||
return "", fmt.Errorf("parse response: %w, body preview: %s", err, errorPreview)
|
||
}
|
||
|
||
fmt.Printf("Gemini: Successfully parsed response, candidates count: %d\n", len(result.Candidates))
|
||
|
||
if len(result.Candidates) == 0 {
|
||
fmt.Printf("Gemini: No candidates in response\n")
|
||
return "", fmt.Errorf("no candidates in response")
|
||
}
|
||
|
||
if len(result.Candidates[0].Content.Parts) == 0 {
|
||
fmt.Printf("Gemini: No parts in first candidate\n")
|
||
return "", fmt.Errorf("no parts in response")
|
||
}
|
||
|
||
responseText := result.Candidates[0].Content.Parts[0].Text
|
||
fmt.Printf("Gemini: Generated text: %s\n", responseText)
|
||
|
||
return responseText, nil
|
||
}
|
||
|
||
func (c *GeminiClient) TestConnection() error {
|
||
fmt.Printf("Gemini: TestConnection called with BaseURL=%s, Model=%s, Endpoint=%s\n", c.BaseURL, c.Model, c.Endpoint)
|
||
_, err := c.GenerateText("Hello", "")
|
||
if err != nil {
|
||
fmt.Printf("Gemini: TestConnection failed: %v\n", err)
|
||
} else {
|
||
fmt.Printf("Gemini: TestConnection succeeded\n")
|
||
}
|
||
return err
|
||
}
|