2026-05-05 17:30:55 +08:00
|
|
|
|
"""
|
2026-05-08 01:59:19 +08:00
|
|
|
|
Agent 主图 - 标准 LangGraph 结构
|
|
|
|
|
|
将 ReAct 循环分离为独立的 agent/tools 节点
|
2026-05-05 17:30:55 +08:00
|
|
|
|
"""
|
|
|
|
|
|
|
2026-05-08 01:59:19 +08:00
|
|
|
|
from typing import Literal
|
2026-05-05 17:30:55 +08:00
|
|
|
|
from langgraph.graph import StateGraph, START, END
|
2026-05-07 02:56:35 +08:00
|
|
|
|
from backend.app.main_graph.state import AgentState
|
|
|
|
|
|
from backend.app.main_graph.nodes.memory_trigger import memory_trigger_node, set_mem0_client
|
|
|
|
|
|
from backend.app.main_graph.nodes.agent import create_agent_node
|
2026-05-08 01:59:19 +08:00
|
|
|
|
from backend.app.main_graph.nodes.tools import tools_node
|
|
|
|
|
|
from backend.app.main_graph.nodes.finalize import finalize_node
|
2026-05-08 00:29:12 +08:00
|
|
|
|
from backend.app.logger import info
|
2026-05-05 17:30:55 +08:00
|
|
|
|
|
|
|
|
|
|
|
2026-05-07 00:48:17 +08:00
|
|
|
|
def build_agent_graph(
|
2026-05-05 17:30:55 +08:00
|
|
|
|
chat_services: dict,
|
|
|
|
|
|
mem0_client=None,
|
2026-05-07 00:48:17 +08:00
|
|
|
|
max_steps: int = 10
|
2026-05-07 02:11:20 +08:00
|
|
|
|
):
|
2026-05-05 17:30:55 +08:00
|
|
|
|
"""
|
2026-05-08 01:59:19 +08:00
|
|
|
|
构建标准 Agent 图
|
|
|
|
|
|
|
|
|
|
|
|
节点:
|
|
|
|
|
|
- init_state: 初始化状态
|
|
|
|
|
|
- memory_trigger: 记忆触发
|
|
|
|
|
|
- agent: 单步推理
|
|
|
|
|
|
- tools: 工具执行
|
|
|
|
|
|
- finalize: 后处理
|
|
|
|
|
|
|
|
|
|
|
|
边:
|
|
|
|
|
|
- START -> init_state -> memory_trigger -> agent
|
|
|
|
|
|
- agent -> (条件边) -> tools 或 finalize
|
|
|
|
|
|
- tools -> agent (循环)
|
|
|
|
|
|
- finalize -> END
|
2026-05-05 17:30:55 +08:00
|
|
|
|
|
|
|
|
|
|
Args:
|
2026-05-07 00:48:17 +08:00
|
|
|
|
chat_services: 模型服务字典
|
|
|
|
|
|
mem0_client: 记忆客户端(可选)
|
|
|
|
|
|
max_steps: 最大步数限制
|
2026-05-05 17:30:55 +08:00
|
|
|
|
|
|
|
|
|
|
Returns:
|
2026-05-07 02:21:09 +08:00
|
|
|
|
构建好的 StateGraph(未编译)
|
2026-05-05 17:30:55 +08:00
|
|
|
|
"""
|
2026-05-08 01:59:19 +08:00
|
|
|
|
# 设置全局客户端
|
2026-05-05 17:30:55 +08:00
|
|
|
|
if mem0_client:
|
|
|
|
|
|
set_mem0_client(mem0_client)
|
|
|
|
|
|
|
2026-05-08 01:59:19 +08:00
|
|
|
|
# ========== 1. init_state 节点 ==========
|
2026-05-07 02:11:20 +08:00
|
|
|
|
async def init_state_node(state: AgentState):
|
2026-05-08 01:59:19 +08:00
|
|
|
|
info("[Init State] 初始化状态")
|
2026-05-07 02:11:20 +08:00
|
|
|
|
return {
|
2026-05-07 02:21:09 +08:00
|
|
|
|
"current_step": 0,
|
2026-05-08 01:59:19 +08:00
|
|
|
|
"max_steps": max_steps,
|
|
|
|
|
|
"tool_call_history": [],
|
|
|
|
|
|
"tool_result_history": [],
|
|
|
|
|
|
"tools_used": [],
|
|
|
|
|
|
"stop": False,
|
|
|
|
|
|
"stop_reason": "",
|
2026-05-07 02:11:20 +08:00
|
|
|
|
}
|
2026-05-07 00:48:17 +08:00
|
|
|
|
|
2026-05-08 01:59:19 +08:00
|
|
|
|
# ========== 2. Agent 节点 ==========
|
2026-05-08 00:29:12 +08:00
|
|
|
|
agent_node_fn = create_agent_node(chat_services)
|
2026-05-07 00:48:17 +08:00
|
|
|
|
|
2026-05-08 01:59:19 +08:00
|
|
|
|
# ========== 3. 条件边函数 ==========
|
|
|
|
|
|
def should_continue(state: AgentState) -> Literal["tools", "finalize"]:
|
|
|
|
|
|
"""根据 agent 节点输出决定下一步"""
|
|
|
|
|
|
# 手动停止标志
|
|
|
|
|
|
if getattr(state, "stop", False):
|
|
|
|
|
|
return "finalize"
|
|
|
|
|
|
|
|
|
|
|
|
# 检查是否有工具调用
|
|
|
|
|
|
last_msg = state.messages[-1]
|
|
|
|
|
|
if hasattr(last_msg, 'tool_calls') and last_msg.tool_calls:
|
|
|
|
|
|
return "tools"
|
2026-05-07 02:21:09 +08:00
|
|
|
|
|
2026-05-08 01:59:19 +08:00
|
|
|
|
return "finalize"
|
|
|
|
|
|
|
|
|
|
|
|
# ========== 4. 构建图 ==========
|
2026-05-07 02:11:20 +08:00
|
|
|
|
graph = StateGraph(AgentState)
|
2026-05-05 17:30:55 +08:00
|
|
|
|
|
2026-05-08 01:59:19 +08:00
|
|
|
|
# 添加节点
|
2026-05-07 00:48:17 +08:00
|
|
|
|
graph.add_node("init_state", init_state_node)
|
2026-05-05 17:30:55 +08:00
|
|
|
|
graph.add_node("memory_trigger", memory_trigger_node)
|
2026-05-07 02:21:09 +08:00
|
|
|
|
graph.add_node("agent", agent_node_fn)
|
2026-05-08 01:59:19 +08:00
|
|
|
|
graph.add_node("tools", tools_node)
|
|
|
|
|
|
graph.add_node("finalize", finalize_node)
|
2026-05-05 17:30:55 +08:00
|
|
|
|
|
2026-05-08 01:59:19 +08:00
|
|
|
|
# 边的连接
|
2026-05-07 00:48:17 +08:00
|
|
|
|
graph.add_edge(START, "init_state")
|
2026-05-08 01:59:19 +08:00
|
|
|
|
graph.add_edge("init_state", "memory_trigger")
|
|
|
|
|
|
graph.add_edge("memory_trigger", "agent")
|
2026-05-05 17:30:55 +08:00
|
|
|
|
|
2026-05-08 01:59:19 +08:00
|
|
|
|
# 条件边: agent -> tools 或 finalize
|
|
|
|
|
|
graph.add_conditional_edges(
|
|
|
|
|
|
"agent",
|
|
|
|
|
|
should_continue,
|
|
|
|
|
|
{
|
|
|
|
|
|
"tools": "tools",
|
|
|
|
|
|
"finalize": "finalize"
|
|
|
|
|
|
}
|
|
|
|
|
|
)
|
2026-05-05 17:30:55 +08:00
|
|
|
|
|
2026-05-08 01:59:19 +08:00
|
|
|
|
# 循环回边: tools -> agent
|
|
|
|
|
|
graph.add_edge("tools", "agent")
|
|
|
|
|
|
|
|
|
|
|
|
# 结束
|
2026-05-07 02:21:09 +08:00
|
|
|
|
graph.add_edge("finalize", END)
|
|
|
|
|
|
|
2026-05-08 01:59:19 +08:00
|
|
|
|
info("✅ [Graph Builder] 标准 Agent 图构建完成")
|
2026-05-07 00:48:17 +08:00
|
|
|
|
return graph
|