mirror of
https://github.com/Ed1s0nZ/CyberStrikeAI.git
synced 2026-07-04 11:37:57 +02:00
264 lines
9.0 KiB
Go
264 lines
9.0 KiB
Go
package handler
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"net/http"
|
|
"strings"
|
|
"time"
|
|
|
|
"cyberstrike-ai/internal/config"
|
|
workflowrunner "cyberstrike-ai/internal/workflow"
|
|
|
|
"github.com/gin-gonic/gin"
|
|
"go.uber.org/zap"
|
|
)
|
|
|
|
func (h *AgentHandler) roleForWorkflow(req *ChatRequest) (config.RoleConfig, bool) {
|
|
if h == nil || h.config == nil || h.config.Roles == nil || req == nil {
|
|
return config.RoleConfig{}, false
|
|
}
|
|
roleName := strings.TrimSpace(req.Role)
|
|
if roleName == "" {
|
|
return config.RoleConfig{}, false
|
|
}
|
|
role, ok := h.config.Roles[roleName]
|
|
if !ok || !role.Enabled {
|
|
return config.RoleConfig{}, false
|
|
}
|
|
if role.Name == "" {
|
|
role.Name = roleName
|
|
}
|
|
if !workflowrunner.ShouldAutoRunRoleWorkflow(role) {
|
|
return config.RoleConfig{}, false
|
|
}
|
|
return role, true
|
|
}
|
|
|
|
func (h *AgentHandler) runRoleWorkflowStreamIfBound(
|
|
req *ChatRequest,
|
|
prep *multiAgentPrepared,
|
|
sendEvent func(eventType, message string, data interface{}),
|
|
) bool {
|
|
role, ok := h.roleForWorkflow(req)
|
|
if !ok || prep == nil {
|
|
return false
|
|
}
|
|
|
|
conversationID := prep.ConversationID
|
|
assistantMessageID := prep.AssistantMessageID
|
|
userMessage := ""
|
|
if req != nil {
|
|
userMessage = req.Message
|
|
}
|
|
|
|
taskStatus := "completed"
|
|
taskOwned := false
|
|
defer func() {
|
|
if taskOwned {
|
|
h.tasks.FinishTask(conversationID, taskStatus)
|
|
}
|
|
}()
|
|
|
|
baseCtx, cancelWithCause := context.WithCancelCause(context.Background())
|
|
defer cancelWithCause(nil)
|
|
taskCtx, timeoutCancel := context.WithTimeout(baseCtx, 600*time.Minute)
|
|
defer timeoutCancel()
|
|
|
|
if _, err := h.tasks.StartTask(conversationID, userMessage, cancelWithCause); err != nil {
|
|
var errorMsg string
|
|
if errors.Is(err, ErrTaskAlreadyRunning) {
|
|
errorMsg = "⚠️ 当前会话已有任务正在执行中,请等待当前任务完成或点击「停止任务」后再尝试。"
|
|
sendEvent("error", errorMsg, map[string]interface{}{
|
|
"conversationId": conversationID,
|
|
"errorType": "task_already_running",
|
|
})
|
|
} else {
|
|
errorMsg = "❌ 无法启动任务: " + err.Error()
|
|
sendEvent("error", errorMsg, nil)
|
|
}
|
|
if assistantMessageID != "" {
|
|
_, _ = h.db.Exec("UPDATE messages SET content = ?, updated_at = ? WHERE id = ?", errorMsg, time.Now(), assistantMessageID)
|
|
}
|
|
sendEvent("done", "", map[string]interface{}{"conversationId": conversationID})
|
|
return true
|
|
}
|
|
taskOwned = true
|
|
|
|
progress := h.createProgressCallback(taskCtx, cancelWithCause, conversationID, assistantMessageID, sendEvent)
|
|
result, err := workflowrunner.RunRoleBoundWorkflow(taskCtx, workflowrunner.RunArgs{
|
|
DB: h.db,
|
|
Logger: h.logger,
|
|
Role: role,
|
|
AppCfg: h.config,
|
|
Agent: h.agent,
|
|
ConversationID: conversationID,
|
|
ProjectID: h.conversationProjectID(conversationID),
|
|
UserMessage: prep.FinalMessage,
|
|
History: prep.History,
|
|
RoleTools: prep.RoleTools,
|
|
AgentsMarkdownDir: h.agentsMarkdownDir,
|
|
SystemPromptExtra: h.agentSessionContextBlock(conversationID),
|
|
AssistantMessageID: assistantMessageID,
|
|
Progress: progress,
|
|
})
|
|
if err != nil {
|
|
cause := context.Cause(baseCtx)
|
|
if errors.Is(cause, ErrTaskCancelled) {
|
|
taskStatus = "cancelled"
|
|
h.tasks.UpdateTaskStatus(conversationID, taskStatus)
|
|
cancelMsg := "任务已被用户取消,后续操作已停止。"
|
|
if assistantMessageID != "" {
|
|
if err := h.appendAssistantMessageNotice(assistantMessageID, cancelMsg); err != nil {
|
|
h.logger.Warn("更新取消后的助手消息失败", zap.Error(err))
|
|
}
|
|
_ = h.db.AddProcessDetail(assistantMessageID, conversationID, "cancelled", cancelMsg, nil)
|
|
}
|
|
sendEvent("cancelled", cancelMsg, map[string]interface{}{
|
|
"conversationId": conversationID,
|
|
"messageId": assistantMessageID,
|
|
})
|
|
sendEvent("done", "", map[string]interface{}{"conversationId": conversationID})
|
|
return true
|
|
}
|
|
if errors.Is(err, context.DeadlineExceeded) || errors.Is(context.Cause(taskCtx), context.DeadlineExceeded) {
|
|
taskStatus = "timeout"
|
|
h.tasks.UpdateTaskStatus(conversationID, taskStatus)
|
|
timeoutMsg := "任务执行超时,已自动终止。"
|
|
if assistantMessageID != "" {
|
|
_, _ = h.db.Exec("UPDATE messages SET content = ?, updated_at = ? WHERE id = ?", timeoutMsg, time.Now(), assistantMessageID)
|
|
_ = h.db.AddProcessDetail(assistantMessageID, conversationID, "timeout", timeoutMsg, nil)
|
|
}
|
|
sendEvent("error", timeoutMsg, map[string]interface{}{
|
|
"conversationId": conversationID,
|
|
"messageId": assistantMessageID,
|
|
"errorType": "timeout",
|
|
})
|
|
sendEvent("done", "", map[string]interface{}{"conversationId": conversationID})
|
|
return true
|
|
}
|
|
errMsg := "执行角色绑定流程失败: " + err.Error()
|
|
taskStatus = "failed"
|
|
h.tasks.UpdateTaskStatus(conversationID, taskStatus)
|
|
if assistantMessageID != "" {
|
|
_, _ = h.db.Exec("UPDATE messages SET content = ?, updated_at = ? WHERE id = ?", errMsg, time.Now(), assistantMessageID)
|
|
_ = h.db.AddProcessDetail(assistantMessageID, conversationID, "error", errMsg, nil)
|
|
}
|
|
sendEvent("error", errMsg, map[string]interface{}{"conversationId": conversationID})
|
|
sendEvent("done", "", map[string]interface{}{"conversationId": conversationID})
|
|
return true
|
|
}
|
|
if prep.AssistantMessageID != "" {
|
|
_ = h.db.UpdateAssistantMessageFinalize(prep.AssistantMessageID, result.Response, nil, "")
|
|
}
|
|
payload := map[string]interface{}{
|
|
"conversationId": prep.ConversationID,
|
|
"messageId": prep.AssistantMessageID,
|
|
"agentMode": "workflow",
|
|
"workflowRunId": result.RunID,
|
|
}
|
|
if result.AwaitingHITL {
|
|
payload["workflowStatus"] = "awaiting_hitl"
|
|
payload["awaitingHitl"] = true
|
|
}
|
|
sendEvent("response", result.Response, payload)
|
|
sendEvent("done", "", map[string]interface{}{"conversationId": prep.ConversationID})
|
|
return true
|
|
}
|
|
|
|
func (h *AgentHandler) runRoleWorkflowJSONIfBound(c *gin.Context, req *ChatRequest, prep *multiAgentPrepared) bool {
|
|
role, ok := h.roleForWorkflow(req)
|
|
if !ok || prep == nil {
|
|
return false
|
|
}
|
|
|
|
conversationID := prep.ConversationID
|
|
assistantMessageID := prep.AssistantMessageID
|
|
userMessage := ""
|
|
if req != nil {
|
|
userMessage = req.Message
|
|
}
|
|
|
|
taskStatus := "completed"
|
|
taskOwned := false
|
|
defer func() {
|
|
if taskOwned {
|
|
h.tasks.FinishTask(conversationID, taskStatus)
|
|
}
|
|
}()
|
|
|
|
baseCtx, cancelWithCause := context.WithCancelCause(c.Request.Context())
|
|
defer cancelWithCause(nil)
|
|
taskCtx, timeoutCancel := context.WithTimeout(baseCtx, 600*time.Minute)
|
|
defer timeoutCancel()
|
|
|
|
if _, err := h.tasks.StartTask(conversationID, userMessage, cancelWithCause); err != nil {
|
|
if errors.Is(err, ErrTaskAlreadyRunning) {
|
|
c.JSON(http.StatusConflict, gin.H{
|
|
"error": "⚠️ 当前会话已有任务正在执行中,请等待当前任务完成或点击「停止任务」后再尝试。",
|
|
"conversationId": conversationID,
|
|
"errorType": "task_already_running",
|
|
})
|
|
} else {
|
|
c.JSON(http.StatusInternalServerError, gin.H{"error": "❌ 无法启动任务: " + err.Error()})
|
|
}
|
|
return true
|
|
}
|
|
taskOwned = true
|
|
|
|
progress := h.createProgressCallback(taskCtx, cancelWithCause, conversationID, assistantMessageID, nil)
|
|
result, err := workflowrunner.RunRoleBoundWorkflow(taskCtx, workflowrunner.RunArgs{
|
|
DB: h.db,
|
|
Logger: h.logger,
|
|
Role: role,
|
|
AppCfg: h.config,
|
|
Agent: h.agent,
|
|
ConversationID: conversationID,
|
|
ProjectID: h.conversationProjectID(conversationID),
|
|
UserMessage: prep.FinalMessage,
|
|
History: prep.History,
|
|
RoleTools: prep.RoleTools,
|
|
AgentsMarkdownDir: h.agentsMarkdownDir,
|
|
SystemPromptExtra: h.agentSessionContextBlock(conversationID),
|
|
AssistantMessageID: assistantMessageID,
|
|
Progress: progress,
|
|
})
|
|
if err != nil {
|
|
cause := context.Cause(baseCtx)
|
|
if errors.Is(cause, ErrTaskCancelled) {
|
|
taskStatus = "cancelled"
|
|
cancelMsg := "任务已被用户取消,后续操作已停止。"
|
|
if assistantMessageID != "" {
|
|
_ = h.appendAssistantMessageNotice(assistantMessageID, cancelMsg)
|
|
_ = h.db.AddProcessDetail(assistantMessageID, conversationID, "cancelled", cancelMsg, nil)
|
|
}
|
|
c.JSON(http.StatusOK, gin.H{
|
|
"status": "cancelled",
|
|
"message": cancelMsg,
|
|
"conversationId": conversationID,
|
|
})
|
|
return true
|
|
}
|
|
errMsg := "执行角色绑定流程失败: " + err.Error()
|
|
taskStatus = "failed"
|
|
if assistantMessageID != "" {
|
|
_, _ = h.db.Exec("UPDATE messages SET content = ?, updated_at = ? WHERE id = ?", errMsg, time.Now(), assistantMessageID)
|
|
}
|
|
c.JSON(http.StatusInternalServerError, gin.H{"error": errMsg, "conversationId": conversationID})
|
|
return true
|
|
}
|
|
if prep.AssistantMessageID != "" {
|
|
_ = h.db.UpdateAssistantMessageFinalize(prep.AssistantMessageID, result.Response, nil, "")
|
|
}
|
|
c.JSON(http.StatusOK, gin.H{
|
|
"response": result.Response,
|
|
"conversationId": prep.ConversationID,
|
|
"assistantMessageId": prep.AssistantMessageID,
|
|
"agentMode": "workflow",
|
|
"workflowRunId": result.RunID,
|
|
"workflowStatus": result.Status,
|
|
"awaitingHitl": result.AwaitingHITL,
|
|
})
|
|
return true
|
|
}
|