2026-03-11 16:25:48 +08:00
|
|
|
|
package service
|
|
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
|
"bytes"
|
|
|
|
|
|
"encoding/json"
|
|
|
|
|
|
"fmt"
|
|
|
|
|
|
"io"
|
2026-03-11 17:22:40 +08:00
|
|
|
|
"log"
|
2026-03-11 16:25:48 +08:00
|
|
|
|
"net/http"
|
|
|
|
|
|
"time"
|
2026-03-11 17:22:40 +08:00
|
|
|
|
|
|
|
|
|
|
"x-agents/server/internal/repository"
|
2026-03-11 16:25:48 +08:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
// AgentChatRequest Python Agent 对话请求
|
|
|
|
|
|
type AgentChatRequest struct {
|
2026-03-11 17:22:40 +08:00
|
|
|
|
AgentID int `json:"agent_id"`
|
|
|
|
|
|
Message string `json:"message"`
|
|
|
|
|
|
UserID int `json:"user_id"`
|
|
|
|
|
|
SessionID string `json:"session_id,omitempty"`
|
|
|
|
|
|
ModelID string `json:"model_id,omitempty"`
|
|
|
|
|
|
ModelName string `json:"model_name,omitempty"`
|
|
|
|
|
|
ModelProvider string `json:"model_provider,omitempty"`
|
|
|
|
|
|
APIKey string `json:"api_key,omitempty"`
|
|
|
|
|
|
BaseURL string `json:"base_url,omitempty"`
|
2026-03-11 16:25:48 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// AgentChatResponse Python Agent 对话响应
|
|
|
|
|
|
type AgentChatResponse struct {
|
|
|
|
|
|
AgentID int `json:"agent_id"`
|
|
|
|
|
|
Response string `json:"response"`
|
|
|
|
|
|
ToolCalls []interface{} `json:"tool_calls"`
|
|
|
|
|
|
TokensUsed int `json:"tokens_used"`
|
|
|
|
|
|
DurationMs int `json:"duration_ms"`
|
|
|
|
|
|
SessionID string `json:"session_id"`
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// TeamChatRequest 多智能体群聊请求
|
|
|
|
|
|
type TeamChatRequest struct {
|
|
|
|
|
|
SupervisorAgentID int `json:"supervisor_agent_id"`
|
|
|
|
|
|
MemberAgentIDs []int `json:"member_agent_ids"`
|
|
|
|
|
|
Message string `json:"message"`
|
|
|
|
|
|
UserID int `json:"user_id"`
|
|
|
|
|
|
SessionID string `json:"session_id,omitempty"`
|
|
|
|
|
|
Strategy string `json:"strategy,omitempty"`
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// TeamChatResponse 多智能体群聊响应
|
|
|
|
|
|
type TeamChatResponse struct {
|
|
|
|
|
|
SupervisorAgentID int `json:"supervisor_agent_id"`
|
|
|
|
|
|
Response string `json:"response"`
|
|
|
|
|
|
SubtaskResults []interface{} `json:"subtask_results"`
|
|
|
|
|
|
Strategy string `json:"strategy"`
|
|
|
|
|
|
DurationMs int `json:"duration_ms"`
|
|
|
|
|
|
SessionID string `json:"session_id"`
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// AgentService Python Agent 服务
|
|
|
|
|
|
type AgentService struct {
|
|
|
|
|
|
pythonURL string
|
|
|
|
|
|
client *http.Client
|
2026-03-11 17:22:40 +08:00
|
|
|
|
modelRepo *repository.ModelRepository
|
2026-03-11 16:25:48 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// NewAgentService 创建 Agent 服务
|
2026-03-11 17:22:40 +08:00
|
|
|
|
func NewAgentService(pythonURL string, modelRepo *repository.ModelRepository) *AgentService {
|
2026-03-11 16:25:48 +08:00
|
|
|
|
return &AgentService{
|
|
|
|
|
|
pythonURL: pythonURL,
|
|
|
|
|
|
client: &http.Client{
|
|
|
|
|
|
Timeout: 120 * time.Second, // Agent 可能需要较长时间
|
|
|
|
|
|
},
|
2026-03-11 17:22:40 +08:00
|
|
|
|
modelRepo: modelRepo,
|
2026-03-11 16:25:48 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Chat 单智能体对话
|
|
|
|
|
|
func (s *AgentService) Chat(req AgentChatRequest) (*AgentChatResponse, error) {
|
2026-03-11 17:22:40 +08:00
|
|
|
|
// 如果传入了 model_id,查询模型配置获取 api_key 和 base_url
|
|
|
|
|
|
log.Printf("[AgentService] Chat called, model_id: %s, modelRepo: %v", req.ModelID, s.modelRepo != nil)
|
|
|
|
|
|
|
|
|
|
|
|
if req.ModelID != "" && s.modelRepo != nil {
|
|
|
|
|
|
model, err := s.modelRepo.FindByID(req.ModelID)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
log.Printf("[AgentService] Error finding model: %v", err)
|
|
|
|
|
|
} else if model != nil {
|
|
|
|
|
|
log.Printf("[AgentService] Found model: id=%s, provider=%s, model=%s, base_url=%s, api_key_len=%d",
|
|
|
|
|
|
model.ID, model.Provider, model.Model, model.BaseURL, len(model.APIKey))
|
|
|
|
|
|
req.APIKey = model.APIKey
|
|
|
|
|
|
req.BaseURL = model.BaseURL
|
|
|
|
|
|
req.ModelProvider = model.Provider
|
|
|
|
|
|
req.ModelName = model.Model
|
|
|
|
|
|
log.Printf("[AgentService] Set req.APIKey=%s, req.BaseURL=%s", req.APIKey[:10]+"...", req.BaseURL)
|
|
|
|
|
|
} else {
|
|
|
|
|
|
log.Printf("[AgentService] Model not found for id: %s", req.ModelID)
|
|
|
|
|
|
}
|
|
|
|
|
|
} else if s.modelRepo == nil {
|
|
|
|
|
|
log.Printf("[AgentService] WARNING: modelRepo is nil!")
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 打印传给 Python 的请求内容
|
|
|
|
|
|
apiKeyPreview := ""
|
|
|
|
|
|
if req.APIKey != "" {
|
|
|
|
|
|
apiKeyPreview = req.APIKey
|
|
|
|
|
|
if len(apiKeyPreview) > 10 {
|
|
|
|
|
|
apiKeyPreview = apiKeyPreview[:10] + "..."
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
log.Printf("[AgentService] Sending to Python: model_id=%s, api_key=%s, base_url=%s, provider=%s, model=%s",
|
|
|
|
|
|
req.ModelID, apiKeyPreview, req.BaseURL, req.ModelProvider, req.ModelName)
|
|
|
|
|
|
|
2026-03-11 16:25:48 +08:00
|
|
|
|
url := fmt.Sprintf("%s/agent/chat", s.pythonURL)
|
|
|
|
|
|
|
|
|
|
|
|
jsonData, err := json.Marshal(req)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return nil, fmt.Errorf("failed to marshal request: %w", err)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
httpReq, err := http.NewRequest("POST", url, bytes.NewBuffer(jsonData))
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return nil, fmt.Errorf("failed to create request: %w", err)
|
|
|
|
|
|
}
|
|
|
|
|
|
httpReq.Header.Set("Content-Type", "application/json")
|
|
|
|
|
|
|
|
|
|
|
|
resp, err := s.client.Do(httpReq)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return nil, fmt.Errorf("failed to call python agent: %w", err)
|
|
|
|
|
|
}
|
|
|
|
|
|
defer resp.Body.Close()
|
|
|
|
|
|
|
|
|
|
|
|
body, err := io.ReadAll(resp.Body)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return nil, fmt.Errorf("failed to read response: %w", err)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if resp.StatusCode != http.StatusOK {
|
|
|
|
|
|
return nil, fmt.Errorf("python agent error: %s", string(body))
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
var result AgentChatResponse
|
|
|
|
|
|
if err := json.Unmarshal(body, &result); err != nil {
|
|
|
|
|
|
return nil, fmt.Errorf("failed to unmarshal response: %w", err)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return &result, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// TeamChat 多智能体群聊
|
|
|
|
|
|
func (s *AgentService) TeamChat(req TeamChatRequest) (*TeamChatResponse, error) {
|
|
|
|
|
|
url := fmt.Sprintf("%s/agent/team/chat", s.pythonURL)
|
|
|
|
|
|
|
|
|
|
|
|
// 设置默认策略
|
|
|
|
|
|
if req.Strategy == "" {
|
|
|
|
|
|
req.Strategy = "parallel"
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
jsonData, err := json.Marshal(req)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return nil, fmt.Errorf("failed to marshal request: %w", err)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
httpReq, err := http.NewRequest("POST", url, bytes.NewBuffer(jsonData))
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return nil, fmt.Errorf("failed to create request: %w", err)
|
|
|
|
|
|
}
|
|
|
|
|
|
httpReq.Header.Set("Content-Type", "application/json")
|
|
|
|
|
|
|
|
|
|
|
|
resp, err := s.client.Do(httpReq)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return nil, fmt.Errorf("failed to call python agent: %w", err)
|
|
|
|
|
|
}
|
|
|
|
|
|
defer resp.Body.Close()
|
|
|
|
|
|
|
|
|
|
|
|
body, err := io.ReadAll(resp.Body)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return nil, fmt.Errorf("failed to read response: %w", err)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if resp.StatusCode != http.StatusOK {
|
|
|
|
|
|
return nil, fmt.Errorf("python agent error: %s", string(body))
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
var result TeamChatResponse
|
|
|
|
|
|
if err := json.Unmarshal(body, &result); err != nil {
|
|
|
|
|
|
return nil, fmt.Errorf("failed to unmarshal response: %w", err)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return &result, nil
|
|
|
|
|
|
}
|