run.py 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199
  1. """
  2. 自动化投放 Agent 系统 — 主入口
  3. 运行方式:
  4. cd /Users/liulidong/project/agent/Agent
  5. python examples/auto_put_ad/run.py
  6. """
  7. import asyncio
  8. import os
  9. import sys
  10. from pathlib import Path
  11. # 代理设置
  12. os.environ.setdefault("HTTP_PROXY", "http://127.0.0.1:29758")
  13. os.environ.setdefault("HTTPS_PROXY", "http://127.0.0.1:29758")
  14. # 添加项目根目录到 Python 路径
  15. sys.path.insert(0, str(Path(__file__).parent.parent.parent))
  16. from dotenv import load_dotenv
  17. load_dotenv()
  18. from agent.core.runner import AgentRunner
  19. from agent.trace import FileSystemTraceStore, Trace, Message
  20. from agent.llm import create_openrouter_llm_call
  21. from agent.utils import setup_logging
  22. # 导入配置(使用绝对路径导入)
  23. from examples.auto_put_ad.config import MAIN_CONFIG, SKILLS_DIR, TRACE_STORE_PATH, LOG_LEVEL, LOG_FILE
  24. # 导入自定义工具(触发 @tool 注册)
  25. from examples.auto_put_ad.tools.ad_api import (
  26. account_get_info, ad_batch_update_status, ad_create, ad_get_list,
  27. ad_get_report, ad_update, asset_get_list, audience_get_list,
  28. creative_create, creative_get_report, creative_update,
  29. )
  30. from examples.auto_put_ad.tools.audience_tools import (
  31. audience_build_targeting, audience_recommend_targeting,
  32. )
  33. from examples.auto_put_ad.tools.budget_calc import (
  34. get_ad_performance, get_account_summary,
  35. compute_budget_thresholds, classify_ads, compute_bid_adjustment,
  36. bid_adjustment_execute,
  37. )
  38. from examples.auto_put_ad.tools.strategy_config import (
  39. load_strategy_config, update_strategy_config, get_config_history,
  40. )
  41. from examples.auto_put_ad.tools.data_query import (
  42. data_aggregate, data_query, get_ad_current_status,
  43. )
  44. from examples.auto_put_ad.tools.monitor_tools import (
  45. monitor_check_metrics, monitor_circuit_break,
  46. )
  47. async def init_project_env(messages=None):
  48. """供 api_server 可视化调用:返回 (runner, messages, config)"""
  49. base_dir = Path(__file__).parent
  50. # 读取 main.prompt
  51. main_prompt_path = base_dir / "prompts" / "main.prompt"
  52. system_prompt = ""
  53. if main_prompt_path.exists():
  54. system_prompt = main_prompt_path.read_text(encoding="utf-8")
  55. # 加载 presets(必须在创建 runner 之前)
  56. presets_path = base_dir / "presets.json"
  57. if presets_path.exists():
  58. from agent.core.presets import load_presets_from_json
  59. load_presets_from_json(str(presets_path))
  60. store = FileSystemTraceStore(base_path=TRACE_STORE_PATH)
  61. runner = AgentRunner(
  62. trace_store=store,
  63. llm_call=create_openrouter_llm_call(model=MAIN_CONFIG.model),
  64. skills_dir=SKILLS_DIR if Path(SKILLS_DIR).exists() else None,
  65. logger_name="agents.auto_put_ad",
  66. )
  67. config = MAIN_CONFIG
  68. if system_prompt:
  69. config.system_prompt = system_prompt
  70. # 如果前端没传 messages,给个默认的
  71. if not messages:
  72. messages = [{"role": "user", "content": "今天小程序预算10w"}]
  73. # 注入 system prompt 到 messages(run_api.py 的 config 合并会丢失 system_prompt,
  74. # 通过 messages 传递确保 system prompt 被 agent 接收)
  75. if system_prompt:
  76. has_system = any(m.get("role") == "system" for m in messages)
  77. if not has_system:
  78. messages = [{"role": "system", "content": system_prompt}] + messages
  79. return runner, messages, config
  80. async def main():
  81. """主函数"""
  82. base_dir = Path(__file__).parent
  83. # 初始化日志
  84. setup_logging(level=LOG_LEVEL, file=LOG_FILE)
  85. # 读取 main.prompt
  86. main_prompt_path = base_dir / "prompts" / "main.prompt"
  87. system_prompt = ""
  88. if main_prompt_path.exists():
  89. system_prompt = main_prompt_path.read_text(encoding="utf-8")
  90. # 加载 presets(必须在创建 runner 之前)
  91. presets_path = base_dir / "presets.json"
  92. if presets_path.exists():
  93. from agent.core.presets import load_presets_from_json
  94. load_presets_from_json(str(presets_path))
  95. # 创建 Runner(启用多 Agent)
  96. store = FileSystemTraceStore(base_path=TRACE_STORE_PATH)
  97. runner = AgentRunner(
  98. trace_store=store,
  99. llm_call=create_openrouter_llm_call(model=MAIN_CONFIG.model),
  100. skills_dir=SKILLS_DIR if Path(SKILLS_DIR).exists() else None,
  101. logger_name="agents.auto_put_ad",
  102. )
  103. # 如果有 system_prompt,注入到 config
  104. config = MAIN_CONFIG
  105. if system_prompt:
  106. config.system_prompt = system_prompt
  107. print("=" * 60)
  108. print(" 自动化投放 Agent 系统已启动")
  109. print("=" * 60)
  110. print("请输入投放任务(输入 'exit' 退出):")
  111. print("示例:")
  112. print(" - 今天小程序预算10w")
  113. print(" - 分析账户昨日投放效果,优化预算分配")
  114. print(" - 查询账户余额和今日消耗")
  115. print()
  116. while True:
  117. try:
  118. user_input = input("\n> ").strip()
  119. if not user_input:
  120. continue
  121. if user_input.lower() in ("exit", "quit", "q"):
  122. print("退出系统")
  123. break
  124. # 构建消息
  125. messages = [{"role": "user", "content": user_input}]
  126. # 每次对话用新的 trace
  127. config.trace_id = None
  128. print(f"\n🚀 执行任务: {user_input}\n")
  129. # 执行 Agent
  130. async for item in runner.run(messages=messages, config=config):
  131. if isinstance(item, Trace):
  132. print(f"[Trace] 状态: {item.status}")
  133. elif isinstance(item, Message):
  134. if item.role == "assistant" and item.content:
  135. content = item.content
  136. if isinstance(content, dict):
  137. text = content.get("text", "")
  138. else:
  139. text = content
  140. if text and text.strip():
  141. print(f"\n💭 {text}\n")
  142. elif item.role == "tool" and item.content:
  143. content = item.content
  144. if isinstance(content, str):
  145. text = content
  146. elif isinstance(content, dict):
  147. text = content.get("text", str(content))
  148. else:
  149. text = str(content)
  150. # 只打印前 500 字符避免刷屏
  151. if len(text) > 500:
  152. text = text[:500] + "..."
  153. print(f" [Tool] {text}")
  154. print("\n" + "=" * 60)
  155. print("✅ 任务完成")
  156. print("=" * 60)
  157. except KeyboardInterrupt:
  158. print("\n用户中断,退出系统")
  159. break
  160. except Exception as e:
  161. print(f"\n❌ 执行失败: {e}")
  162. import traceback
  163. traceback.print_exc()
  164. if __name__ == "__main__":
  165. asyncio.run(main())