refactor: split order handling and ai routing flow from agent

This commit is contained in:
2026-03-01 15:02:27 +08:00
parent e62b39e0c3
commit dff4a8baaa
3 changed files with 207 additions and 122 deletions

View File

@@ -27,6 +27,8 @@ from core.quote_state_machine import QuoteStateMachine
from services.risk_service import RiskService
from core.agent_pre_rules import AgentPreRuleService
from core.find_image_flow import handle_find_image_batch_flow
from core.order_flow import handle_order_notification
from core.ai_reply_flow import execute_ai_turn
load_dotenv()
@@ -1790,41 +1792,9 @@ class CustomerServiceAgent:
state.last_update = datetime.now().isoformat()
# 订单通知前置处理
if "系统订单信息" in message.msg or "订单状态" in message.msg:
_, order_block = self._split_customer_text(message.msg)
customer_text, _ = self._split_customer_text(message.msg)
order = self._parse_order_info(order_block or message.msg)
pay_status = order.get("pay_status", "")
order_status = order.get("order_status", "")
paid_keywords = ["等待发货", "已付款", "付款成功", "买家已付款"]
is_paid = any(kw in pay_status or kw in order_status for kw in paid_keywords)
if is_paid:
# 订单金额核查:对比报价和实付金额
asyncio.create_task(self._check_order_amount(
message.from_id, order, message.acc_id
))
# 成交记录:写入数据库供日报分析
asyncio.create_task(self._record_deal_success(
message.from_id, message.from_name, message.acc_id, message.acc_type,
order, state
))
# 已付款:触发 Gemini 作图
try:
from core.workflow import workflow
asyncio.create_task(workflow.trigger_processing_on_payment(
customer_id=message.from_id,
acc_id=message.acc_id,
acc_type=message.acc_type,
))
except Exception as e:
print(f"[Agent] 触发作图失败: {e}")
elif not customer_text:
# 非付款 + 没有客户文字 → 直接静默,不调用 AI
print(f"[Agent] 订单通知静默({pay_status or order_status}),跳过回复")
return AgentResponse(reply="", should_reply=False, need_transfer=False)
order_response = await handle_order_notification(self, message=message, state=state)
if isinstance(order_response, AgentResponse):
return order_response
customer_text, _ = self._split_customer_text(message.msg)
shop_type = _get_shop_type(message.acc_id or "", message.goods_name or "")
@@ -1874,93 +1844,14 @@ class CustomerServiceAgent:
self._log_block("PROMPT->AI 前置提示词", user_prompt)
try:
msg_lower = message.msg.lower()
pricing_kw = ["多少钱", "多少一张", "报价", "给个价", "几块", "价位", "能便宜点吗"]
processing_kw = ["安排", "处理一下", "开始做", "做一下", "尽快", "加急", "付款了", "已付款"]
similar_kw = ["有一样的", "有一样吗", "一样的吗", "类似的", "类似的吗", "同款", "相似", "类似吗"]
order_markers = ["[系统订单信息]", "订单状态", "买家已付款"]
risk_kw = [
"黄色", "擦边", "色情", "涉黄", "涉政", "政治", "", "不雅",
"天安门", "政治人物", "政治事件", "领导人", "党政",
"习近平", "毛泽东", "邓小平", "江泽民", "胡锦涛",
"特朗普", "拜登", "普京", "泽连斯基",
"地图", "地形图", "行政区划图", "卫星地图",
]
target_agent = self.agent_after_sale if state.stage == "售后" else self.agent
risk_hit = any(k in msg_lower for k in risk_kw) or self._is_political_inquiry(message.msg) or self._is_map_inquiry(message.msg)
if risk_hit:
target_agent = self.agent_risk
elif any(k in message.msg for k in order_markers):
target_agent = self.agent_order
elif any(k in msg_lower for k in processing_kw):
target_agent = self.agent_processing
elif any(k in msg_lower for k in pricing_kw):
target_agent = self.agent_pricing
elif any(k in msg_lower for k in similar_kw):
target_agent = self.agent_similar
result = await target_agent.run(user_prompt, deps=deps, message_history=history)
# 更新历史,最多保留最近 30 条消息防止 token 超限
self.message_histories[message.from_id] = result.all_messages()[-30:]
reply_text = self._colloquialize_reply(self._normalize_reply_text(result.output))
# 价格谈判与信任建立固定策略(避免只回“最低了/先拍下”)
strategy_reply = self._negotiation_strategy_reply(message.msg, state)
if strategy_reply:
reply_text = strategy_reply
# 拦截超低杀价:客户报价低于底线时,统一礼貌拒绝
try:
from config.config import MIN_PRICE_FLOOR
import re
offer = None
m = re.search(r'(\d{1,4})\s*(?:元|块|块钱|元钱)\b', message.msg)
if m:
offer = int(m.group(1))
else:
m2 = re.search(r'(?:能|可以|可否|能否)\s*(\d{1,4})\b', message.msg)
offer = int(m2.group(1)) if m2 else None
st = self._get_conversation_state(message.from_id)
floor = st.last_min_price if isinstance(st.last_min_price, int) and st.last_min_price > 0 else MIN_PRICE_FLOOR
if offer is not None and offer < floor:
reply_text = "不好意思"
except Exception:
pass
# 降限若AI在回复中给出小于底线的报价提升到>=底线且为5的倍数
try:
from config.config import MIN_PRICE_FLOOR
st = self._get_conversation_state(message.from_id)
floor = st.last_min_price if isinstance(st.last_min_price, int) and st.last_min_price > 0 else MIN_PRICE_FLOOR
def _adjust(text: str) -> str:
import re
def _repl(m):
num = int(m.group(1))
adj = max(floor, round(num / 5) * 5)
return m.group(0).replace(str(num), str(adj))
patterns = [
r'按(\d{1,4})元',
r'报价[:]\s*(\d{1,4})\s*元',
r'(\d{1,4})\s*元一张',
r'打包(\d{1,4})\s*元',
]
t = text
for p in patterns:
t = re.sub(p, _repl, t)
return t
reply_text = _adjust(reply_text or "")
except Exception:
pass
# 打印工具调用记录
for msg in result.new_messages():
for part in getattr(msg, 'parts', []):
part_type = type(part).__name__
if 'ToolCall' in part_type:
print(f"{self.C_TOOL}[THINK/TOOL_CALL]{self.C_RESET} {getattr(part, 'tool_name', '')}({getattr(part, 'args', '')})")
elif 'ToolReturn' in part_type:
ret = str(getattr(part, 'content', ''))[:120]
print(f"{self.C_TOOL}[THINK/TOOL_RETURN]{self.C_RESET} {ret}")
print(f"{self.C_THINK}[THINK/RAW_OUTPUT]{self.C_RESET} {repr(reply_text)}")
reply_text = await execute_ai_turn(
self,
message=message,
state=state,
user_prompt=user_prompt,
deps=deps,
history=history,
)
except Exception as e:
err_str = str(e)
print(f"[Agent] AI 调用失败: {e},使用兜底回复")