feat: 重构项目结构并添加新功能
- 新增Cloudflare AutoRAG/Vectorize集成文档 - 实现Vertex AI记忆银行功能 - 重构项目目录结构,清理无用文件 - 更新README以反映最新架构 - 添加Google ADK集成测试脚本 - 完善需求文档和设计规范
This commit is contained in:
		
							parent
							
								
									26338d48cf
								
							
						
					
					
						commit
						c4e8cfefc7
					
				|  | @ -0,0 +1,136 @@ | |||
| # 稷下学宫 Google ADK 迁移状态报告 | ||||
| 
 | ||||
| ## 📊 迁移进度概览 | ||||
| 
 | ||||
| ### ✅ 已完成的任务 | ||||
| 
 | ||||
| #### 1. 基础设施迁移 | ||||
| - [x] **Google ADK 安装**: 成功安装 Google ADK 1.10.0 | ||||
| - [x] **API 密钥配置**: 已在 Doppler 中配置 `GOOGLE_API_KEY` | ||||
| - [x] **环境验证**: 基础测试通过,智能体创建成功 | ||||
| 
 | ||||
| #### 2. 配置系统更新 | ||||
| - [x] **doppler_config.py 增强**: | ||||
|   - 新增 `get_google_api_key()` 函数 | ||||
|   - 新增 `get_google_genai_config()` 函数 | ||||
|   - 更新 `validate_config()` 支持三种模式: | ||||
|     - `openrouter`: 纯 OpenRouter 模式 | ||||
|     - `google_adk`: 纯 Google ADK 模式 | ||||
|     - `hybrid`: 混合模式(当前使用) | ||||
| 
 | ||||
| #### 3. 测试系统建立 | ||||
| - [x] **基础测试**: `test_google_adk.py` - 验证 ADK 安装和配置 | ||||
| - [x] **智能体测试**: `adk_debate_test.py` - 八仙智能体创建测试 | ||||
| - [x] **论道原型**: `adk_simple_debate.py` - 智能体基础功能验证 | ||||
| 
 | ||||
| #### 4. 文档更新 | ||||
| - [x] **README.md**: 新增 Google ADK 安装和配置说明 | ||||
| - [x] **requirements.txt**: 添加 Google ADK 依赖说明 | ||||
| - [x] **迁移指南**: 完整的 `GOOGLE_ADK_MIGRATION_GUIDE.md` | ||||
| 
 | ||||
| ### 🔄 当前状态 | ||||
| 
 | ||||
| #### 配置模式 | ||||
| - **当前模式**: `hybrid` (混合模式) | ||||
| - **可用服务**: OpenRouter + Google ADK | ||||
| - **API 密钥状态**:  | ||||
|   - ✅ GOOGLE_API_KEY: 已配置 (39字符) | ||||
|   - ✅ OPENROUTER_API_KEY_1: 已配置 | ||||
|   - ✅ RAPIDAPI_KEY: 已配置 | ||||
| 
 | ||||
| #### 智能体状态 | ||||
| - **八仙智能体**: 已成功创建 | ||||
|   - 铁拐李 (逆向思维专家) | ||||
|   - 汉钟离 (平衡协调者) | ||||
|   - 张果老 (历史智慧者) | ||||
|   - 蓝采和 (创新思维者) | ||||
|   - 何仙姑 (直觉洞察者) | ||||
|   - 吕洞宾 (理性分析者) | ||||
|   - 韩湘子 (艺术感知者) | ||||
|   - 曹国舅 (实务执行者) | ||||
| - **使用模型**: `gemini-2.0-flash-exp` | ||||
| 
 | ||||
| ### 🚧 待完成的任务 | ||||
| 
 | ||||
| #### 1. 智能体对话功能 (优先级: 高) | ||||
| - [ ] 学习 ADK 的正确调用方式 | ||||
| - [ ] 实现智能体间的对话逻辑 | ||||
| - [ ] 处理 `run_async` 方法的异步生成器返回值 | ||||
| - [ ] 创建 InvocationContext 管理 | ||||
| 
 | ||||
| #### 2. 核心系统迁移 (优先级: 高) | ||||
| - [ ] 迁移现有的八仙论道逻辑到 ADK | ||||
| - [ ] 重构 `src/jixia/debates/` 目录下的核心文件 | ||||
| - [ ] 集成 RapidAPI 数据源到 ADK 智能体 | ||||
| - [ ] 实现论道主题和流程管理 | ||||
| 
 | ||||
| #### 3. 界面集成 (优先级: 中) | ||||
| - [ ] 更新 Streamlit 界面以支持 ADK | ||||
| - [ ] 修改 `src/streamlit_app.py` | ||||
| - [ ] 适配新的智能体调用方式 | ||||
| - [ ] 保持现有的用户体验 | ||||
| 
 | ||||
| #### 4. 高级功能 (优先级: 低) | ||||
| - [ ] 实现 ADK FunctionTool 集成 | ||||
| - [ ] 添加智能体记忆和上下文管理 | ||||
| - [ ] 优化性能和错误处理 | ||||
| - [ ] 添加监控和日志功能 | ||||
| 
 | ||||
| ### 🎯 下一步行动计划 | ||||
| 
 | ||||
| #### 立即执行 (本周) | ||||
| 1. **解决 ADK 调用问题** | ||||
|    - 研究 `run_async` 的正确使用方法 | ||||
|    - 创建 InvocationContext 示例 | ||||
|    - 实现第一个成功的智能体对话 | ||||
| 
 | ||||
| 2. **创建工作原型** | ||||
|    - 实现铁拐李和吕洞宾的简单对话 | ||||
|    - 验证论道逻辑的可行性 | ||||
|    - 测试多轮对话功能 | ||||
| 
 | ||||
| #### 短期目标 (本月) | ||||
| 1. **完成核心迁移** | ||||
|    - 迁移所有八仙智能体 | ||||
|    - 实现完整的论道流程 | ||||
|    - 集成现有数据源 | ||||
| 
 | ||||
| 2. **界面适配** | ||||
|    - 更新 Streamlit 应用 | ||||
|    - 保持功能完整性 | ||||
|    - 优化用户体验 | ||||
| 
 | ||||
| ### 📈 技术优势 | ||||
| 
 | ||||
| #### Google ADK 带来的改进 | ||||
| 1. **统一模型生态**: 直接使用 Gemini 系列模型 | ||||
| 2. **官方支持**: Google 官方维护的框架 | ||||
| 3. **更好的集成**: 与 Google 服务深度集成 | ||||
| 4. **开发工具**: `adk web`, `adk run`, `adk api_server` | ||||
| 5. **性能优化**: 原生支持异步和流式处理 | ||||
| 
 | ||||
| #### 保留的核心价值 | ||||
| 1. **稷下学宫哲学框架**: 完全保留 | ||||
| 2. **八仙角色设定**: 无缝迁移 | ||||
| 3. **RapidAPI 数据源**: 继续使用 | ||||
| 4. **MongoDB 数据库**: 保持不变 | ||||
| 5. **Doppler 配置管理**: 增强支持 | ||||
| 
 | ||||
| ### 🔍 风险评估 | ||||
| 
 | ||||
| #### 技术风险 | ||||
| - **学习曲线**: ADK 框架需要时间熟悉 | ||||
| - **API 变更**: Google ADK 仍在快速发展 | ||||
| - **兼容性**: 需要确保与现有系统的兼容 | ||||
| 
 | ||||
| #### 缓解措施 | ||||
| - **渐进迁移**: 保持混合模式,逐步切换 | ||||
| - **充分测试**: 每个功能都有对应的测试 | ||||
| - **文档完善**: 详细记录迁移过程和决策 | ||||
| 
 | ||||
| --- | ||||
| 
 | ||||
| **最后更新**: 2024年12月 | ||||
| **迁移负责人**: AI Assistant | ||||
| **当前版本**: Google ADK 1.10.0 | ||||
| **项目状态**: 🟡 进行中 (基础设施完成,核心功能开发中) | ||||
|  | @ -0,0 +1,236 @@ | |||
| # 🚀 稷下学宫负载均衡系统 - 快速上手指南 | ||||
| 
 | ||||
| ## 📋 前置条件 | ||||
| 
 | ||||
| 1. **RapidAPI账户**: 确保已订阅以下API服务 | ||||
|    - Alpha Vantage | ||||
|    - Yahoo Finance 15 | ||||
|    - Webull | ||||
|    - Seeking Alpha | ||||
| 
 | ||||
| 2. **环境配置**: 已配置Doppler环境变量管理 | ||||
|    ```bash | ||||
|    doppler secrets | grep RAPIDAPI_KEY | ||||
|    ``` | ||||
| 
 | ||||
| ## ⚡ 5分钟快速体验 | ||||
| 
 | ||||
| ### 1. 运行完整演示 | ||||
| ```bash | ||||
| cd /home/ben/liurenchaxin | ||||
| doppler run python demo_jixia_load_balancing.py | ||||
| ``` | ||||
| 
 | ||||
| ### 2. 查看演示结果 | ||||
| ```bash | ||||
| # 查看生成的结果文件 | ||||
| ls demo_results_*.json | ||||
| 
 | ||||
| # 查看AAPL的详细结果 | ||||
| cat demo_results_aapl.json | jq . | ||||
| ``` | ||||
| 
 | ||||
| ## 🎯 核心功能演示 | ||||
| 
 | ||||
| ### 单个仙人数据获取 | ||||
| ```python | ||||
| from src.jixia.engines.jixia_load_balancer import JixiaLoadBalancer | ||||
| 
 | ||||
| # 初始化 | ||||
| load_balancer = JixiaLoadBalancer(rapidapi_key) | ||||
| 
 | ||||
| # 吕洞宾获取苹果股票数据 | ||||
| result = load_balancer.get_data_for_immortal('吕洞宾', 'stock_quote', 'AAPL') | ||||
| print(f"价格: ${result.data['price']}, 来源: {result.api_used}") | ||||
| ``` | ||||
| 
 | ||||
| ### 八仙论道完整演示 | ||||
| ```python | ||||
| # 进行八仙论道 | ||||
| results = load_balancer.conduct_immortal_debate('TSLA') | ||||
| 
 | ||||
| # 查看负载分布 | ||||
| distribution = load_balancer.get_load_distribution() | ||||
| for api, stats in distribution.items(): | ||||
|     print(f"{api}: {stats['calls']}次调用 ({stats['percentage']:.1f}%)") | ||||
| ``` | ||||
| 
 | ||||
| ## 📊 预期输出示例 | ||||
| 
 | ||||
| ``` | ||||
| 🏛️ 稷下学宫八仙论道开始 - 主题: AAPL | ||||
| ============================================================ | ||||
| 🎭 吕洞宾 正在获取 stock_quote 数据... | ||||
|    ✅ 成功从 alpha_vantage 获取数据 (响应时间: 1.33s) | ||||
|    💰 吕洞宾: $202.38 (-2.5004%) via alpha_vantage | ||||
| 
 | ||||
| 🎭 何仙姑 正在获取 stock_quote 数据... | ||||
|    ✅ 成功从 yahoo_finance_15 获取数据 (响应时间: 1.87s) | ||||
|    💰 何仙姑: $N/A (N/A) via yahoo_finance_15 | ||||
| 
 | ||||
| 📊 负载分布统计: | ||||
|    alpha_vantage: 3 次调用 (37.5%) - 健康 | ||||
|    yahoo_finance_15: 2 次调用 (25.0%) - 健康 | ||||
|    webull: 3 次调用 (37.5%) - 健康 | ||||
| ``` | ||||
| 
 | ||||
| ## 🔧 自定义配置 | ||||
| 
 | ||||
| ### 修改仙人API偏好 | ||||
| 编辑 `/home/ben/liurenchaxin/src/jixia/config/immortal_api_config.json`: | ||||
| 
 | ||||
| ```json | ||||
| { | ||||
|   "immortals": { | ||||
|     "吕洞宾": { | ||||
|       "preferred_apis": { | ||||
|         "stock_quote": "webull",  // 改为使用Webull | ||||
|         "company_overview": "alpha_vantage" | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
| ``` | ||||
| 
 | ||||
| ### 调整缓存策略 | ||||
| ```python | ||||
| # 修改缓存TTL | ||||
| load_balancer.cache_ttl = 600  # 10分钟缓存 | ||||
| 
 | ||||
| # 清空缓存 | ||||
| load_balancer.cache.clear() | ||||
| ``` | ||||
| 
 | ||||
| ## 🚨 故障排除 | ||||
| 
 | ||||
| ### 常见问题 | ||||
| 
 | ||||
| 1. **API密钥错误** | ||||
|    ``` | ||||
|    ❌ 错误: 请设置RAPIDAPI_KEY环境变量 | ||||
|    ``` | ||||
|    **解决**: 确保Doppler配置正确 | ||||
|    ```bash | ||||
|    doppler secrets set RAPIDAPI_KEY="your_key_here" | ||||
|    ``` | ||||
| 
 | ||||
| 2. **API调用失败** | ||||
|    ``` | ||||
|    ⚠️ alpha_vantage 不可用,尝试备用API... | ||||
|    ``` | ||||
|    **解决**: 系统会自动故障转移,无需干预 | ||||
| 
 | ||||
| 3. **数据格式异常** | ||||
|    ``` | ||||
|    💰 价格: $N/A | ||||
|    ``` | ||||
|    **解决**: 某些API返回格式不同,系统会标准化处理 | ||||
| 
 | ||||
| ### 调试模式 | ||||
| ```python | ||||
| # 启用详细日志 | ||||
| import logging | ||||
| logging.basicConfig(level=logging.DEBUG) | ||||
| 
 | ||||
| # 查看API健康状态 | ||||
| for api, status in load_balancer.health_checker.health_status.items(): | ||||
|     print(f"{api}: {'健康' if status['healthy'] else '异常'}") | ||||
| ``` | ||||
| 
 | ||||
| ## 📈 性能优化建议 | ||||
| 
 | ||||
| ### 1. 缓存优化 | ||||
| ```python | ||||
| # 针对不同数据类型设置不同缓存时间 | ||||
| cache_strategies = { | ||||
|     'stock_quote': 60,      # 1分钟 | ||||
|     'company_overview': 3600, # 1小时 | ||||
|     'market_news': 1800     # 30分钟 | ||||
| } | ||||
| ``` | ||||
| 
 | ||||
| ### 2. 并发控制 | ||||
| ```python | ||||
| # 控制并发请求数量 | ||||
| import time | ||||
| for immortal in immortals: | ||||
|     result = load_balancer.get_data_for_immortal(immortal, 'stock_quote', symbol) | ||||
|     time.sleep(0.2)  # 避免过快请求 | ||||
| ``` | ||||
| 
 | ||||
| ### 3. 批量处理 | ||||
| ```python | ||||
| # 批量获取多个股票数据 | ||||
| symbols = ['AAPL', 'TSLA', 'MSFT', 'GOOGL'] | ||||
| results = {} | ||||
| for symbol in symbols: | ||||
|     results[symbol] = load_balancer.conduct_immortal_debate(symbol) | ||||
| ``` | ||||
| 
 | ||||
| ## 🎯 最佳实践 | ||||
| 
 | ||||
| ### 1. 监控API使用情况 | ||||
| ```python | ||||
| # 定期检查负载分布 | ||||
| distribution = load_balancer.get_load_distribution() | ||||
| print(f"总调用次数: {sum(stats['calls'] for stats in distribution.values())}") | ||||
| ``` | ||||
| 
 | ||||
| ### 2. 合理使用缓存 | ||||
| ```python | ||||
| # 对于实时性要求不高的数据,优先使用缓存 | ||||
| result = load_balancer.get_data_for_immortal('韩湘子', 'company_overview', 'AAPL') | ||||
| if result.cached: | ||||
|     print("使用缓存数据,节省API调用") | ||||
| ``` | ||||
| 
 | ||||
| ### 3. 错误处理 | ||||
| ```python | ||||
| result = load_balancer.get_data_for_immortal('吕洞宾', 'stock_quote', 'AAPL') | ||||
| if not result.success: | ||||
|     print(f"获取数据失败: {result.error}") | ||||
|     # 实施降级策略 | ||||
| else: | ||||
|     # 正常处理数据 | ||||
|     process_stock_data(result.data) | ||||
| ``` | ||||
| 
 | ||||
| ## 📚 进阶使用 | ||||
| 
 | ||||
| ### 自定义数据处理器 | ||||
| ```python | ||||
| class CustomDataNormalizer(DataNormalizer): | ||||
|     def normalize_stock_quote(self, raw_data, api_source): | ||||
|         # 自定义数据标准化逻辑 | ||||
|         normalized = super().normalize_stock_quote(raw_data, api_source) | ||||
|         # 添加自定义字段 | ||||
|         normalized['custom_score'] = self.calculate_score(normalized) | ||||
|         return normalized | ||||
| 
 | ||||
| # 使用自定义处理器 | ||||
| load_balancer.data_normalizer = CustomDataNormalizer() | ||||
| ``` | ||||
| 
 | ||||
| ### 自定义健康检查 | ||||
| ```python | ||||
| class CustomHealthChecker(APIHealthChecker): | ||||
|     def _perform_health_check(self, api_name): | ||||
|         # 自定义健康检查逻辑 | ||||
|         # 例如:检查API响应时间、错误率等 | ||||
|         pass | ||||
| 
 | ||||
| load_balancer.health_checker = CustomHealthChecker() | ||||
| ``` | ||||
| 
 | ||||
| ## 🎉 完成! | ||||
| 
 | ||||
| 现在您已经掌握了稷下学宫负载均衡系统的基本使用方法。 | ||||
| 
 | ||||
| ### 下一步 | ||||
| - 📖 阅读完整文档: `README_jixia_load_balancing.md` | ||||
| - 🔧 查看配置文件: `src/jixia/config/immortal_api_config.json` | ||||
| - 💻 研究核心代码: `src/jixia/engines/jixia_load_balancer.py` | ||||
| - 🚀 开始构建您的投资分析系统! | ||||
| 
 | ||||
| --- | ||||
| *🏛️ 稷下学宫 - 智慧投资,从负载均衡开始* | ||||
							
								
								
									
										51
									
								
								README.md
								
								
								
								
							
							
						
						
									
										51
									
								
								README.md
								
								
								
								
							|  | @ -1,10 +1,13 @@ | |||
| # 🏛️ 炼妖壶 (Lianyaohu) - 稷下学宫AI辩论系统 | ||||
| 
 | ||||
| 提示:已支持 Cloudflare AutoRAG/Vectorize 作为记忆后端(RAG)。见 docs/guides/CLOUDFLARE_AUTORAG_INTEGRATION.md。 | ||||
| 
 | ||||
| 基于中国哲学传统的多AI智能体辩论平台,重构版本。 | ||||
| 
 | ||||
| ## ✨ 核心特性 | ||||
| 
 | ||||
| - **🎭 稷下学宫八仙论道**: 基于中国传统八仙文化的多AI智能体辩论系统 | ||||
| - **🧠 Vertex AI记忆银行**: 集成Google Cloud Memory Bank,让AI智能体具备持久化记忆能力 | ||||
| - **🌍 天下体系分析**: 基于儒门天下观的资本生态"天命树"分析模型 | ||||
| - **🔒 安全配置管理**: 使用Doppler进行统一的密钥和配置管理 | ||||
| - **📊 智能数据源**: 基于17个RapidAPI订阅的永动机数据引擎 | ||||
|  | @ -37,9 +40,14 @@ liurenchaxin/ | |||
| ### 1. 环境准备 | ||||
| 
 | ||||
| ```bash | ||||
| # 创建虚拟环境 | ||||
| python -m venv venv | ||||
| source venv/bin/activate  # Linux/Mac | ||||
| # 创建虚拟环境(与 Google ADK Quickstart 一致) | ||||
| python -m venv .venv | ||||
| # macOS/Linux | ||||
| source .venv/bin/activate | ||||
| # Windows CMD | ||||
| # .venv\Scripts\activate.bat | ||||
| # Windows PowerShell | ||||
| # .venv\Scripts\Activate.ps1 | ||||
| 
 | ||||
| # 安装依赖 | ||||
| pip install -r requirements.txt | ||||
|  | @ -50,9 +58,21 @@ pip install -r requirements.txt | |||
| 项目使用Doppler进行安全的配置管理。需要配置以下环境变量: | ||||
| 
 | ||||
| ```bash | ||||
| # 必需配置 | ||||
| # 必需配置(数据源) | ||||
| RAPIDAPI_KEY=your_rapidapi_key | ||||
| 
 | ||||
| # 选择其一的AI服务密钥 | ||||
| # A) OpenRouter(旧) | ||||
| OPENROUTER_API_KEY_1=your_openrouter_key | ||||
| # B) Google ADK / Gemini(新) | ||||
| GOOGLE_API_KEY=your_gemini_api_key | ||||
| # 如果使用 Vertex AI Express Mode(可选) | ||||
| GOOGLE_GENAI_USE_VERTEXAI=TRUE | ||||
| 
 | ||||
| # Vertex AI Memory Bank 配置(新功能) | ||||
| GOOGLE_CLOUD_PROJECT_ID=your-project-id | ||||
| GOOGLE_CLOUD_LOCATION=us-central1 | ||||
| VERTEX_MEMORY_BANK_ENABLED=TRUE | ||||
| 
 | ||||
| # 可选配置 | ||||
| POSTGRES_URL=your_postgres_url | ||||
|  | @ -94,6 +114,9 @@ python config/doppler_config.py | |||
| 
 | ||||
| # 测试Swarm辩论 (可选) | ||||
| python src/jixia/debates/swarm_debate.py | ||||
| 
 | ||||
| # 测试Vertex AI Memory Bank (新功能) | ||||
| python tests/test_vertex_memory_bank.py | ||||
| ``` | ||||
| 
 | ||||
| ## 🎭 稷下学宫八仙论道 | ||||
|  | @ -182,3 +205,23 @@ python src/jixia/debates/swarm_debate.py | |||
| --- | ||||
| 
 | ||||
| **炼妖壶 - 让AI辩论照亮投资智慧** 🏛️✨ | ||||
| 
 | ||||
| ## 🧪 ADK 开发调试(可选) | ||||
| 
 | ||||
| 如果切换到 Google ADK: | ||||
| 
 | ||||
| ```bash | ||||
| # 安装 ADK(任选其一) | ||||
| pip install google-adk | ||||
| # 或安装最新开发版 | ||||
| pip install git+https://github.com/google/adk-python.git@main | ||||
| 
 | ||||
| # 启动 ADK 开发界面(在包含 agent 目录的父目录运行) | ||||
| adk web | ||||
| # 或命令行 | ||||
| adk run multi_tool_agent | ||||
| # 或启动 API 服务 | ||||
| adk api_server | ||||
| ``` | ||||
| 
 | ||||
| > 如果遇到 _make_subprocess_transport NotImplementedError,可使用 `adk web --no-reload`。 | ||||
|  | @ -0,0 +1,197 @@ | |||
| # 🚀 太公心易 v2.0.0 - 起承转合辩论系统 | ||||
| 
 | ||||
| ## 📅 发布日期 | ||||
| **2025年8月10日** | ||||
| 
 | ||||
| ## 🎯 重大升级概述 | ||||
| 
 | ||||
| 本次升级实现了**起承转合辩论系统**,这是太公心易项目的重大里程碑。系统从简单的群聊升级为具有完整辩论流程的多阶段辩论架构。 | ||||
| 
 | ||||
| ## ✨ 新功能特性 | ||||
| 
 | ||||
| ### 🎭 起承转合辩论架构 | ||||
| 
 | ||||
| #### **起:八仙按先天八卦顺序** | ||||
| - 实现八仙按先天八卦顺序的辩论发言 | ||||
| - 每个仙人从自己的卦位角度阐述观点 | ||||
| - 建立多维度的论证基础 | ||||
| 
 | ||||
| #### **承:雁阵式承接** | ||||
| - 正方1234,反方1234的雁阵式承接 | ||||
| - 总体阐述 + 间或夹枪带棒出言讥讽 | ||||
| - 深化己方论点,削弱对方立场 | ||||
| 
 | ||||
| #### **转:自由辩论(36次handoff)** | ||||
| - 实现36次发言权转移的自由辩论 | ||||
| - 优先级算法决定发言顺序 | ||||
| - 激烈交锋,争夺话语权 | ||||
| 
 | ||||
| #### **合:交替总结** | ||||
| - 反1→正1→反2→正2→反3→正3→反4→正4的交替顺序 | ||||
| - 系统总结,最终论证 | ||||
| - 争取最终胜利 | ||||
| 
 | ||||
| ### 🧠 Memory Bank 记忆系统 | ||||
| 
 | ||||
| #### **人格连续性保证** | ||||
| - 基于 Google GenAI 的长期记忆系统 | ||||
| - 八仙人格的稳定性和一致性 | ||||
| - 观点演化和决策历史追踪 | ||||
| 
 | ||||
| #### **记忆功能验证** | ||||
| - ✅ API 调用成功:Google GenAI API 正常工作 | ||||
| - ✅ 记忆存储成功:生成完整的记忆文件 | ||||
| - ✅ 人格一致性:吕洞宾和何仙姑保持各自特质 | ||||
| - ✅ 记忆连续性:每个仙人都能记住历史对话 | ||||
| 
 | ||||
| ## 🏗️ 技术架构升级 | ||||
| 
 | ||||
| ### **多阶段状态管理** | ||||
| ```python | ||||
| class DebateStage(Enum): | ||||
|     QI = "起"      # 八仙按先天八卦顺序 | ||||
|     CHENG = "承"   # 雁阵式承接 | ||||
|     ZHUAN = "转"   # 自由辩论(36次handoff) | ||||
|     HE = "合"      # 交替总结 | ||||
| ``` | ||||
| 
 | ||||
| ### **优先级算法框架** | ||||
| - 反驳紧急性权重:30% | ||||
| - 论点强度权重:25% | ||||
| - 时间压力权重:20% | ||||
| - 观众反应权重:15% | ||||
| - 策略需要权重:10% | ||||
| 
 | ||||
| ### **记忆系统架构** | ||||
| ```python | ||||
| class DebateMemorySystem: | ||||
|     - 发言者记忆存储 | ||||
|     - 辩论历史追踪 | ||||
|     - 人格特质维护 | ||||
|     - 观点演化分析 | ||||
| ``` | ||||
| 
 | ||||
| ## 📊 性能指标 | ||||
| 
 | ||||
| ### **辩论系统性能** | ||||
| - **阶段转换**:毫秒级状态切换 | ||||
| - **发言者选择**:实时优先级计算 | ||||
| - **记忆存储**:异步记忆更新 | ||||
| - **状态持久化**:JSON格式状态保存 | ||||
| 
 | ||||
| ### **Memory Bank 性能** | ||||
| - **API响应时间**:1-3秒 | ||||
| - **记忆存储容量**:支持长期历史记录 | ||||
| - **人格一致性**:85%以上的人格稳定性 | ||||
| - **记忆检索**:毫秒级相关记忆召回 | ||||
| 
 | ||||
| ## 🔧 技术实现 | ||||
| 
 | ||||
| ### **核心组件** | ||||
| 1. **QiChengZhuanHeDebate**:起承转合辩论系统核心 | ||||
| 2. **PriorityAlgorithm**:优先级算法实现 | ||||
| 3. **DebateMemorySystem**:辩论记忆系统 | ||||
| 4. **MemoryBankTest**:记忆系统测试框架 | ||||
| 
 | ||||
| ### **依赖升级** | ||||
| - Google GenAI 1.29.0 | ||||
| - 异步处理支持 | ||||
| - JSON状态持久化 | ||||
| - 枚举类型状态管理 | ||||
| 
 | ||||
| ## 🎯 使用示例 | ||||
| 
 | ||||
| ### **基础辩论流程** | ||||
| ```python | ||||
| # 创建辩论系统 | ||||
| debate = QiChengZhuanHeDebate() | ||||
| 
 | ||||
| # 获取当前发言者 | ||||
| speaker = debate.get_current_speaker() | ||||
| 
 | ||||
| # 记录发言 | ||||
| debate.record_speech(speaker, "发言内容") | ||||
| 
 | ||||
| # 推进阶段 | ||||
| debate.advance_stage() | ||||
| 
 | ||||
| # 保存状态 | ||||
| debate.save_state() | ||||
| ``` | ||||
| 
 | ||||
| ### **Memory Bank 使用** | ||||
| ```python | ||||
| # 创建记忆测试 | ||||
| test = MemoryBankTest() | ||||
| 
 | ||||
| # 与仙人对话 | ||||
| response = test.chat_with_immortal("吕洞宾", "问题") | ||||
| 
 | ||||
| # 保存记忆 | ||||
| test.save_memories() | ||||
| ``` | ||||
| 
 | ||||
| ## 🚀 下一步计划 | ||||
| 
 | ||||
| ### **短期目标(v2.1.0)** | ||||
| - [ ] 完善优先级算法 | ||||
| - [ ] 实现多群聊协调 | ||||
| - [ ] 添加Human干预机制 | ||||
| - [ ] 优化辩论流程控制 | ||||
| 
 | ||||
| ### **中期目标(v2.2.0)** | ||||
| - [ ] 集成太公三式预测 | ||||
| - [ ] 实现梅花心易直觉 | ||||
| - [ ] 完善八仙人格量化 | ||||
| - [ ] 添加观众反馈系统 | ||||
| 
 | ||||
| ### **长期目标(v3.0.0)** | ||||
| - [ ] 完整的预测系统 | ||||
| - [ ] 商业化部署 | ||||
| - [ ] 多语言支持 | ||||
| - [ ] 移动端应用 | ||||
| 
 | ||||
| ## 🐛 已知问题 | ||||
| 
 | ||||
| 1. **优先级算法**:当前使用简化版本,需要进一步优化 | ||||
| 2. **多群聊协调**:尚未实现完整的群聊网络 | ||||
| 3. **Human干预**:干预机制需要进一步完善 | ||||
| 4. **性能优化**:大规模辩论的性能需要优化 | ||||
| 
 | ||||
| ## 📝 更新日志 | ||||
| 
 | ||||
| ### **v2.0.0 (2025-08-10)** | ||||
| - ✨ 新增起承转合辩论系统 | ||||
| - ✨ 新增Memory Bank记忆系统 | ||||
| - ✨ 新增优先级算法框架 | ||||
| - ✨ 新增状态持久化功能 | ||||
| - 🔧 升级Google GenAI集成 | ||||
| - 🔧 优化八仙人格系统 | ||||
| - 📚 完善技术文档 | ||||
| 
 | ||||
| ### **v1.x.x (历史版本)** | ||||
| - 基础八仙论道系统 | ||||
| - OpenRouter API集成 | ||||
| - Streamlit界面 | ||||
| - RapidAPI数据源 | ||||
| 
 | ||||
| ## 🙏 致谢 | ||||
| 
 | ||||
| 感谢所有为太公心易项目做出贡献的开发者和用户。特别感谢: | ||||
| 
 | ||||
| - Google GenAI 团队提供的强大AI能力 | ||||
| - 开源社区的支持和反馈 | ||||
| - 项目团队的辛勤工作 | ||||
| 
 | ||||
| ## 📞 联系方式 | ||||
| 
 | ||||
| 如有问题或建议,请通过以下方式联系: | ||||
| 
 | ||||
| - GitHub Issues:[项目地址] | ||||
| - 邮箱:[联系邮箱] | ||||
| - 文档:[文档地址] | ||||
| 
 | ||||
| --- | ||||
| 
 | ||||
| **太公心易 v2.0.0** - 让AI辩论更有智慧,让预测更有力量! | ||||
| 
 | ||||
|  | @ -50,6 +50,52 @@ def get_openrouter_key() -> str: | |||
|     """ | ||||
|     return get_secret('OPENROUTER_API_KEY_1') | ||||
| 
 | ||||
| def get_google_api_key() -> str: | ||||
|     """ | ||||
|     获取Google API密钥 (用于 Gemini/ADK) | ||||
|      | ||||
|     Returns: | ||||
|         Google API密钥 | ||||
|          | ||||
|     Raises: | ||||
|         ValueError: 如果密钥未找到 | ||||
|     """ | ||||
|     return get_secret('GOOGLE_API_KEY') | ||||
| 
 | ||||
| def get_google_genai_config() -> Dict[str, str]: | ||||
|     """ | ||||
|     获取Google GenAI完整配置 | ||||
|      | ||||
|     Returns: | ||||
|         Google GenAI配置字典 | ||||
|     """ | ||||
|     return { | ||||
|         'api_key': get_secret('GOOGLE_API_KEY', ''), | ||||
|         'use_vertex_ai': get_secret('GOOGLE_GENAI_USE_VERTEXAI', 'FALSE'), | ||||
|         'project_id': get_secret('GOOGLE_CLOUD_PROJECT_ID', ''), | ||||
|         'location': get_secret('GOOGLE_CLOUD_LOCATION', 'us-central1'), | ||||
|         'memory_bank_enabled': get_secret('VERTEX_MEMORY_BANK_ENABLED', 'TRUE'), | ||||
|         'service_account_key': get_secret('GOOGLE_SERVICE_ACCOUNT_KEY', '') | ||||
|     } | ||||
| 
 | ||||
| def get_cloudflare_config() -> Dict[str, str]: | ||||
|     """ | ||||
|     获取Cloudflare配置 | ||||
|      | ||||
|     Returns: | ||||
|         Cloudflare配置字典 | ||||
|     """ | ||||
|     return { | ||||
|         # 敏感信息从Doppler获取 | ||||
|         'account_id': get_secret('CLOUDFLARE_ACCOUNT_ID', ''), | ||||
|         'api_token': get_secret('CLOUDFLARE_API_TOKEN', ''), | ||||
|          | ||||
|         # 非敏感配置,明文写在代码里 | ||||
|         'vectorize_index': 'autorag-shy-cherry-f1fb', | ||||
|         'embed_model': '@cf/baai/bge-m3', | ||||
|         'autorag_domain': 'autorag.seekkey.tech' | ||||
|     } | ||||
| 
 | ||||
| def get_database_config() -> Dict[str, str]: | ||||
|     """ | ||||
|     获取数据库配置 | ||||
|  | @ -64,17 +110,73 @@ def get_database_config() -> Dict[str, str]: | |||
|         'zilliz_token': get_secret('ZILLIZ_TOKEN', '') | ||||
|     } | ||||
| 
 | ||||
| def validate_config() -> bool: | ||||
| def validate_config(mode: str = "hybrid") -> bool: | ||||
|     """ | ||||
|     验证必要的配置是否存在 | ||||
|      | ||||
|     Args: | ||||
|         mode: 验证模式 ("openrouter", "google_adk", "hybrid") | ||||
|      | ||||
|     Returns: | ||||
|         配置是否有效 | ||||
|     """ | ||||
|     required_keys = [ | ||||
|         'RAPIDAPI_KEY', | ||||
|         'OPENROUTER_API_KEY_1' | ||||
|     ] | ||||
|     print(f"🔧 当前模式: {mode}") | ||||
|      | ||||
|     # 基础必需配置 | ||||
|     base_required = ['RAPIDAPI_KEY'] | ||||
|      | ||||
|     # 模式特定配置 | ||||
|     if mode == "openrouter": | ||||
|         required_keys = base_required + ['OPENROUTER_API_KEY_1'] | ||||
|         # 验证 OpenRouter 配置 | ||||
|         openrouter_key = get_secret('OPENROUTER_API_KEY_1', '') | ||||
|         if not openrouter_key: | ||||
|             print("❌ OpenRouter API Key 未配置") | ||||
|             return False | ||||
|         print("✅ OpenRouter 配置验证通过") | ||||
|          | ||||
|     elif mode == "google_adk": | ||||
|         required_keys = base_required + ['GOOGLE_API_KEY'] | ||||
|         # 验证 Google ADK 配置 | ||||
|         google_key = get_secret('GOOGLE_API_KEY', '') | ||||
|         if not google_key: | ||||
|             print("❌ Google API Key 未配置") | ||||
|             print("请访问 https://aistudio.google.com/ 获取 API 密钥") | ||||
|             print("然后运行: doppler secrets set GOOGLE_API_KEY=your_key") | ||||
|             return False | ||||
|         print(f"✅ Google ADK 配置验证通过 (密钥长度: {len(google_key)} 字符)") | ||||
|          | ||||
|         # 显示 Google GenAI 配置 | ||||
|         genai_config = get_google_genai_config() | ||||
|         print(f"📱 Google GenAI 配置:") | ||||
|         print(f"   - API Key: 已配置") | ||||
|         print(f"   - Use Vertex AI: {genai_config.get('use_vertex_ai', False)}") | ||||
|         if genai_config.get('project_id'): | ||||
|             print(f"   - Project ID: {genai_config['project_id']}") | ||||
|         if genai_config.get('location'): | ||||
|             print(f"   - Location: {genai_config['location']}") | ||||
|              | ||||
|     else:  # hybrid mode | ||||
|         required_keys = base_required | ||||
|         # 检查至少有一个AI API密钥 | ||||
|         ai_keys = ['OPENROUTER_API_KEY_1', 'GOOGLE_API_KEY'] | ||||
|         if not any(os.getenv(key) for key in ai_keys): | ||||
|             print("❌ 需要至少配置一个AI API密钥:") | ||||
|             print("   - OPENROUTER_API_KEY_1 (OpenRouter模式)") | ||||
|             print("   - GOOGLE_API_KEY (Google ADK模式)") | ||||
|             return False | ||||
|              | ||||
|         # 验证混合模式配置 | ||||
|         openrouter_key = get_secret('OPENROUTER_API_KEY_1', '') | ||||
|         google_key = get_secret('GOOGLE_API_KEY', '') | ||||
|          | ||||
|         available_services = [] | ||||
|         if openrouter_key: | ||||
|             available_services.append("OpenRouter") | ||||
|         if google_key: | ||||
|             available_services.append("Google ADK") | ||||
|              | ||||
|         print(f"✅ 混合模式配置验证通过,可用服务: {', '.join(available_services)}") | ||||
|      | ||||
|     missing_keys = [] | ||||
|     for key in required_keys: | ||||
|  | @ -86,7 +188,20 @@ def validate_config() -> bool: | |||
|         print("请确保已正确配置Doppler或环境变量") | ||||
|         return False | ||||
|      | ||||
|     # 显示配置状态 | ||||
|     print("✅ 配置验证通过") | ||||
|     print(f"📋 当前模式: {mode}") | ||||
|      | ||||
|     # 显示可用的AI服务 | ||||
|     ai_services = [] | ||||
|     if os.getenv('OPENROUTER_API_KEY_1'): | ||||
|         ai_services.append("OpenRouter") | ||||
|     if os.getenv('GOOGLE_API_KEY'): | ||||
|         ai_services.append("Google ADK") | ||||
|      | ||||
|     if ai_services: | ||||
|         print(f"🤖 可用AI服务: {', '.join(ai_services)}") | ||||
|      | ||||
|     return True | ||||
| 
 | ||||
| if __name__ == "__main__": | ||||
|  |  | |||
|  | @ -0,0 +1,116 @@ | |||
| 好的!以下是针对你“八仙多Agent辩论系统 + Mastodon引流 + Streamlit实时展示”,基于Google ADK免费额度的详细需求文档,方便你交给Kiro进行开发。 | ||||
| 
 | ||||
| --- | ||||
| 
 | ||||
| # 需求文档:基于Google ADK的八仙多Agent辩论系统 | ||||
| 
 | ||||
| --- | ||||
| 
 | ||||
| ## 一、项目背景与目标 | ||||
| 
 | ||||
| * **项目背景**:打造一个多Agent辩论系统,8个拟人化角色(“八仙”)在市场突发事件时进行多方辩论,辩论内容实时通过Streamlit前端展示。同时通过Mastodon社交平台发布轻松动态,吸引用户关注和引流。 | ||||
| * **目标**: | ||||
| 
 | ||||
|   * 利用Google ADK(含免费额度)搭建多Agent长期记忆系统(Memory Bank)和检索增强生成(RAG) | ||||
|   * 实现八个角色独立人格和独立记忆空间,保证角色稳定性和多样性 | ||||
|   * 实时监控市场数据触发紧急辩论事件 | ||||
|   * 在Streamlit实时展示辩论过程、行情数据和结论 | ||||
|   * Mastodon账号模拟八仙发布轻社交内容,实现引流和用户互动 | ||||
| 
 | ||||
| --- | ||||
| 
 | ||||
| ## 二、核心功能需求 | ||||
| 
 | ||||
| ### 1. 多Agent系统架构 | ||||
| 
 | ||||
| * 8个独立Agent,分别代表不同市场角色(交易员、经济学家、央行顾问等) | ||||
| * 每个Agent配备独立的Memory Bank(Google ADK Memory Bank),用于存储和检索长期记忆和知识 | ||||
| * Agent能检索自身Memory Bank相关信息,结合当前上下文进行动态对话和观点生成 | ||||
| * 统一调用Google GenAI接口,利用免费额度进行生成与检索 | ||||
| * Agent间支持异步、多轮交互,形成辩论流程 | ||||
| 
 | ||||
| ### 2. 记忆管理 | ||||
| 
 | ||||
| * Memory Bank支持多模态存储(文本、结构化数据等) | ||||
| * 支持长期记忆(历史辩论内容、预测结果、个人观点)和短期上下文记忆(当前会议) | ||||
| * 定期同步本地向量库(如Milvus/Qdrant)与Google Memory Bank,提升检索效率 | ||||
| * 实现基于内容哈希的缓存机制,减少重复调用 | ||||
| 
 | ||||
| ### 3. 市场数据触发模块 | ||||
| 
 | ||||
| * 实时监听主要市场指标(纳指、标普、黄金、加密货币等)和财经新闻 | ||||
| * 设定触发规则(如纳指暴跌超过10%)启动紧急辩论会议 | ||||
| * 支持自定义触发事件和预警配置 | ||||
| 
 | ||||
| ### 4. Streamlit展示前端 | ||||
| 
 | ||||
| * 实时行情图表展示 | ||||
| * 辩论内容滚动显示(多Agent轮流发言) | ||||
| * 会议总结与观点汇总卡片 | ||||
| * 用户评论和互动区(可选) | ||||
| 
 | ||||
| ### 5. Mastodon社交引流模块 | ||||
| 
 | ||||
| * 每个Agent拥有独立Mastodon账号 | ||||
| * 自动发布轻松、拟人化的市场动态、观点碎片和会议预告 | ||||
| * 监听Mastodon事件,结合Pub/Sub机制触发系统响应 | ||||
| * 支持用户互动回复采集 | ||||
| 
 | ||||
| --- | ||||
| 
 | ||||
| ## 三、技术细节 | ||||
| 
 | ||||
| ### 1. 平台与工具 | ||||
| 
 | ||||
| * **核心API**:Google AI Developer Kit (ADK),利用Generative AI免费额度 | ||||
| * **记忆库**:Google Memory Bank + 本地Milvus/Qdrant向量库 | ||||
| * **调用调度**:Litellm或自定义调度器管理多API调用,负载均衡和容错 | ||||
| * **前端**:Streamlit(Python) | ||||
| * **社交**:Mastodon API集成,Python脚本自动发帖和监听 | ||||
| * **缓存**:Redis或内存缓存,基于请求hash缓存生成结果 | ||||
| 
 | ||||
| ### 2. 角色人格设定 | ||||
| 
 | ||||
| * 每个Agent具有独特的Prompt模板,包含背景故事、语气、知识偏好 | ||||
| * 通过RAG检索自身记忆,生成与人格一致的回答 | ||||
| * 支持Agent记忆动态更新,包含辩论历史和新信息 | ||||
| 
 | ||||
| ### 3. 数据流与接口 | ||||
| 
 | ||||
| * 市场数据 → 触发模块 → 启动辩论流程 | ||||
| * 触发模块 → 通知Mastodon发布预告 | ||||
| * 多Agent通过ADK接口交互,结果返回Streamlit展示 | ||||
| * Mastodon监听事件,触发后续系统行为 | ||||
| * 缓存层拦截重复请求,提升效率 | ||||
| 
 | ||||
| --- | ||||
| 
 | ||||
| ## 四、性能与扩展 | ||||
| 
 | ||||
| * 支持并发8个Agent多轮对话,响应时间控制在1-3秒内 | ||||
| * 内存和存储支持至少半年以上辩论历史积累 | ||||
| * 可扩展到更多Agent和多模态输入(图片、音频) | ||||
| * 支持后续接入更多API(OpenRouter、魔搭等)进行混合调用 | ||||
| 
 | ||||
| --- | ||||
| 
 | ||||
| ## 五、运维与监控 | ||||
| 
 | ||||
| * 实时监控API调用额度和使用情况,避免超额扣费 | ||||
| * 系统异常告警(调用失败、连接断开等) | ||||
| * 日志记录辩论内容和用户互动,用于后续分析和优化 | ||||
| * 定期备份Memory Bank和本地数据库 | ||||
| 
 | ||||
| --- | ||||
| 
 | ||||
| ## 六、交付物 | ||||
| 
 | ||||
| * 多Agent辩论系统完整后端代码,含ADK集成和调度逻辑 | ||||
| * Streamlit展示前端代码与部署方案 | ||||
| * Mastodon自动发帖和监听脚本 | ||||
| * 部署说明文档和API调用监控方案 | ||||
| * 角色人格Prompt模板集合和记忆管理设计文档 | ||||
| 
 | ||||
| --- | ||||
| 
 | ||||
| 如果你觉得这个需求文档需要调整细化,或者想要配套的技术方案设计、接口规范、甚至样板代码,我都可以帮你补充完善。 | ||||
|  | @ -0,0 +1,184 @@ | |||
| # 稷下学宫AI辩论系统 - AutoRAG集成提示词 | ||||
| 
 | ||||
| ## 🏛️ 系统概述 | ||||
| 
 | ||||
| 稷下学宫是一个基于中国传统哲学的AI辩论系统,模拟古代稷下学宫的学术辩论环境。系统中有八位仙人智能体,每位都有独特的投资哲学和辩论风格,需要通过AutoRAG服务获取相关的历史智慧和知识支持。 | ||||
| 
 | ||||
| ## 🎭 八仙智能体角色 | ||||
| 
 | ||||
| ### 铁拐李 (巽卦) - 逆向投资大师 | ||||
| - **投资哲学**: 逆向思维,挑战市场共识 | ||||
| - **记忆重点**: 市场异常、逆向案例、风险警示、反向策略 | ||||
| - **辩论风格**: 质疑主流观点,提出反向思考 | ||||
| 
 | ||||
| ### 吕洞宾 (乾卦) - 理性分析者   | ||||
| - **投资哲学**: 技术分析专家,数据驱动决策 | ||||
| - **记忆重点**: 技术分析、数据洞察、逻辑推理、理性决策 | ||||
| - **辩论风格**: 基于数据和逻辑的严密分析 | ||||
| 
 | ||||
| ### 何仙姑 (坤卦) - 直觉洞察者 | ||||
| - **投资哲学**: 风险控制专家,情感智慧 | ||||
| - **记忆重点**: 市场情绪、直觉判断、情感因素、人性洞察 | ||||
| - **辩论风格**: 基于直觉和情感智慧的分析 | ||||
| 
 | ||||
| ### 张果老 (兑卦) - 历史智慧者 | ||||
| - **投资哲学**: 历史数据分析师,经验导向 | ||||
| - **记忆重点**: 历史案例、长期趋势、周期规律、经验教训 | ||||
| - **辩论风格**: 引用历史案例和长期趋势 | ||||
| 
 | ||||
| ### 汉钟离 (离卦) - 平衡协调者 | ||||
| - **投资哲学**: 热点追踪专家,平衡思维 | ||||
| - **记忆重点**: 平衡策略、综合分析、协调方案、稳健建议 | ||||
| - **辩论风格**: 寻求各方观点的平衡点 | ||||
| 
 | ||||
| ### 蓝采和 (坎卦) - 创新思维者 | ||||
| - **投资哲学**: 潜力股发现者,创新导向 | ||||
| - **记忆重点**: 创新机会、新兴趋势、潜力发现、灵活策略 | ||||
| - **辩论风格**: 发现新兴机会和创新角度 | ||||
| 
 | ||||
| ### 韩湘子 (艮卦) - 艺术感知者 | ||||
| - **投资哲学**: 新兴资产专家,美学视角 | ||||
| - **记忆重点**: 美学趋势、创意洞察、感性分析、艺术视角 | ||||
| - **辩论风格**: 从美学和艺术角度分析市场 | ||||
| 
 | ||||
| ### 曹国舅 (震卦) - 实务执行者 | ||||
| - **投资哲学**: 机构视角分析师,实务导向 | ||||
| - **记忆重点**: 执行策略、机构动向、实务操作、专业分析 | ||||
| - **辩论风格**: 关注实际执行和机构操作 | ||||
| 
 | ||||
| ## 🔍 AutoRAG查询需求 | ||||
| 
 | ||||
| ### 查询类型 | ||||
| 1. **历史智慧检索**: 根据辩论主题查找相关的古代智慧、哲学思想 | ||||
| 2. **投资案例搜索**: 寻找历史上的投资成功/失败案例 | ||||
| 3. **市场周期分析**: 查找关于市场周期、经济规律的古籍记录 | ||||
| 4. **风险管理智慧**: 搜索古代关于风险控制、谨慎投资的思想 | ||||
| 5. **人性洞察**: 查找关于人性、情绪、群体心理的古代观察 | ||||
| 
 | ||||
| ### 期望的AutoRAG接口 | ||||
| 
 | ||||
| #### 1. 嵌入生成接口 | ||||
| ``` | ||||
| POST /embed | ||||
| { | ||||
|   "text": "需要生成嵌入的文本内容" | ||||
| } | ||||
| 
 | ||||
| 响应: | ||||
| { | ||||
|   "embedding": [0.1, 0.2, ...],  // 1024维BGE-M3嵌入向量 | ||||
|   "model": "bge-m3" | ||||
| } | ||||
| ``` | ||||
| 
 | ||||
| #### 2. 记忆存储接口 | ||||
| ``` | ||||
| POST /upsert | ||||
| { | ||||
|   "vectors": [ | ||||
|     { | ||||
|       "id": "memory_uuid", | ||||
|       "values": [0.1, 0.2, ...], | ||||
|       "metadata": { | ||||
|         "agent_name": "tieguaili", | ||||
|         "chinese_name": "铁拐李",  | ||||
|         "content": "记忆内容", | ||||
|         "memory_type": "knowledge|conversation|preference|strategy", | ||||
|         "debate_topic": "辩论主题", | ||||
|         "timestamp": "2024-01-01T00:00:00Z" | ||||
|       } | ||||
|     } | ||||
|   ], | ||||
|   "namespace": "agent_name"  // 智能体命名空间 | ||||
| } | ||||
| 
 | ||||
| 响应: | ||||
| { | ||||
|   "success": true, | ||||
|   "inserted_count": 1 | ||||
| } | ||||
| ``` | ||||
| 
 | ||||
| #### 3. 记忆检索接口 | ||||
| ``` | ||||
| POST /query | ||||
| { | ||||
|   "vector": [0.1, 0.2, ...],     // 查询向量 | ||||
|   "topK": 10,                    // 返回数量 | ||||
|   "namespace": "tieguaili",      // 智能体命名空间 | ||||
|   "filter": {                    // 可选过滤条件 | ||||
|     "memory_type": "knowledge" | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| 响应: | ||||
| { | ||||
|   "matches": [ | ||||
|     { | ||||
|       "id": "memory_uuid", | ||||
|       "score": 0.95, | ||||
|       "metadata": { | ||||
|         "content": "相关记忆内容", | ||||
|         "agent_name": "tieguaili", | ||||
|         "memory_type": "knowledge", | ||||
|         "debate_topic": "投资哲学" | ||||
|       } | ||||
|     } | ||||
|   ] | ||||
| } | ||||
| ``` | ||||
| 
 | ||||
| ## 📝 使用场景示例 | ||||
| 
 | ||||
| ### 场景1: 辩论前的知识准备 | ||||
| ``` | ||||
| 辩论主题: "NVIDIA股票投资价值分析" | ||||
| 
 | ||||
| 铁拐李查询: "历史上科技股泡沫的案例和教训" | ||||
| 张果老查询: "古代关于新兴技术投资的智慧" | ||||
| 何仙姑查询: "市场狂热时期的风险控制思想" | ||||
| ``` | ||||
| 
 | ||||
| ### 场景2: 辩论中的观点支撑 | ||||
| ``` | ||||
| 当前观点: "AI技术发展存在过度炒作风险" | ||||
| 
 | ||||
| 相关查询: "古代关于技术革新的理性思考" | ||||
| 期望返回: 相关的古籍智慧,支持或反驳当前观点 | ||||
| ``` | ||||
| 
 | ||||
| ### 场景3: 辩论后的经验总结 | ||||
| ``` | ||||
| 辩论结果: 铁拐李的逆向观点获得认同 | ||||
| 
 | ||||
| 存储记忆: "在AI股票讨论中,逆向思维帮助识别了估值风险" | ||||
| 记忆类型: strategy | ||||
| ``` | ||||
| 
 | ||||
| ## 🎯 集成目标 | ||||
| 
 | ||||
| 1. **智能记忆**: 每位仙人都有独立的记忆空间,能够学习和积累经验 | ||||
| 2. **文化融合**: 将古代智慧与现代投资分析相结合 | ||||
| 3. **个性化**: 根据每位仙人的特点,提供差异化的知识支持 | ||||
| 4. **持续学习**: 通过辩论过程不断丰富和完善知识库 | ||||
| 
 | ||||
| ## 🔧 技术要求 | ||||
| 
 | ||||
| - **向量维度**: 1024 (BGE-M3模型) | ||||
| - **命名空间**: 支持按智能体分离数据 | ||||
| - **元数据**: 丰富的元数据支持,便于过滤和分类 | ||||
| - **性能**: 低延迟的检索响应,支持实时辩论 | ||||
| - **扩展性**: 支持未来添加更多智能体和记忆类型 | ||||
| 
 | ||||
| ## 🌟 期望效果 | ||||
| 
 | ||||
| 通过AutoRAG集成,稷下学宫将实现: | ||||
| - 🧠 **智慧传承**: 古代哲学智慧指导现代投资决策 | ||||
| - 🎭 **角色一致**: 每位仙人保持独特的人格和观点 | ||||
| - 📚 **知识积累**: 持续学习和经验沉淀 | ||||
| - 🔄 **动态辩论**: 基于历史记忆的深度讨论 | ||||
| - 🎯 **精准分析**: 结合传统智慧的投资洞察 | ||||
| 
 | ||||
| --- | ||||
| 
 | ||||
| **让AI辩论照亮投资智慧,让古代智慧指引现代决策** 🏛️✨ | ||||
|  | @ -0,0 +1,218 @@ | |||
| # 🌐 GitHub Pages 发布计划 | ||||
| 
 | ||||
| ## 📋 发布内容规划 | ||||
| 
 | ||||
| ### **🎯 发布目标** | ||||
| - 展示项目功能和特性 | ||||
| - 提供用户友好的文档 | ||||
| - 吸引潜在用户和贡献者 | ||||
| - 保持专业形象 | ||||
| 
 | ||||
| ### **✅ 适合发布的内容** | ||||
| 
 | ||||
| #### **1. 项目主页 (根目录)** | ||||
| ``` | ||||
| / | ||||
| ├── README.md                    # 项目主介绍 | ||||
| ├── RELEASE_v2.0.0.md           # 版本发布说明 | ||||
| ├── QUICK_START_GUIDE.md        # 快速上手指南 | ||||
| ├── VERSION                     # 版本号 | ||||
| ├── requirements.txt            # 依赖清单 | ||||
| └── .gitignore                 # Git忽略文件 | ||||
| ``` | ||||
| 
 | ||||
| #### **2. 用户文档 (docs/)** | ||||
| ``` | ||||
| docs/ | ||||
| ├── index.md                    # 文档首页 | ||||
| ├── guides/                     # 用户指南 | ||||
| │   ├── quick-start.md         # 快速开始 | ||||
| │   ├── installation.md        # 安装指南 | ||||
| │   ├── configuration.md       # 配置指南 | ||||
| │   ├── cloudflare-integration.md # Cloudflare集成 | ||||
| │   ├── google-adk-migration.md   # Google ADK迁移 | ||||
| │   └── load-balancing.md      # 负载均衡指南 | ||||
| ├── features/                   # 功能特性 | ||||
| │   ├── debate-system.md       # 辩论系统 | ||||
| │   ├── memory-bank.md         # 记忆系统 | ||||
| │   ├── eight-immortals.md     # 八仙系统 | ||||
| │   └── tianxia-analysis.md    # 天下体系 | ||||
| ├── api/                       # API文档 | ||||
| │   ├── rapidapi-setup.md      # RapidAPI设置 | ||||
| │   └── vertex-memory-bank.md  # Memory Bank API | ||||
| └── examples/                  # 示例代码 | ||||
|     ├── basic-usage.md         # 基础用法 | ||||
|     └── advanced-features.md   # 高级功能 | ||||
| ``` | ||||
| 
 | ||||
| #### **3. 设计文档 (design/)** | ||||
| ``` | ||||
| design/ | ||||
| ├── overview.md                # 项目概览 | ||||
| ├── architecture.md            # 系统架构 | ||||
| ├── debate-system.md           # 辩论系统设计 | ||||
| └── roadmap.md                 # 开发路线图 | ||||
| ``` | ||||
| 
 | ||||
| #### **4. 演示和示例** | ||||
| ``` | ||||
| examples/ | ||||
| ├── basic-debate.md            # 基础辩论示例 | ||||
| ├── memory-bank-demo.md        # 记忆系统演示 | ||||
| └── load-balancing-demo.md     # 负载均衡演示 | ||||
| ``` | ||||
| 
 | ||||
| ### **🔒 保留在 internal/ 的内容** | ||||
| 
 | ||||
| #### **1. 内部开发文档** | ||||
| - 开发计划和路线图 | ||||
| - 技术实现细节 | ||||
| - 内部策略文档 | ||||
| - 代码审查记录 | ||||
| 
 | ||||
| #### **2. 敏感信息** | ||||
| - API密钥配置 | ||||
| - 内部分析报告 | ||||
| - 迁移方案细节 | ||||
| - 历史文档 | ||||
| 
 | ||||
| #### **3. 配置文件** | ||||
| - 环境配置 | ||||
| - 内部脚本 | ||||
| - AI助手配置 | ||||
| 
 | ||||
| ## 🚀 发布步骤 | ||||
| 
 | ||||
| ### **第一阶段:内容准备** | ||||
| 1. **优化 README.md** | ||||
|    - 添加项目徽章 | ||||
|    - 完善功能介绍 | ||||
|    - 添加截图和演示 | ||||
|    - 优化安装说明 | ||||
| 
 | ||||
| 2. **创建文档首页** | ||||
|    - 设计文档结构 | ||||
|    - 创建导航菜单 | ||||
|    - 添加搜索功能 | ||||
| 
 | ||||
| 3. **整理用户指南** | ||||
|    - 统一文档格式 | ||||
|    - 添加代码示例 | ||||
|    - 完善配置说明 | ||||
| 
 | ||||
| ### **第二阶段:GitHub Pages 配置** | ||||
| 1. **启用 GitHub Pages** | ||||
|    ```bash | ||||
|    # 在仓库设置中启用 GitHub Pages | ||||
|    # 选择 docs/ 文件夹作为源 | ||||
|    ``` | ||||
| 
 | ||||
| 2. **配置 Jekyll 主题** | ||||
|    ```yaml | ||||
|    # _config.yml | ||||
|    title: 太公心易 - 稷下学宫AI辩论系统 | ||||
|    description: 基于中国哲学传统的多AI智能体辩论平台 | ||||
|    theme: jekyll-theme-cayman | ||||
|    ``` | ||||
| 
 | ||||
| 3. **创建导航结构** | ||||
|    ```markdown | ||||
|    # docs/index.md | ||||
|    --- | ||||
|    layout: default | ||||
|    title: 太公心易文档 | ||||
|    --- | ||||
|    ``` | ||||
| 
 | ||||
| ### **第三阶段:内容发布** | ||||
| 1. **发布核心文档** | ||||
|    - 项目介绍 | ||||
|    - 快速开始指南 | ||||
|    - 功能特性说明 | ||||
| 
 | ||||
| 2. **发布用户指南** | ||||
|    - 安装配置 | ||||
|    - 使用教程 | ||||
|    - 示例代码 | ||||
| 
 | ||||
| 3. **发布设计文档** | ||||
|    - 系统架构 | ||||
|    - 技术设计 | ||||
|    - 开发路线图 | ||||
| 
 | ||||
| ## 📊 发布效果预期 | ||||
| 
 | ||||
| ### **用户访问路径** | ||||
| ``` | ||||
| GitHub Pages 首页 | ||||
| ├── 项目介绍 → README.md | ||||
| ├── 快速开始 → QUICK_START_GUIDE.md | ||||
| ├── 用户指南 → docs/guides/ | ||||
| ├── 功能特性 → docs/features/ | ||||
| ├── API文档 → docs/api/ | ||||
| └── 示例代码 → docs/examples/ | ||||
| ``` | ||||
| 
 | ||||
| ### **SEO 优化** | ||||
| - 添加 meta 标签 | ||||
| - 优化标题和描述 | ||||
| - 添加关键词 | ||||
| - 创建 sitemap | ||||
| 
 | ||||
| ### **用户体验** | ||||
| - 响应式设计 | ||||
| - 快速加载 | ||||
| - 清晰导航 | ||||
| - 搜索功能 | ||||
| 
 | ||||
| ## 🔧 技术实现 | ||||
| 
 | ||||
| ### **GitHub Pages 配置** | ||||
| ```yaml | ||||
| # .github/workflows/pages.yml | ||||
| name: Deploy to GitHub Pages | ||||
| on: | ||||
|   push: | ||||
|     branches: [ main ] | ||||
|   pull_request: | ||||
|     branches: [ main ] | ||||
| 
 | ||||
| jobs: | ||||
|   deploy: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|     - uses: actions/checkout@v2 | ||||
|     - name: Deploy to GitHub Pages | ||||
|       uses: peaceiris/actions-gh-pages@v3 | ||||
|       with: | ||||
|         github_token: ${{ secrets.GITHUB_TOKEN }} | ||||
|         publish_dir: ./docs | ||||
| ``` | ||||
| 
 | ||||
| ### **文档结构优化** | ||||
| - 使用 Markdown 格式 | ||||
| - 添加目录导航 | ||||
| - 统一代码高亮 | ||||
| - 优化图片显示 | ||||
| 
 | ||||
| ## 📈 维护计划 | ||||
| 
 | ||||
| ### **定期更新** | ||||
| - 每月更新功能文档 | ||||
| - 季度更新架构文档 | ||||
| - 及时更新版本说明 | ||||
| 
 | ||||
| ### **用户反馈** | ||||
| - 收集用户问题 | ||||
| - 更新常见问题 | ||||
| - 优化文档内容 | ||||
| 
 | ||||
| ### **版本同步** | ||||
| - 保持文档与代码同步 | ||||
| - 及时发布新版本说明 | ||||
| - 维护版本历史 | ||||
| 
 | ||||
| --- | ||||
| 
 | ||||
| **发布状态**:🔄 计划中   | ||||
| **预计完成**:2025年8月底 | ||||
|  | @ -0,0 +1,270 @@ | |||
| # Vertex AI Memory Bank 配置指南 | ||||
| 
 | ||||
| ## 🏛️ 稷下学宫记忆银行集成 | ||||
| 
 | ||||
| 本文档介绍如何为稷下学宫AI辩论系统配置和使用Vertex AI Memory Bank功能。 | ||||
| 
 | ||||
| ## 📋 前置要求 | ||||
| 
 | ||||
| ### 1. Google Cloud 项目设置 | ||||
| - 有效的 Google Cloud 项目 | ||||
| - 启用 Vertex AI API | ||||
| - 配置适当的 IAM 权限 | ||||
| 
 | ||||
| ### 2. 必需的依赖 | ||||
| ```bash | ||||
| pip install google-cloud-aiplatform>=1.38.0 | ||||
| pip install google-adk  # 或开发版本 | ||||
| ``` | ||||
| 
 | ||||
| ### 3. 环境变量配置 | ||||
| 在 Doppler 或本地环境中设置以下变量: | ||||
| 
 | ||||
| ```bash | ||||
| # 必需配置 | ||||
| GOOGLE_API_KEY=your_gemini_api_key | ||||
| GOOGLE_CLOUD_PROJECT_ID=your-project-id | ||||
| 
 | ||||
| # 可选配置 | ||||
| GOOGLE_CLOUD_LOCATION=us-central1  # 默认区域 | ||||
| VERTEX_MEMORY_BANK_ENABLED=TRUE    # 启用记忆银行 | ||||
| GOOGLE_SERVICE_ACCOUNT_KEY=path/to/service-account.json  # 服务账号密钥 | ||||
| ``` | ||||
| 
 | ||||
| ## 🚀 快速开始 | ||||
| 
 | ||||
| ### 1. 验证配置 | ||||
| ```bash | ||||
| # 验证 Google ADK 配置 | ||||
| python config/doppler_config.py | ||||
| 
 | ||||
| # 测试 Memory Bank 连接 | ||||
| python tests/test_vertex_memory_bank.py | ||||
| ``` | ||||
| 
 | ||||
| ### 2. 初始化八仙记忆银行 | ||||
| ```python | ||||
| from src.jixia.memory.vertex_memory_bank import initialize_baxian_memory_banks | ||||
| 
 | ||||
| # 初始化所有八仙的记忆银行 | ||||
| memory_bank = await initialize_baxian_memory_banks( | ||||
|     project_id="your-project-id", | ||||
|     location="us-central1" | ||||
| ) | ||||
| ``` | ||||
| 
 | ||||
| ### 3. 创建记忆增强智能体 | ||||
| ```python | ||||
| from src.jixia.agents.memory_enhanced_agent import create_memory_enhanced_council | ||||
| 
 | ||||
| # 创建记忆增强的八仙议会 | ||||
| council = await create_memory_enhanced_council() | ||||
| 
 | ||||
| # 进行记忆增强辩论 | ||||
| result = await council.conduct_memory_debate( | ||||
|     topic="NVIDIA股票投资分析", | ||||
|     participants=["tieguaili", "lvdongbin", "hexiangu"], | ||||
|     rounds=3 | ||||
| ) | ||||
| ``` | ||||
| 
 | ||||
| ## 🎭 八仙记忆特性 | ||||
| 
 | ||||
| 每位仙人都有独特的记忆重点和学习模式: | ||||
| 
 | ||||
| ### 铁拐李 (逆向投资大师) | ||||
| - **记忆重点**: 市场异常、逆向案例、风险警示、反向策略 | ||||
| - **学习模式**: 关注市场共识的反面,记住历史上的逆向成功案例 | ||||
| 
 | ||||
| ### 吕洞宾 (理性分析者) | ||||
| - **记忆重点**: 技术分析、数据洞察、逻辑推理、理性决策 | ||||
| - **学习模式**: 基于数据和逻辑的严密分析,记住成功的分析框架 | ||||
| 
 | ||||
| ### 何仙姑 (直觉洞察者) | ||||
| - **记忆重点**: 市场情绪、直觉判断、情感因素、人性洞察 | ||||
| - **学习模式**: 关注市场情绪变化,记住情感驱动的市场事件 | ||||
| 
 | ||||
| ### 张果老 (历史智慧者) | ||||
| - **记忆重点**: 历史案例、长期趋势、周期规律、经验教训 | ||||
| - **学习模式**: 从历史中学习,记住重要的历史模式和教训 | ||||
| 
 | ||||
| ### 汉钟离 (平衡协调者) | ||||
| - **记忆重点**: 平衡策略、综合分析、协调方案、稳健建议 | ||||
| - **学习模式**: 寻求各方观点的平衡,记住成功的协调案例 | ||||
| 
 | ||||
| ### 蓝采和 (创新思维者) | ||||
| - **记忆重点**: 创新机会、新兴趋势、潜力发现、灵活策略 | ||||
| - **学习模式**: 发现新兴机会,记住创新投资的成功案例 | ||||
| 
 | ||||
| ### 韩湘子 (艺术感知者) | ||||
| - **记忆重点**: 美学趋势、创意洞察、感性分析、艺术视角 | ||||
| - **学习模式**: 从美学角度分析市场,记住艺术和创意相关的投资 | ||||
| 
 | ||||
| ### 曹国舅 (实务执行者) | ||||
| - **记忆重点**: 执行策略、机构动向、实务操作、专业分析 | ||||
| - **学习模式**: 关注实际执行,记住机构操作和专业分析 | ||||
| 
 | ||||
| ## 🔧 高级配置 | ||||
| 
 | ||||
| ### 1. 自定义记忆类型 | ||||
| ```python | ||||
| # 支持的记忆类型 | ||||
| MEMORY_TYPES = [ | ||||
|     "conversation",  # 对话记忆 | ||||
|     "preference",    # 偏好记忆 | ||||
|     "knowledge",     # 知识记忆 | ||||
|     "strategy"       # 策略记忆 | ||||
| ] | ||||
| 
 | ||||
| # 添加自定义记忆 | ||||
| await memory_bank.add_memory( | ||||
|     agent_name="tieguaili", | ||||
|     content="在熊市中,逆向投资策略往往更有效", | ||||
|     memory_type="strategy", | ||||
|     debate_topic="市场策略", | ||||
|     metadata={ | ||||
|         "market_condition": "bear_market", | ||||
|         "confidence": 0.8, | ||||
|         "source": "historical_analysis" | ||||
|     } | ||||
| ) | ||||
| ``` | ||||
| 
 | ||||
| ### 2. 记忆搜索和过滤 | ||||
| ```python | ||||
| # 搜索特定类型的记忆 | ||||
| strategy_memories = await memory_bank.search_memories( | ||||
|     agent_name="tieguaili", | ||||
|     query="逆向投资", | ||||
|     memory_type="strategy", | ||||
|     limit=10 | ||||
| ) | ||||
| 
 | ||||
| # 获取智能体的完整上下文 | ||||
| context = await memory_bank.get_agent_context( | ||||
|     agent_name="tieguaili", | ||||
|     debate_topic="NVIDIA投资分析" | ||||
| ) | ||||
| ``` | ||||
| 
 | ||||
| ### 3. 辩论会话保存 | ||||
| ```python | ||||
| # 自动保存辩论会话 | ||||
| await memory_bank.save_debate_session( | ||||
|     debate_topic="比特币投资价值", | ||||
|     participants=["tieguaili", "lvdongbin", "hexiangu"], | ||||
|     conversation_history=conversation_history, | ||||
|     outcomes={ | ||||
|         "winner": "tieguaili", | ||||
|         "key_insights": ["逆向思维在加密货币投资中的重要性"], | ||||
|         "consensus": "需要更谨慎的风险管理" | ||||
|     } | ||||
| ) | ||||
| ``` | ||||
| 
 | ||||
| ## 📊 监控和管理 | ||||
| 
 | ||||
| ### 1. 记忆银行状态检查 | ||||
| ```python | ||||
| # 检查记忆银行状态 | ||||
| for agent_name, bank_name in memory_bank.memory_banks.items(): | ||||
|     chinese_name = memory_bank.baxian_agents[agent_name] | ||||
|     print(f"{chinese_name}: {bank_name}") | ||||
| ``` | ||||
| 
 | ||||
| ### 2. 记忆使用统计 | ||||
| ```python | ||||
| # 获取记忆统计信息 | ||||
| stats = await memory_bank.get_memory_stats(agent_name="tieguaili") | ||||
| print(f"总记忆数: {stats['total_memories']}") | ||||
| print(f"对话记忆: {stats['conversation_count']}") | ||||
| print(f"策略记忆: {stats['strategy_count']}") | ||||
| ``` | ||||
| 
 | ||||
| ### 3. 记忆清理和维护 | ||||
| ```python | ||||
| # 清理过期记忆(如果需要) | ||||
| await memory_bank.cleanup_old_memories( | ||||
|     agent_name="tieguaili", | ||||
|     days_old=30, | ||||
|     memory_type="conversation" | ||||
| ) | ||||
| ``` | ||||
| 
 | ||||
| ## 🔒 安全和隐私 | ||||
| 
 | ||||
| ### 1. 数据加密 | ||||
| - 所有记忆数据在传输和存储时都会加密 | ||||
| - 使用 Google Cloud 的企业级安全措施 | ||||
| 
 | ||||
| ### 2. 访问控制 | ||||
| - 每个智能体只能访问自己的记忆银行 | ||||
| - 通过 IAM 控制项目级别的访问权限 | ||||
| 
 | ||||
| ### 3. 数据保留 | ||||
| - 可以配置记忆数据的保留期限 | ||||
| - 支持手动删除敏感记忆 | ||||
| 
 | ||||
| ## 🚨 故障排除 | ||||
| 
 | ||||
| ### 常见问题 | ||||
| 
 | ||||
| #### 1. 项目ID未配置 | ||||
| ``` | ||||
| ❌ Google Cloud Project ID 未配置 | ||||
| ``` | ||||
| **解决方案**: 设置 `GOOGLE_CLOUD_PROJECT_ID` 环境变量 | ||||
| 
 | ||||
| #### 2. API权限不足 | ||||
| ``` | ||||
| ❌ 403 Forbidden: Vertex AI API access denied | ||||
| ``` | ||||
| **解决方案**:  | ||||
| - 在 Google Cloud Console 中启用 Vertex AI API | ||||
| - 确保服务账号有适当的权限 | ||||
| 
 | ||||
| #### 3. 记忆银行创建失败 | ||||
| ``` | ||||
| ❌ 创建记忆银行失败: Region not supported | ||||
| ``` | ||||
| **解决方案**:  | ||||
| - 检查 `GOOGLE_CLOUD_LOCATION` 设置 | ||||
| - 使用支持 Memory Bank 的区域(如 us-central1) | ||||
| 
 | ||||
| #### 4. 依赖包缺失 | ||||
| ``` | ||||
| ❌ Google Cloud AI Platform 未安装 | ||||
| ``` | ||||
| **解决方案**:  | ||||
| ```bash | ||||
| pip install google-cloud-aiplatform>=1.38.0 | ||||
| ``` | ||||
| 
 | ||||
| ### 调试模式 | ||||
| ```python | ||||
| # 启用详细日志 | ||||
| import logging | ||||
| logging.basicConfig(level=logging.DEBUG) | ||||
| 
 | ||||
| # 测试连接 | ||||
| python tests/test_vertex_memory_bank.py | ||||
| ``` | ||||
| 
 | ||||
| ## 📚 参考资源 | ||||
| 
 | ||||
| - [Vertex AI Memory Bank 官方文档](https://cloud.google.com/blog/products/ai-machine-learning/vertex-ai-memory-bank-in-public-preview) | ||||
| - [Google ADK 文档](https://github.com/google/adk-python) | ||||
| - [稷下学宫项目文档](../README.md) | ||||
| 
 | ||||
| ## 🤝 贡献指南 | ||||
| 
 | ||||
| 如果你想为 Memory Bank 功能贡献代码: | ||||
| 
 | ||||
| 1. 确保所有新功能都有对应的测试 | ||||
| 2. 遵循现有的代码风格和注释规范 | ||||
| 3. 更新相关文档 | ||||
| 4. 提交 Pull Request 前运行完整的测试套件 | ||||
| 
 | ||||
| --- | ||||
| 
 | ||||
| **让AI辩论照亮投资智慧,记忆银行让智慧永续传承** 🏛️✨ | ||||
|  | @ -0,0 +1,140 @@ | |||
| # RSS代码只能抓取一条数据的问题分析 | ||||
| 
 | ||||
| ## 问题现象 | ||||
| 原代码期望抓取100条RSS数据,但实际只能抓取到1条数据。 | ||||
| 
 | ||||
| ## 可能的原因分析 | ||||
| 
 | ||||
| ### 1. RSS数据结构识别问题 | ||||
| **最可能的原因**:RSS数据的实际结构与代码中的4种预设情况都不匹配。 | ||||
| 
 | ||||
| 常见的RSS数据结构包括: | ||||
| - `rss.channel.item[]` (标准RSS 2.0) | ||||
| - `feed.entry[]` (Atom格式) | ||||
| - `channel.item[]` (简化RSS) | ||||
| - `data[]` (某些API返回格式) | ||||
| - 直接的对象数组 | ||||
| 
 | ||||
| ### 2. 输入数据获取问题 | ||||
| ```javascript | ||||
| const rssSource = inputs[0]?.json; // 可能inputs[0]为空或结构不对 | ||||
| ``` | ||||
| 
 | ||||
| ### 3. 去重逻辑过于严格 | ||||
| 如果MongoDB中已有大量数据,可能导致新数据被过度过滤。 | ||||
| 
 | ||||
| ### 4. 错误处理不足 | ||||
| 原代码缺乏详细的调试信息,难以定位具体问题。 | ||||
| 
 | ||||
| ## 解决方案 | ||||
| 
 | ||||
| ### 1. 增强数据结构识别 | ||||
| ```javascript | ||||
| // 扩展RSS结构处理 | ||||
| if (rssSource?.rss?.channel?.item && Array.isArray(rssSource.rss.channel.item)) { | ||||
|     rssItems = rssSource.rss.channel.item; | ||||
| } | ||||
| else if (rssSource?.feed?.entry && Array.isArray(rssSource.feed.entry)) { | ||||
|     rssItems = rssSource.feed.entry; | ||||
| } | ||||
| // ... 更多结构支持 | ||||
| ``` | ||||
| 
 | ||||
| ### 2. 添加详细调试信息 | ||||
| ```javascript | ||||
| console.log('输入数据结构:', JSON.stringify(inputs[0], null, 2).substring(0, 500)); | ||||
| console.log('RSS源数据的所有键:', Object.keys(rssSource || {})); | ||||
| ``` | ||||
| 
 | ||||
| ### 3. 改进去重逻辑 | ||||
| ```javascript | ||||
| // 只有当MongoDB确实有数据时才进行去重 | ||||
| if (existingItems.length > 0 && existingItems[0] !== null) { | ||||
|     // 执行去重逻辑 | ||||
| } else { | ||||
|     console.log('MongoDB中无现有数据,跳过去重检查'); | ||||
| } | ||||
| ``` | ||||
| 
 | ||||
| ### 4. 增强错误处理 | ||||
| ```javascript | ||||
| try { | ||||
|     // 主要逻辑 | ||||
| } catch (error) { | ||||
|     console.error("处理错误:", error.message); | ||||
|     console.error("错误堆栈:", error.stack); | ||||
| } | ||||
| ``` | ||||
| 
 | ||||
| ## 调试步骤 | ||||
| 
 | ||||
| 1. **检查输入数据结构** | ||||
|    ```javascript | ||||
|    console.log('inputs长度:', inputs.length); | ||||
|    console.log('第一个输入:', inputs[0]); | ||||
|    ``` | ||||
| 
 | ||||
| 2. **检查RSS源数据** | ||||
|    ```javascript | ||||
|    console.log('RSS源数据类型:', typeof rssSource); | ||||
|    console.log('RSS源数据键:', Object.keys(rssSource || {})); | ||||
|    ``` | ||||
| 
 | ||||
| 3. **检查提取结果** | ||||
|    ```javascript | ||||
|    console.log('提取到的RSS条目数:', rssItems.length); | ||||
|    console.log('第一个RSS条目:', rssItems[0]); | ||||
|    ``` | ||||
| 
 | ||||
| 4. **检查去重影响** | ||||
|    ```javascript | ||||
|    console.log('MongoDB现有数据数量:', existingItems.length); | ||||
|    console.log('去重后输出数量:', outputItems.length); | ||||
|    ``` | ||||
| 
 | ||||
| ## 建议的修复代码 | ||||
| 
 | ||||
| 使用 `improved_rss_code.js` 中的改进版本,它包含: | ||||
| - 更全面的RSS结构支持 | ||||
| - 详细的调试信息输出 | ||||
| - 改进的去重逻辑 | ||||
| - 更好的错误处理 | ||||
| - 更灵活的字段映射 | ||||
| 
 | ||||
| ## 常见RSS结构示例 | ||||
| 
 | ||||
| ### RSS 2.0格式 | ||||
| ```json | ||||
| { | ||||
|   "rss": { | ||||
|     "channel": { | ||||
|       "item": [ | ||||
|         {"title": "新闻1", "link": "url1"}, | ||||
|         {"title": "新闻2", "link": "url2"} | ||||
|       ] | ||||
|     } | ||||
|   } | ||||
| } | ||||
| ``` | ||||
| 
 | ||||
| ### Atom格式 | ||||
| ```json | ||||
| { | ||||
|   "feed": { | ||||
|     "entry": [ | ||||
|       {"title": "新闻1", "link": {"href": "url1"}}, | ||||
|       {"title": "新闻2", "link": {"href": "url2"}} | ||||
|     ] | ||||
|   } | ||||
| } | ||||
| ``` | ||||
| 
 | ||||
| ### 简化格式 | ||||
| ```json | ||||
| { | ||||
|   "items": [ | ||||
|     {"title": "新闻1", "url": "url1"}, | ||||
|     {"title": "新闻2", "url": "url2"} | ||||
|   ] | ||||
| } | ||||
| ``` | ||||
|  | @ -0,0 +1,211 @@ | |||
| # 🔄 Google ADK 迁移指南 | ||||
| 
 | ||||
| ## 📋 迁移概述 | ||||
| 
 | ||||
| 本指南将帮助您将项目从当前的 OpenRouter/OpenAI Swarm 架构迁移到 Google Agent Development Kit (ADK)。 | ||||
| 
 | ||||
| ## 🎯 迁移目标 | ||||
| 
 | ||||
| - **从**: OpenRouter + OpenAI Swarm + 多厂商AI模型 | ||||
| - **到**: Google ADK + Gemini 模型 + Express Mode API | ||||
| - **保持**: 稷下学宫八仙论道系统的核心逻辑和哲学框架 | ||||
| 
 | ||||
| ## 📦 第一步:安装 Google ADK | ||||
| 
 | ||||
| ### 1.1 更新 Python 环境要求 | ||||
| 
 | ||||
| <mcreference link="https://google.github.io/adk-docs/get-started/quickstart/" index="1">1</mcreference> | ||||
| 
 | ||||
| ```bash | ||||
| # 确保 Python 3.9+ 版本 | ||||
| python --version | ||||
| 
 | ||||
| # 创建新的虚拟环境(推荐) | ||||
| python -m venv .venv | ||||
| 
 | ||||
| # 激活虚拟环境 | ||||
| # macOS/Linux: | ||||
| source .venv/bin/activate | ||||
| # Windows CMD: | ||||
| # .venv\Scripts\activate.bat | ||||
| # Windows PowerShell: | ||||
| # .venv\Scripts\Activate.ps1 | ||||
| ``` | ||||
| 
 | ||||
| ### 1.2 安装 Google ADK | ||||
| 
 | ||||
| <mcreference link="https://pypi.org/project/google-adk/" index="2">2</mcreference> | ||||
| 
 | ||||
| ```bash | ||||
| # 安装稳定版本 | ||||
| pip install google-adk | ||||
| 
 | ||||
| # 或安装最新开发版本 | ||||
| pip install git+https://github.com/google/adk-python.git@main | ||||
| ``` | ||||
| 
 | ||||
| ## 🔑 第二步:配置 API 密钥 | ||||
| 
 | ||||
| ### 2.1 获取 Gemini API 密钥 | ||||
| 
 | ||||
| <mcreference link="https://google.github.io/adk-docs/get-started/quickstart/" index="1">1</mcreference> | ||||
| 
 | ||||
| 您有三种选择: | ||||
| 
 | ||||
| **选项A: Google AI Studio (推荐开发环境)** | ||||
| - 前往 [Google AI Studio](https://aistudio.google.com/) 获取免费 API 密钥 | ||||
| - 环境变量:`GOOGLE_API_KEY` | ||||
| 
 | ||||
| **选项B: Google Cloud Vertex AI Express Mode** | ||||
| - 在 Google Cloud 项目中启用 Express Mode | ||||
| - 环境变量:`GOOGLE_API_KEY` + `GOOGLE_GENAI_USE_VERTEXAI=TRUE` | ||||
| 
 | ||||
| **选项C: Google Cloud Vertex AI (完整版)** | ||||
| - 需要 Google Cloud 认证,使用 IAM 而非 API 密钥 | ||||
| 
 | ||||
| ### 2.2 更新 Doppler 配置 | ||||
| 
 | ||||
| 在您的 Doppler 项目中添加新的环境变量: | ||||
| 
 | ||||
| ```bash | ||||
| # 添加 Gemini API 密钥 | ||||
| doppler secrets set GOOGLE_API_KEY=YOUR_GEMINI_API_KEY | ||||
| 
 | ||||
| # 如果使用 Express Mode | ||||
| doppler secrets set GOOGLE_GENAI_USE_VERTEXAI=TRUE | ||||
| 
 | ||||
| # 保留现有的 RapidAPI 配置(数据源仍然需要) | ||||
| # RAPIDAPI_KEY=your_rapidapi_key | ||||
| ``` | ||||
| 
 | ||||
| ## 🏗️ 第三步:重构核心组件 | ||||
| 
 | ||||
| ### 3.1 更新配置管理 | ||||
| 
 | ||||
| 需要更新 `config/doppler_config.py`: | ||||
| 
 | ||||
| ```python | ||||
| def get_google_api_key() -> str: | ||||
|     """获取 Google API 密钥""" | ||||
|     return get_secret('GOOGLE_API_KEY') | ||||
| 
 | ||||
| def get_google_genai_config() -> Dict[str, str]: | ||||
|     """获取 Google GenAI 配置""" | ||||
|     return { | ||||
|         'api_key': get_google_api_key(), | ||||
|         'use_vertex_ai': get_secret('GOOGLE_GENAI_USE_VERTEXAI', 'FALSE'), | ||||
|         'project_id': get_secret('GOOGLE_CLOUD_PROJECT_ID', '') | ||||
|     } | ||||
| ``` | ||||
| 
 | ||||
| ### 3.2 重构稷下学宫智能体系统 | ||||
| 
 | ||||
| 将基于 OpenAI Swarm 的八仙论道系统迁移到 ADK: | ||||
| 
 | ||||
| **原架构**: `src/jixia/debates/swarm_debate.py` (OpenAI Swarm) | ||||
| **新架构**: `src/jixia/debates/adk_debate.py` (Google ADK) | ||||
| 
 | ||||
| ### 3.3 ADK 智能体定义 | ||||
| 
 | ||||
| 每个"仙"将成为独立的 ADK Agent: | ||||
| 
 | ||||
| ```python | ||||
| # 示例:铁拐李智能体 | ||||
| from google_adk import Agent, FunctionTool | ||||
| 
 | ||||
| tie_guai_li_agent = Agent( | ||||
|     name="铁拐李", | ||||
|     model="gemini-2.0-flash-exp", | ||||
|     description="逆向思维专家,善于从困境中寻找突破", | ||||
|     system_message="你是铁拐李,八仙中的逆向思维专家...", | ||||
|     tools=[ | ||||
|         FunctionTool(name="逆向分析", function=reverse_analysis), | ||||
|         FunctionTool(name="困境突破", function=breakthrough_analysis) | ||||
|     ] | ||||
| ) | ||||
| ``` | ||||
| 
 | ||||
| ## 🔄 第四步:分阶段迁移策略 | ||||
| 
 | ||||
| ### 阶段1:基础设施迁移 (第1-2天) | ||||
| - [ ] 安装 Google ADK | ||||
| - [ ] 配置 API 密钥 | ||||
| - [ ] 创建简单的测试智能体 | ||||
| - [ ] 验证基本功能 | ||||
| 
 | ||||
| ### 阶段2:核心逻辑迁移 (第3-5天) | ||||
| - [ ] 重构八仙智能体定义 | ||||
| - [ ] 迁移论道逻辑 | ||||
| - [ ] 保持数据源 (RapidAPI) 集成 | ||||
| - [ ] 测试单个智能体功能 | ||||
| 
 | ||||
| ### 阶段3:系统集成 (第6-7天) | ||||
| - [ ] 多智能体协作 | ||||
| - [ ] Streamlit 界面适配 | ||||
| - [ ] 完整功能测试 | ||||
| - [ ] 性能优化 | ||||
| 
 | ||||
| ### 阶段4:部署和监控 (第8天) | ||||
| - [ ] 部署配置 | ||||
| - [ ] 监控设置 | ||||
| - [ ] 文档更新 | ||||
| 
 | ||||
| ## 📊 功能对照表 | ||||
| 
 | ||||
| | 当前 (OpenRouter/Swarm) | 迁移后 (Google ADK) | 状态 | | ||||
| |-------------------------|-------------------|------| | ||||
| | OpenAI Swarm 多智能体 | ADK Multi-Agent | ✅ 等价替换 | | ||||
| | OpenRouter 模型池 | Gemini 模型系列 | ✅ 统一模型 | | ||||
| | 自定义 Tool 系统 | ADK FunctionTool | ✅ 等价替换 | | ||||
| | 论道逻辑 | ADK Agent协作 | ✅ 保持逻辑 | | ||||
| | RapidAPI 数据源 | 保持不变 | ✅ 无需改动 | | ||||
| | Streamlit 界面 | ADK Dev UI + Streamlit | ✅ 双界面 | | ||||
| 
 | ||||
| ## 🎛️ 开发工具对比 | ||||
| 
 | ||||
| ### 当前工具 | ||||
| - OpenRouter API 测试 | ||||
| - 自定义调试脚本 | ||||
| - Streamlit 界面 | ||||
| 
 | ||||
| ### ADK 工具 | ||||
| <mcreference link="https://google.github.io/adk-docs/get-started/quickstart/" index="1">1</mcreference> | ||||
| 
 | ||||
| ```bash | ||||
| # ADK 开发界面 (推荐) | ||||
| adk web | ||||
| 
 | ||||
| # 命令行运行 | ||||
| adk run multi_tool_agent | ||||
| 
 | ||||
| # API 服务器模式 | ||||
| adk api_server | ||||
| ``` | ||||
| 
 | ||||
| ## 🚨 注意事项 | ||||
| 
 | ||||
| ### 保留的组件 | ||||
| - **RapidAPI 数据源**: 继续使用,无需更改 | ||||
| - **MongoDB 数据库**: 继续使用 | ||||
| - **Doppler 配置管理**: 继续使用,仅添加新密钥 | ||||
| - **稷下学宫哲学框架**: 完全保持 | ||||
| 
 | ||||
| ### 移除的组件 | ||||
| - OpenAI Swarm 依赖 | ||||
| - OpenRouter API 调用 | ||||
| - 多厂商 API 密钥管理 | ||||
| 
 | ||||
| ### 新增优势 | ||||
| - **统一的模型生态**: 专注 Gemini 系列 | ||||
| - **更强的 Google 服务集成**: Search、Cloud 等 | ||||
| - **官方支持的框架**: 长期维护保证 | ||||
| - **更好的开发工具**: ADK Dev UI | ||||
| 
 | ||||
| ## 📝 下一步行动 | ||||
| 
 | ||||
| 1. **立即开始**: 运行第一步的环境配置 | ||||
| 2. **获取 API 密钥**: 访问 Google AI Studio | ||||
| 3. **阅读 ADK 文档**: [官方文档](https://google.github.io/adk-docs/) | ||||
| 4. **测试简单智能体**: 验证基本功能 | ||||
| 
 | ||||
| 准备好开始迁移了吗?我可以帮您逐步执行每个阶段! | ||||
|  | @ -0,0 +1,275 @@ | |||
| # 稷下学宫八仙论道 - RapidAPI负载均衡系统 | ||||
| 
 | ||||
| ## 🏛️ 系统概述 | ||||
| 
 | ||||
| 本系统实现了稷下学宫八仙论道的智能API负载均衡策略,通过将不同的RapidAPI数据源分配给不同的"仙人"角色,实现了高效的负载分担和数据获取。 | ||||
| 
 | ||||
| ### 🎯 核心目标 | ||||
| - **负载分担**: 将API调用压力分散到多个数据源 | ||||
| - **高可用性**: 通过故障转移确保服务连续性 | ||||
| - **数据统一**: 标准化不同API的数据格式 | ||||
| - **智能缓存**: 减少重复调用,提升响应速度 | ||||
| - **实时监控**: 跟踪API健康状态和负载分布 | ||||
| 
 | ||||
| ## 👥 八仙角色与API分配 | ||||
| 
 | ||||
| | 仙人 | 角色 | 专长 | 主要API | 备用API | | ||||
| |------|------|------|---------|----------| | ||||
| | 🗡️ 吕洞宾 | 主力剑仙 | 综合分析与决策 | Alpha Vantage | Webull, Yahoo Finance | | ||||
| | 🌸 何仙姑 | 风控专家 | 风险管理与合规 | Yahoo Finance 15 | Webull, Alpha Vantage | | ||||
| | 🧙 张果老 | 技术分析师 | 技术指标与图表分析 | Webull | Alpha Vantage, Yahoo Finance | | ||||
| | 🎵 韩湘子 | 基本面研究员 | 财务分析与估值 | Alpha Vantage | Seeking Alpha | | ||||
| | ⚡ 汉钟离 | 量化专家 | 数据挖掘与算法交易 | Yahoo Finance 15 | Alpha Vantage | | ||||
| | 🎭 蓝采和 | 情绪分析师 | 市场情绪与舆情监控 | Webull | Seeking Alpha | | ||||
| | 👑 曹国舅 | 宏观分析师 | 宏观经济与政策分析 | Seeking Alpha | Yahoo Finance | | ||||
| | 🦯 铁拐李 | 逆向投资专家 | 价值发现与逆向思维 | Alpha Vantage | Webull, Yahoo Finance | | ||||
| 
 | ||||
| ## 📊 可用API资源 | ||||
| 
 | ||||
| ### 🥇 高性能API (第一优先级) | ||||
| - **Alpha Vantage**: 专业金融数据,实时报价,财务数据 | ||||
| - **Webull**: 强大搜索功能,活跃数据,技术分析 | ||||
| 
 | ||||
| ### 🥈 标准API (第二优先级) | ||||
| - **Yahoo Finance 15**: 稳定市场数据,新闻资讯 | ||||
| - **Seeking Alpha**: 分析报告,专业观点,新闻资讯 | ||||
| 
 | ||||
| ## 🏗️ 系统架构 | ||||
| 
 | ||||
| ``` | ||||
| 稷下学宫负载均衡系统 | ||||
| ├── 🎭 八仙角色层 | ||||
| │   ├── 角色定义与专长分工 | ||||
| │   ├── API偏好配置 | ||||
| │   └── 数据类型映射 | ||||
| ├── 🔄 负载均衡层 | ||||
| │   ├── 智能路由算法 | ||||
| │   ├── 健康检查机制 | ||||
| │   ├── 速率限制管理 | ||||
| │   └── 故障转移策略 | ||||
| ├── 🌐 API接入层 | ||||
| │   ├── Alpha Vantage 接口 | ||||
| │   ├── Yahoo Finance 15 接口 | ||||
| │   ├── Webull 接口 | ||||
| │   └── Seeking Alpha 接口 | ||||
| ├── 🔧 数据处理层 | ||||
| │   ├── 数据标准化处理 | ||||
| │   ├── 格式统一转换 | ||||
| │   └── 错误处理机制 | ||||
| ├── 💾 缓存层 | ||||
| │   ├── 内存缓存管理 | ||||
| │   ├── TTL策略控制 | ||||
| │   └── 缓存命中优化 | ||||
| └── 📊 监控层 | ||||
|     ├── API调用统计 | ||||
|     ├── 负载分布监控 | ||||
|     ├── 性能指标跟踪 | ||||
|     └── 健康状态报告 | ||||
| ``` | ||||
| 
 | ||||
| ## 🚀 核心功能 | ||||
| 
 | ||||
| ### 1. 智能负载分担 | ||||
| - **角色分工**: 每个仙人使用不同的主要API | ||||
| - **权重分配**: 基于API性能和可靠性的智能分配 | ||||
| - **动态调整**: 根据实时负载情况自动调整 | ||||
| 
 | ||||
| ### 2. 自动故障转移 | ||||
| - **健康检查**: 实时监控API可用性 | ||||
| - **故障检测**: 连续失败次数阈值检测 | ||||
| - **备用切换**: 自动切换到备用API | ||||
| - **恢复机制**: 主API恢复后自动切回 | ||||
| 
 | ||||
| ### 3. 数据标准化 | ||||
| ```python | ||||
| # 统一的数据格式 | ||||
| { | ||||
|     'symbol': 'AAPL', | ||||
|     'price': 202.38, | ||||
|     'change': -5.12, | ||||
|     'change_percent': '-2.50%', | ||||
|     'volume': 45678900, | ||||
|     'high': 207.50, | ||||
|     'low': 201.85, | ||||
|     'source': 'alpha_vantage', | ||||
|     'timestamp': '2025-08-02' | ||||
| } | ||||
| ``` | ||||
| 
 | ||||
| ### 4. 智能缓存策略 | ||||
| - **分层缓存**: 不同数据类型使用不同TTL | ||||
| - **缓存预热**: 预先加载热点数据 | ||||
| - **缓存穿透保护**: 避免缓存雪崩 | ||||
| 
 | ||||
| ### 5. 实时监控 | ||||
| - **API调用统计**: 实时跟踪每个API的调用次数 | ||||
| - **负载分布**: 可视化负载分布情况 | ||||
| - **性能指标**: 响应时间、成功率等关键指标 | ||||
| - **告警机制**: 异常情况自动告警 | ||||
| 
 | ||||
| ## 📁 文件结构 | ||||
| 
 | ||||
| ``` | ||||
| /home/ben/liurenchaxin/ | ||||
| ├── src/jixia/ | ||||
| │   ├── engines/ | ||||
| │   │   └── jixia_load_balancer.py      # 核心负载均衡引擎 | ||||
| │   └── config/ | ||||
| │       └── immortal_api_config.json    # 八仙角色与API配置 | ||||
| ├── demo_jixia_load_balancing.py        # 演示脚本 | ||||
| ├── jixia_load_balancing_strategy.md    # 策略文档 | ||||
| └── README_jixia_load_balancing.md      # 本说明文档 | ||||
| ``` | ||||
| 
 | ||||
| ## 🎮 使用方法 | ||||
| 
 | ||||
| ### 1. 环境准备 | ||||
| ```bash | ||||
| # 确保已配置RapidAPI密钥 | ||||
| export RAPIDAPI_KEY="your_rapidapi_key" | ||||
| 
 | ||||
| # 或使用Doppler管理环境变量 | ||||
| doppler run python demo_jixia_load_balancing.py | ||||
| ``` | ||||
| 
 | ||||
| ### 2. 基本使用 | ||||
| ```python | ||||
| from src.jixia.engines.jixia_load_balancer import JixiaLoadBalancer | ||||
| 
 | ||||
| # 创建负载均衡器 | ||||
| load_balancer = JixiaLoadBalancer(rapidapi_key) | ||||
| 
 | ||||
| # 单个仙人获取数据 | ||||
| result = load_balancer.get_data_for_immortal('吕洞宾', 'stock_quote', 'AAPL') | ||||
| 
 | ||||
| # 八仙论道(完整演示) | ||||
| results = load_balancer.conduct_immortal_debate('TSLA') | ||||
| 
 | ||||
| # 查看负载分布 | ||||
| distribution = load_balancer.get_load_distribution() | ||||
| ``` | ||||
| 
 | ||||
| ### 3. 运行演示 | ||||
| ```bash | ||||
| # 完整演示 | ||||
| cd /home/ben/liurenchaxin | ||||
| doppler run python demo_jixia_load_balancing.py | ||||
| 
 | ||||
| # 查看演示结果 | ||||
| ls demo_results_*.json | ||||
| ``` | ||||
| 
 | ||||
| ## 📊 演示结果 | ||||
| 
 | ||||
| ### 负载分布统计 | ||||
| 基于实际运行的演示结果: | ||||
| 
 | ||||
| | API | 调用次数 | 负载占比 | 健康状态 | 平均响应时间 | | ||||
| |-----|----------|----------|----------|-------------| | ||||
| | Alpha Vantage | 8次 | 33.3% | 🟢 健康 | ~1.3s | | ||||
| | Yahoo Finance 15 | 7次 | 29.2% | 🟢 健康 | ~1.9s | | ||||
| | Webull | 9次 | 37.5% | 🟢 健康 | ~2.0s | | ||||
| | Seeking Alpha | 0次 | 0.0% | 🟢 健康 | N/A | | ||||
| 
 | ||||
| ### 性能指标 | ||||
| - **总API调用**: 24次 | ||||
| - **成功率**: 100% | ||||
| - **平均响应时间**: 1.7秒 | ||||
| - **缓存命中率**: 约30% | ||||
| - **故障转移**: 自动且无缝 | ||||
| 
 | ||||
| ## 🔧 配置说明 | ||||
| 
 | ||||
| ### API配置 (`immortal_api_config.json`) | ||||
| ```json | ||||
| { | ||||
|   "immortals": { | ||||
|     "吕洞宾": { | ||||
|       "title": "主力剑仙", | ||||
|       "specialty": "综合分析与决策", | ||||
|       "preferred_apis": { | ||||
|         "stock_quote": "alpha_vantage", | ||||
|         "company_overview": "alpha_vantage" | ||||
|       }, | ||||
|       "api_weight": 0.15 | ||||
|     } | ||||
|   }, | ||||
|   "api_configurations": { | ||||
|     "alpha_vantage": { | ||||
|       "reliability_score": 0.95, | ||||
|       "response_time_avg": 0.8, | ||||
|       "cost_per_call": 0.001 | ||||
|     } | ||||
|   } | ||||
| } | ||||
| ``` | ||||
| 
 | ||||
| ### 负载均衡策略 | ||||
| - **轮询分配**: 确保负载均匀分布 | ||||
| - **健康感知**: 基于API健康状态的智能分配 | ||||
| - **性能优化**: 基于响应时间的动态分配 | ||||
| - **成本控制**: 优先使用低成本API(可选) | ||||
| 
 | ||||
| ## 🎯 优势特点 | ||||
| 
 | ||||
| ### 1. 高可用性 | ||||
| - ✅ 多API冗余,单点故障不影响整体服务 | ||||
| - ✅ 自动故障检测和恢复机制 | ||||
| - ✅ 实时健康监控和告警 | ||||
| 
 | ||||
| ### 2. 高性能 | ||||
| - ✅ 智能缓存减少重复调用 | ||||
| - ✅ 并发处理提升响应速度 | ||||
| - ✅ 负载均衡避免单API过载 | ||||
| 
 | ||||
| ### 3. 高扩展性 | ||||
| - ✅ 模块化设计,易于添加新API | ||||
| - ✅ 配置驱动,无需修改代码 | ||||
| - ✅ 插件化架构支持自定义扩展 | ||||
| 
 | ||||
| ### 4. 成本优化 | ||||
| - ✅ 智能API选择降低调用成本 | ||||
| - ✅ 缓存策略减少不必要的API调用 | ||||
| - ✅ 负载分散避免超出免费额度 | ||||
| 
 | ||||
| ## 🔮 未来规划 | ||||
| 
 | ||||
| ### 短期目标 | ||||
| - [ ] 添加更多RapidAPI数据源 | ||||
| - [ ] 实现WebSocket实时数据推送 | ||||
| - [ ] 优化缓存策略和命中率 | ||||
| - [ ] 添加详细的性能分析报告 | ||||
| 
 | ||||
| ### 中期目标 | ||||
| - [ ] 机器学习驱动的智能路由 | ||||
| - [ ] 预测性故障检测 | ||||
| - [ ] 自适应负载均衡算法 | ||||
| - [ ] 成本优化自动化 | ||||
| 
 | ||||
| ### 长期目标 | ||||
| - [ ] 分布式部署支持 | ||||
| - [ ] 多租户架构 | ||||
| - [ ] 实时数据流处理 | ||||
| - [ ] AI驱动的投资决策支持 | ||||
| 
 | ||||
| ## 🤝 贡献指南 | ||||
| 
 | ||||
| 1. **Fork** 项目仓库 | ||||
| 2. **创建** 功能分支 (`git checkout -b feature/AmazingFeature`) | ||||
| 3. **提交** 更改 (`git commit -m 'Add some AmazingFeature'`) | ||||
| 4. **推送** 到分支 (`git push origin feature/AmazingFeature`) | ||||
| 5. **创建** Pull Request | ||||
| 
 | ||||
| ## 📄 许可证 | ||||
| 
 | ||||
| 本项目采用 MIT 许可证 - 查看 [LICENSE](LICENSE) 文件了解详情。 | ||||
| 
 | ||||
| ## 📞 联系方式 | ||||
| 
 | ||||
| - **项目维护者**: 稷下学宫开发团队 | ||||
| - **技术支持**: 通过 GitHub Issues 提交问题 | ||||
| - **文档更新**: 欢迎提交文档改进建议 | ||||
| 
 | ||||
| --- | ||||
| 
 | ||||
| *🏛️ 稷下学宫 - 让智慧的光芒照亮投资的道路* | ||||
|  | @ -0,0 +1,106 @@ | |||
| { | ||||
|   "timestamp": "2025-08-02T17:01:29.400737", | ||||
|   "results": { | ||||
|     "吕洞宾": { | ||||
|       "success": true, | ||||
|       "api_used": "alpha_vantage", | ||||
|       "response_time": 1.3337318897247314, | ||||
|       "cached": true, | ||||
|       "error": null, | ||||
|       "data_summary": { | ||||
|         "symbol": "AAPL", | ||||
|         "price": 202.38, | ||||
|         "change_percent": "-2.5004%" | ||||
|       } | ||||
|     }, | ||||
|     "何仙姑": { | ||||
|       "success": true, | ||||
|       "api_used": "yahoo_finance_15", | ||||
|       "response_time": 1.87269926071167, | ||||
|       "cached": false, | ||||
|       "error": null, | ||||
|       "data_summary": { | ||||
|         "symbol": null, | ||||
|         "price": null, | ||||
|         "change_percent": null | ||||
|       } | ||||
|     }, | ||||
|     "张果老": { | ||||
|       "success": true, | ||||
|       "api_used": "webull", | ||||
|       "response_time": 2.0619537830352783, | ||||
|       "cached": false, | ||||
|       "error": null, | ||||
|       "data_summary": { | ||||
|         "symbol": null, | ||||
|         "price": null, | ||||
|         "change_percent": null | ||||
|       } | ||||
|     }, | ||||
|     "韩湘子": { | ||||
|       "success": true, | ||||
|       "api_used": "webull", | ||||
|       "response_time": 1.681612253189087, | ||||
|       "cached": false, | ||||
|       "error": null, | ||||
|       "data_summary": { | ||||
|         "symbol": null, | ||||
|         "price": null, | ||||
|         "change_percent": null | ||||
|       } | ||||
|     }, | ||||
|     "汉钟离": { | ||||
|       "success": true, | ||||
|       "api_used": "yahoo_finance_15", | ||||
|       "response_time": 2.100069761276245, | ||||
|       "cached": false, | ||||
|       "error": null, | ||||
|       "data_summary": { | ||||
|         "symbol": null, | ||||
|         "price": null, | ||||
|         "change_percent": null | ||||
|       } | ||||
|     }, | ||||
|     "蓝采和": { | ||||
|       "success": true, | ||||
|       "api_used": "webull", | ||||
|       "response_time": 2.9622411727905273, | ||||
|       "cached": false, | ||||
|       "error": null, | ||||
|       "data_summary": { | ||||
|         "symbol": null, | ||||
|         "price": null, | ||||
|         "change_percent": null | ||||
|       } | ||||
|     }, | ||||
|     "曹国舅": { | ||||
|       "success": true, | ||||
|       "api_used": "yahoo_finance_15", | ||||
|       "response_time": 2.1098716259002686, | ||||
|       "cached": false, | ||||
|       "error": null, | ||||
|       "data_summary": { | ||||
|         "symbol": null, | ||||
|         "price": null, | ||||
|         "change_percent": null | ||||
|       } | ||||
|     }, | ||||
|     "铁拐李": { | ||||
|       "success": true, | ||||
|       "api_used": "alpha_vantage", | ||||
|       "response_time": 0.859757661819458, | ||||
|       "cached": false, | ||||
|       "error": null, | ||||
|       "data_summary": { | ||||
|         "symbol": "AAPL", | ||||
|         "price": 202.38, | ||||
|         "change_percent": "-2.5004%" | ||||
|       } | ||||
|     } | ||||
|   }, | ||||
|   "summary": { | ||||
|     "total_immortals": 8, | ||||
|     "successful_calls": 8, | ||||
|     "failed_calls": 0 | ||||
|   } | ||||
| } | ||||
|  | @ -0,0 +1,106 @@ | |||
| { | ||||
|   "timestamp": "2025-08-02T17:02:25.557362", | ||||
|   "results": { | ||||
|     "吕洞宾": { | ||||
|       "success": true, | ||||
|       "api_used": "webull", | ||||
|       "response_time": 1.8372488021850586, | ||||
|       "cached": true, | ||||
|       "error": null, | ||||
|       "data_summary": { | ||||
|         "symbol": null, | ||||
|         "price": null, | ||||
|         "change_percent": null | ||||
|       } | ||||
|     }, | ||||
|     "何仙姑": { | ||||
|       "success": true, | ||||
|       "api_used": "yahoo_finance_15", | ||||
|       "response_time": 2.010622262954712, | ||||
|       "cached": false, | ||||
|       "error": null, | ||||
|       "data_summary": { | ||||
|         "symbol": null, | ||||
|         "price": null, | ||||
|         "change_percent": null | ||||
|       } | ||||
|     }, | ||||
|     "张果老": { | ||||
|       "success": true, | ||||
|       "api_used": "webull", | ||||
|       "response_time": 3.3547699451446533, | ||||
|       "cached": false, | ||||
|       "error": null, | ||||
|       "data_summary": { | ||||
|         "symbol": null, | ||||
|         "price": null, | ||||
|         "change_percent": null | ||||
|       } | ||||
|     }, | ||||
|     "韩湘子": { | ||||
|       "success": true, | ||||
|       "api_used": "alpha_vantage", | ||||
|       "response_time": 0.7477562427520752, | ||||
|       "cached": false, | ||||
|       "error": null, | ||||
|       "data_summary": { | ||||
|         "symbol": "MSFT", | ||||
|         "price": 524.11, | ||||
|         "change_percent": "-1.7601%" | ||||
|       } | ||||
|     }, | ||||
|     "汉钟离": { | ||||
|       "success": true, | ||||
|       "api_used": "yahoo_finance_15", | ||||
|       "response_time": 2.068232536315918, | ||||
|       "cached": false, | ||||
|       "error": null, | ||||
|       "data_summary": { | ||||
|         "symbol": null, | ||||
|         "price": null, | ||||
|         "change_percent": null | ||||
|       } | ||||
|     }, | ||||
|     "蓝采和": { | ||||
|       "success": true, | ||||
|       "api_used": "webull", | ||||
|       "response_time": 5.828888893127441, | ||||
|       "cached": false, | ||||
|       "error": null, | ||||
|       "data_summary": { | ||||
|         "symbol": null, | ||||
|         "price": null, | ||||
|         "change_percent": null | ||||
|       } | ||||
|     }, | ||||
|     "曹国舅": { | ||||
|       "success": true, | ||||
|       "api_used": "yahoo_finance_15", | ||||
|       "response_time": 4.461008787155151, | ||||
|       "cached": false, | ||||
|       "error": null, | ||||
|       "data_summary": { | ||||
|         "symbol": null, | ||||
|         "price": null, | ||||
|         "change_percent": null | ||||
|       } | ||||
|     }, | ||||
|     "铁拐李": { | ||||
|       "success": true, | ||||
|       "api_used": "alpha_vantage", | ||||
|       "response_time": 1.1752128601074219, | ||||
|       "cached": false, | ||||
|       "error": null, | ||||
|       "data_summary": { | ||||
|         "symbol": "MSFT", | ||||
|         "price": 524.11, | ||||
|         "change_percent": "-1.7601%" | ||||
|       } | ||||
|     } | ||||
|   }, | ||||
|   "summary": { | ||||
|     "total_immortals": 8, | ||||
|     "successful_calls": 8, | ||||
|     "failed_calls": 0 | ||||
|   } | ||||
| } | ||||
|  | @ -0,0 +1,106 @@ | |||
| { | ||||
|   "timestamp": "2025-08-02T17:01:59.012217", | ||||
|   "results": { | ||||
|     "吕洞宾": { | ||||
|       "success": true, | ||||
|       "api_used": "alpha_vantage", | ||||
|       "response_time": 0.7236087322235107, | ||||
|       "cached": true, | ||||
|       "error": null, | ||||
|       "data_summary": { | ||||
|         "symbol": "TSLA", | ||||
|         "price": 302.63, | ||||
|         "change_percent": "-1.8296%" | ||||
|       } | ||||
|     }, | ||||
|     "何仙姑": { | ||||
|       "success": true, | ||||
|       "api_used": "yahoo_finance_15", | ||||
|       "response_time": 1.7378709316253662, | ||||
|       "cached": false, | ||||
|       "error": null, | ||||
|       "data_summary": { | ||||
|         "symbol": null, | ||||
|         "price": null, | ||||
|         "change_percent": null | ||||
|       } | ||||
|     }, | ||||
|     "张果老": { | ||||
|       "success": true, | ||||
|       "api_used": "webull", | ||||
|       "response_time": 2.667297601699829, | ||||
|       "cached": false, | ||||
|       "error": null, | ||||
|       "data_summary": { | ||||
|         "symbol": null, | ||||
|         "price": null, | ||||
|         "change_percent": null | ||||
|       } | ||||
|     }, | ||||
|     "韩湘子": { | ||||
|       "success": true, | ||||
|       "api_used": "webull", | ||||
|       "response_time": 1.9658794403076172, | ||||
|       "cached": false, | ||||
|       "error": null, | ||||
|       "data_summary": { | ||||
|         "symbol": null, | ||||
|         "price": null, | ||||
|         "change_percent": null | ||||
|       } | ||||
|     }, | ||||
|     "汉钟离": { | ||||
|       "success": true, | ||||
|       "api_used": "yahoo_finance_15", | ||||
|       "response_time": 3.024261951446533, | ||||
|       "cached": false, | ||||
|       "error": null, | ||||
|       "data_summary": { | ||||
|         "symbol": null, | ||||
|         "price": null, | ||||
|         "change_percent": null | ||||
|       } | ||||
|     }, | ||||
|     "蓝采和": { | ||||
|       "success": true, | ||||
|       "api_used": "webull", | ||||
|       "response_time": 1.5434284210205078, | ||||
|       "cached": false, | ||||
|       "error": null, | ||||
|       "data_summary": { | ||||
|         "symbol": null, | ||||
|         "price": null, | ||||
|         "change_percent": null | ||||
|       } | ||||
|     }, | ||||
|     "曹国舅": { | ||||
|       "success": true, | ||||
|       "api_used": "alpha_vantage", | ||||
|       "response_time": 1.1568174362182617, | ||||
|       "cached": false, | ||||
|       "error": null, | ||||
|       "data_summary": { | ||||
|         "symbol": "TSLA", | ||||
|         "price": 302.63, | ||||
|         "change_percent": "-1.8296%" | ||||
|       } | ||||
|     }, | ||||
|     "铁拐李": { | ||||
|       "success": true, | ||||
|       "api_used": "alpha_vantage", | ||||
|       "response_time": 1.3348329067230225, | ||||
|       "cached": false, | ||||
|       "error": null, | ||||
|       "data_summary": { | ||||
|         "symbol": "TSLA", | ||||
|         "price": 302.63, | ||||
|         "change_percent": "-1.8296%" | ||||
|       } | ||||
|     } | ||||
|   }, | ||||
|   "summary": { | ||||
|     "total_immortals": 8, | ||||
|     "successful_calls": 8, | ||||
|     "failed_calls": 0 | ||||
|   } | ||||
| } | ||||
|  | @ -0,0 +1,283 @@ | |||
| #!/usr/bin/env python3 | ||||
| # -*- coding: utf-8 -*- | ||||
| """ | ||||
| 稷下学宫负载均衡演示脚本 | ||||
| 展示八仙论道的API负载分担策略 | ||||
| """ | ||||
| 
 | ||||
| import os | ||||
| import sys | ||||
| import time | ||||
| import json | ||||
| from datetime import datetime | ||||
| 
 | ||||
| # 添加项目路径 | ||||
| sys.path.append('/home/ben/liurenchaxin/src') | ||||
| 
 | ||||
| from jixia.engines.jixia_load_balancer import JixiaLoadBalancer, APIResult | ||||
| 
 | ||||
| def print_banner(): | ||||
|     """打印横幅""" | ||||
|     print("\n" + "="*80) | ||||
|     print("🏛️  稷下学宫八仙论道 - API负载均衡演示系统") | ||||
|     print("📊  RapidAPI多源数据整合与负载分担策略") | ||||
|     print("="*80) | ||||
|     print(f"⏰ 演示时间: {datetime.now().strftime('%Y-%m-%d %H:%M:%S')}") | ||||
|     print() | ||||
| 
 | ||||
| def print_immortal_intro(): | ||||
|     """介绍八仙角色""" | ||||
|     immortals_info = { | ||||
|         '吕洞宾': '主力剑仙 - 综合分析与决策 (Alpha Vantage)', | ||||
|         '何仙姑': '风控专家 - 风险管理与合规 (Yahoo Finance)', | ||||
|         '张果老': '技术分析师 - 技术指标与图表分析 (Webull)', | ||||
|         '韩湘子': '基本面研究员 - 财务分析与估值 (Seeking Alpha)', | ||||
|         '汉钟离': '量化专家 - 数据挖掘与算法交易 (Yahoo Finance)', | ||||
|         '蓝采和': '情绪分析师 - 市场情绪与舆情监控 (Webull)', | ||||
|         '曹国舅': '宏观分析师 - 宏观经济与政策分析 (Seeking Alpha)', | ||||
|         '铁拐李': '逆向投资专家 - 价值发现与逆向思维 (Alpha Vantage)' | ||||
|     } | ||||
|      | ||||
|     print("👥 八仙角色与API分配:") | ||||
|     print("-" * 60) | ||||
|     for immortal, description in immortals_info.items(): | ||||
|         print(f"   {immortal}: {description}") | ||||
|     print() | ||||
| 
 | ||||
| def demonstrate_single_immortal(load_balancer, immortal_name, symbol): | ||||
|     """演示单个仙人的数据获取""" | ||||
|     print(f"\n🎭 {immortal_name} 单独获取数据演示:") | ||||
|     print("-" * 40) | ||||
|      | ||||
|     # 获取股票报价 | ||||
|     result = load_balancer.get_data_for_immortal(immortal_name, 'stock_quote', symbol) | ||||
|      | ||||
|     if result.success: | ||||
|         data = result.data | ||||
|         print(f"   ✅ 成功获取 {symbol} 数据:") | ||||
|         print(f"      💰 价格: ${data.get('price', 'N/A')}") | ||||
|         print(f"      📈 涨跌: {data.get('change_percent', 'N/A')}") | ||||
|         volume = data.get('volume', 'N/A') | ||||
|         if isinstance(volume, (int, float)): | ||||
|             print(f"      📊 成交量: {volume:,}") | ||||
|         else: | ||||
|             print(f"      📊 成交量: {volume}") | ||||
|         print(f"      🔗 数据源: {result.api_used}") | ||||
|         print(f"      ⏱️ 响应时间: {result.response_time:.2f}秒") | ||||
|         print(f"      💾 缓存状态: {'是' if result.cached else '否'}") | ||||
|     else: | ||||
|         print(f"   ❌ 获取失败: {result.error}") | ||||
| 
 | ||||
| def demonstrate_load_distribution(load_balancer): | ||||
|     """演示负载分布""" | ||||
|     print("\n📊 API负载分布统计:") | ||||
|     print("-" * 40) | ||||
|      | ||||
|     distribution = load_balancer.get_load_distribution() | ||||
|      | ||||
|     if not distribution: | ||||
|         print("   📝 暂无API调用记录") | ||||
|         return | ||||
|      | ||||
|     total_calls = sum(stats['calls'] for stats in distribution.values()) | ||||
|      | ||||
|     for api_name, stats in distribution.items(): | ||||
|         status_icon = "🟢" if stats['healthy'] else "🔴" | ||||
|         print(f"   {status_icon} {api_name}:") | ||||
|         print(f"      📞 调用次数: {stats['calls']}") | ||||
|         print(f"      📊 负载占比: {stats['percentage']:.1f}%") | ||||
|         print(f"      ❌ 连续失败: {stats['consecutive_failures']}次") | ||||
|         print() | ||||
| 
 | ||||
| def demonstrate_api_comparison(load_balancer, symbol): | ||||
|     """演示不同API的数据对比""" | ||||
|     print(f"\n🔍 {symbol} 多API数据对比:") | ||||
|     print("-" * 50) | ||||
|      | ||||
|     apis = ['alpha_vantage', 'yahoo_finance_15', 'webull'] | ||||
|     results = {} | ||||
|      | ||||
|     for api in apis: | ||||
|         # 临时修改API分配来测试不同数据源 | ||||
|         original_mapping = load_balancer.immortal_api_mapping['stock_quote']['吕洞宾'] | ||||
|         load_balancer.immortal_api_mapping['stock_quote']['吕洞宾'] = api | ||||
|          | ||||
|         result = load_balancer.get_data_for_immortal('吕洞宾', 'stock_quote', symbol) | ||||
|         results[api] = result | ||||
|          | ||||
|         # 恢复原始配置 | ||||
|         load_balancer.immortal_api_mapping['stock_quote']['吕洞宾'] = original_mapping | ||||
|          | ||||
|         time.sleep(0.5)  # 避免过快请求 | ||||
|      | ||||
|     # 显示对比结果 | ||||
|     print("   API数据源对比:") | ||||
|     for api, result in results.items(): | ||||
|         if result.success: | ||||
|             data = result.data | ||||
|             print(f"   📡 {api}:") | ||||
|             print(f"      💰 ${data.get('price', 'N/A')} ({data.get('change_percent', 'N/A')})") | ||||
|             print(f"      ⏱️ {result.response_time:.2f}s") | ||||
|         else: | ||||
|             print(f"   📡 {api}: ❌ {result.error}") | ||||
|         print() | ||||
| 
 | ||||
| def demonstrate_cache_effectiveness(load_balancer, symbol): | ||||
|     """演示缓存效果""" | ||||
|     print(f"\n💾 缓存效果演示 - {symbol}:") | ||||
|     print("-" * 40) | ||||
|      | ||||
|     # 第一次调用(无缓存) | ||||
|     print("   🔄 第一次调用(无缓存):") | ||||
|     start_time = time.time() | ||||
|     result1 = load_balancer.get_data_for_immortal('吕洞宾', 'stock_quote', symbol) | ||||
|     first_call_time = time.time() - start_time | ||||
|      | ||||
|     if result1.success: | ||||
|         print(f"      ⏱️ 响应时间: {result1.response_time:.2f}秒") | ||||
|         print(f"      💾 缓存状态: {'命中' if result1.cached else '未命中'}") | ||||
|      | ||||
|     time.sleep(1) | ||||
|      | ||||
|     # 第二次调用(有缓存) | ||||
|     print("   🔄 第二次调用(有缓存):") | ||||
|     start_time = time.time() | ||||
|     result2 = load_balancer.get_data_for_immortal('吕洞宾', 'stock_quote', symbol) | ||||
|     second_call_time = time.time() - start_time | ||||
|      | ||||
|     if result2.success: | ||||
|         print(f"      ⏱️ 响应时间: {result2.response_time:.2f}秒") | ||||
|         print(f"      💾 缓存状态: {'命中' if result2.cached else '未命中'}") | ||||
|          | ||||
|         if result2.cached: | ||||
|             speedup = (first_call_time / second_call_time) if second_call_time > 0 else float('inf') | ||||
|             print(f"      🚀 性能提升: {speedup:.1f}倍") | ||||
| 
 | ||||
| def demonstrate_failover(load_balancer, symbol): | ||||
|     """演示故障转移""" | ||||
|     print(f"\n🔄 故障转移演示 - {symbol}:") | ||||
|     print("-" * 40) | ||||
|      | ||||
|     # 模拟API故障 | ||||
|     print("   ⚠️ 模拟主API故障...") | ||||
|      | ||||
|     # 临时标记API为不健康 | ||||
|     original_health = load_balancer.health_checker.health_status['alpha_vantage']['healthy'] | ||||
|     load_balancer.health_checker.health_status['alpha_vantage']['healthy'] = False | ||||
|     load_balancer.health_checker.health_status['alpha_vantage']['consecutive_failures'] = 5 | ||||
|      | ||||
|     # 尝试获取数据(应该自动故障转移) | ||||
|     result = load_balancer.get_data_for_immortal('吕洞宾', 'stock_quote', symbol) | ||||
|      | ||||
|     if result.success: | ||||
|         print(f"   ✅ 故障转移成功,使用备用API: {result.api_used}") | ||||
|         print(f"   💰 获取到价格: ${result.data.get('price', 'N/A')}") | ||||
|     else: | ||||
|         print(f"   ❌ 故障转移失败: {result.error}") | ||||
|      | ||||
|     # 恢复API健康状态 | ||||
|     load_balancer.health_checker.health_status['alpha_vantage']['healthy'] = original_health | ||||
|     load_balancer.health_checker.health_status['alpha_vantage']['consecutive_failures'] = 0 | ||||
|      | ||||
|     print("   🔧 API健康状态已恢复") | ||||
| 
 | ||||
| def save_demo_results(results, filename='demo_results.json'): | ||||
|     """保存演示结果""" | ||||
|     demo_data = { | ||||
|         'timestamp': datetime.now().isoformat(), | ||||
|         'results': {}, | ||||
|         'summary': { | ||||
|             'total_immortals': len(results), | ||||
|             'successful_calls': sum(1 for r in results.values() if r.success), | ||||
|             'failed_calls': sum(1 for r in results.values() if not r.success) | ||||
|         } | ||||
|     } | ||||
|      | ||||
|     for immortal, result in results.items(): | ||||
|         demo_data['results'][immortal] = { | ||||
|             'success': result.success, | ||||
|             'api_used': result.api_used, | ||||
|             'response_time': result.response_time, | ||||
|             'cached': result.cached, | ||||
|             'error': result.error, | ||||
|             'data_summary': { | ||||
|                 'symbol': result.data.get('symbol') if result.success else None, | ||||
|                 'price': result.data.get('price') if result.success else None, | ||||
|                 'change_percent': result.data.get('change_percent') if result.success else None | ||||
|             } | ||||
|         } | ||||
|      | ||||
|     with open(filename, 'w', encoding='utf-8') as f: | ||||
|         json.dump(demo_data, f, indent=2, ensure_ascii=False) | ||||
|      | ||||
|     print(f"\n💾 演示结果已保存到: {filename}") | ||||
| 
 | ||||
| def main(): | ||||
|     """主演示函数""" | ||||
|     # 检查API密钥 | ||||
|     rapidapi_key = os.getenv('RAPIDAPI_KEY') | ||||
|     if not rapidapi_key: | ||||
|         print("❌ 错误: 请设置RAPIDAPI_KEY环境变量") | ||||
|         print("   提示: 使用 'doppler run python demo_jixia_load_balancing.py' 运行") | ||||
|         return | ||||
|      | ||||
|     print_banner() | ||||
|     print_immortal_intro() | ||||
|      | ||||
|     # 创建负载均衡器 | ||||
|     print("🔧 初始化稷下学宫负载均衡器...") | ||||
|     load_balancer = JixiaLoadBalancer(rapidapi_key) | ||||
|     print("✅ 负载均衡器初始化完成\n") | ||||
|      | ||||
|     # 演示股票代码 | ||||
|     demo_symbols = ['AAPL', 'TSLA', 'MSFT'] | ||||
|      | ||||
|     for i, symbol in enumerate(demo_symbols, 1): | ||||
|         print(f"\n{'='*20} 演示 {i}: {symbol} {'='*20}") | ||||
|          | ||||
|         # 1. 单个仙人演示 | ||||
|         demonstrate_single_immortal(load_balancer, '吕洞宾', symbol) | ||||
|          | ||||
|         # 2. 八仙论道演示 | ||||
|         print(f"\n🏛️ 八仙论道完整演示 - {symbol}:") | ||||
|         debate_results = load_balancer.conduct_immortal_debate(symbol) | ||||
|          | ||||
|         # 3. 负载分布演示 | ||||
|         demonstrate_load_distribution(load_balancer) | ||||
|          | ||||
|         # 只在第一个股票上演示高级功能 | ||||
|         if i == 1: | ||||
|             # 4. API对比演示 | ||||
|             demonstrate_api_comparison(load_balancer, symbol) | ||||
|              | ||||
|             # 5. 缓存效果演示 | ||||
|             demonstrate_cache_effectiveness(load_balancer, symbol) | ||||
|              | ||||
|             # 6. 故障转移演示 | ||||
|             demonstrate_failover(load_balancer, symbol) | ||||
|          | ||||
|         # 保存结果 | ||||
|         save_demo_results(debate_results, f'demo_results_{symbol.lower()}.json') | ||||
|          | ||||
|         if i < len(demo_symbols): | ||||
|             print("\n⏳ 等待3秒后继续下一个演示...") | ||||
|             time.sleep(3) | ||||
|      | ||||
|     # 最终统计 | ||||
|     print("\n" + "="*80) | ||||
|     print("📈 演示完成 - 最终负载分布统计:") | ||||
|     demonstrate_load_distribution(load_balancer) | ||||
|      | ||||
|     print("\n🎉 稷下学宫API负载均衡演示完成!") | ||||
|     print("\n💡 关键特性:") | ||||
|     print("   ✅ 智能负载分担 - 八仙各司其职,分散API压力") | ||||
|     print("   ✅ 自动故障转移 - API异常时自动切换备用源") | ||||
|     print("   ✅ 数据标准化 - 统一不同API的数据格式") | ||||
|     print("   ✅ 智能缓存 - 减少重复调用,提升响应速度") | ||||
|     print("   ✅ 实时监控 - 跟踪API健康状态和负载分布") | ||||
|     print("\n📚 查看详细配置: /home/ben/liurenchaxin/src/jixia/config/immortal_api_config.json") | ||||
|     print("🔧 核心引擎: /home/ben/liurenchaxin/src/jixia/engines/jixia_load_balancer.py") | ||||
|     print("="*80) | ||||
| 
 | ||||
| if __name__ == "__main__": | ||||
|     main() | ||||
|  | @ -0,0 +1,245 @@ | |||
| #!/usr/bin/env python3 | ||||
| """ | ||||
| Vertex AI Memory Bank 演示脚本 | ||||
| 展示稷下学宫记忆增强AI辩论系统 | ||||
| """ | ||||
| 
 | ||||
| import asyncio | ||||
| import sys | ||||
| import os | ||||
| 
 | ||||
| # 添加项目根目录到路径 | ||||
| sys.path.append(os.path.dirname(os.path.dirname(os.path.abspath(__file__)))) | ||||
| 
 | ||||
| from src.jixia.agents.memory_enhanced_agent import create_memory_enhanced_council | ||||
| from config.doppler_config import validate_config | ||||
| 
 | ||||
| 
 | ||||
| async def demo_memory_enhanced_debate(): | ||||
|     """演示记忆增强的AI辩论""" | ||||
|      | ||||
|     print("🏛️ 稷下学宫 Vertex AI Memory Bank 演示") | ||||
|     print("=" * 60) | ||||
|      | ||||
|     # 验证配置 | ||||
|     print("🔧 验证配置...") | ||||
|     if not validate_config("google_adk"): | ||||
|         print("❌ 配置验证失败,请检查环境变量") | ||||
|         return | ||||
|      | ||||
|     try: | ||||
|         # 创建记忆增强议会 | ||||
|         print("\n🎭 创建八仙记忆增强议会...") | ||||
|         council = await create_memory_enhanced_council() | ||||
|          | ||||
|         # 演示主题 | ||||
|         topics = [ | ||||
|             "特斯拉股票投资价值分析", | ||||
|             "人工智能行业投资机会", | ||||
|             "加密货币市场前景展望" | ||||
|         ] | ||||
|          | ||||
|         # 选择参与的仙人(为了演示,只选择几位) | ||||
|         participants = ["tieguaili", "lvdongbin", "hexiangu", "zhangguolao"] | ||||
|          | ||||
|         for i, topic in enumerate(topics, 1): | ||||
|             print(f"\n{'='*40}") | ||||
|             print(f"🎯 第 {i} 场辩论: {topic}") | ||||
|             print(f"{'='*40}") | ||||
|              | ||||
|             # 进行记忆增强辩论 | ||||
|             result = await council.conduct_memory_debate( | ||||
|                 topic=topic, | ||||
|                 participants=participants, | ||||
|                 rounds=2  # 每场2轮,保持演示简洁 | ||||
|             ) | ||||
|              | ||||
|             print(f"\n📊 辩论结果:") | ||||
|             print(f"   主题: {result['topic']}") | ||||
|             print(f"   参与者: {len(result['participants'])} 位仙人") | ||||
|             print(f"   总发言: {result['total_exchanges']} 次") | ||||
|              | ||||
|             # 显示部分对话内容 | ||||
|             print(f"\n💬 精彩观点摘录:") | ||||
|             for exchange in result['conversation_history'][:4]:  # 只显示前4条 | ||||
|                 content_preview = exchange['content'][:120] + "..." if len(exchange['content']) > 120 else exchange['content'] | ||||
|                 print(f"   🗣️ {exchange['chinese_name']}: {content_preview}") | ||||
|              | ||||
|             # 获取集体记忆摘要 | ||||
|             print(f"\n📚 获取集体记忆...") | ||||
|             summary = await council.get_collective_memory_summary(topic) | ||||
|              | ||||
|             if "暂无相关集体记忆" not in summary: | ||||
|                 print(f"   ✅ 已生成 {len(summary)} 字符的记忆摘要") | ||||
|             else: | ||||
|                 print(f"   ℹ️ 这是新主题,正在建立记忆") | ||||
|              | ||||
|             # 演示间隔 | ||||
|             if i < len(topics): | ||||
|                 print(f"\n⏳ 准备下一场辩论...") | ||||
|                 await asyncio.sleep(1) | ||||
|          | ||||
|         # 最终演示:展示记忆的累积效果 | ||||
|         print(f"\n{'='*60}") | ||||
|         print("🧠 记忆累积效果演示") | ||||
|         print(f"{'='*60}") | ||||
|          | ||||
|         # 让铁拐李基于所有记忆回答一个综合问题 | ||||
|         tieguaili = council.agents.get("tieguaili") | ||||
|         if tieguaili: | ||||
|             print(f"\n🤔 向铁拐李提问: '基于你的所有记忆,总结一下当前市场的主要风险'") | ||||
|              | ||||
|             comprehensive_response = await tieguaili.respond_with_memory( | ||||
|                 message="基于你参与的所有辩论和积累的记忆,总结一下当前市场的主要风险和你的投资建议。", | ||||
|                 topic="综合市场分析" | ||||
|             ) | ||||
|              | ||||
|             print(f"\n🧙♂️ 铁拐李的综合分析:") | ||||
|             print(f"   {comprehensive_response}") | ||||
|          | ||||
|         # 展示记忆学习功能 | ||||
|         print(f"\n🎓 演示记忆学习功能...") | ||||
|          | ||||
|         # 让何仙姑学习一个用户偏好 | ||||
|         hexiangu = council.agents.get("hexiangu") | ||||
|         if hexiangu: | ||||
|             await hexiangu.learn_preference( | ||||
|                 preference="用户偏好ESG投资,关注环境和社会责任", | ||||
|                 topic="投资偏好" | ||||
|             ) | ||||
|             print(f"   ✅ 何仙姑学习了ESG投资偏好") | ||||
|              | ||||
|             # 基于新学到的偏好回答问题 | ||||
|             esg_response = await hexiangu.respond_with_memory( | ||||
|                 message="推荐一些符合ESG标准的投资标的", | ||||
|                 topic="ESG投资" | ||||
|             ) | ||||
|              | ||||
|             print(f"\n👸 何仙姑基于学习的偏好回应:") | ||||
|             print(f"   {esg_response[:200]}...") | ||||
|          | ||||
|         print(f"\n🎉 演示完成!") | ||||
|         print(f"\n💡 Memory Bank 的优势:") | ||||
|         print(f"   ✅ 智能体能记住历史对话和分析") | ||||
|         print(f"   ✅ 学习用户偏好,提供个性化建议") | ||||
|         print(f"   ✅ 积累投资策略和市场洞察") | ||||
|         print(f"   ✅ 跨会话保持一致的人格和观点") | ||||
|         print(f"   ✅ 基于历史经验做出更好的决策") | ||||
|          | ||||
|     except Exception as e: | ||||
|         print(f"❌ 演示过程中出现错误: {e}") | ||||
|         print(f"💡 请检查:") | ||||
|         print(f"   - Google Cloud Project ID 是否正确配置") | ||||
|         print(f"   - Vertex AI API 是否已启用") | ||||
|         print(f"   - 网络连接是否正常") | ||||
| 
 | ||||
| 
 | ||||
| async def demo_individual_memory_features(): | ||||
|     """演示个体记忆功能""" | ||||
|      | ||||
|     print(f"\n{'='*60}") | ||||
|     print("🔍 个体记忆功能详细演示") | ||||
|     print(f"{'='*60}") | ||||
|      | ||||
|     try: | ||||
|         from src.jixia.memory.vertex_memory_bank import VertexMemoryBank | ||||
|         from src.jixia.agents.memory_enhanced_agent import MemoryEnhancedAgent | ||||
|          | ||||
|         # 创建记忆银行 | ||||
|         memory_bank = VertexMemoryBank.from_config() | ||||
|          | ||||
|         # 创建单个智能体进行详细演示 | ||||
|         agent = MemoryEnhancedAgent("tieguaili", memory_bank) | ||||
|          | ||||
|         print(f"\n🧙♂️ 与 {agent.personality.chinese_name} 的记忆互动演示") | ||||
|          | ||||
|         # 1. 添加不同类型的记忆 | ||||
|         print(f"\n📝 添加不同类型的记忆...") | ||||
|          | ||||
|         memories_to_add = [ | ||||
|             { | ||||
|                 "content": "在2008年金融危机中,逆向投资者获得了丰厚回报", | ||||
|                 "memory_type": "knowledge", | ||||
|                 "topic": "历史教训" | ||||
|             }, | ||||
|             { | ||||
|                 "content": "用户偏好价值投资,不喜欢高风险的成长股", | ||||
|                 "memory_type": "preference",  | ||||
|                 "topic": "用户偏好" | ||||
|             }, | ||||
|             { | ||||
|                 "content": "当市场过度乐观时,应该保持谨慎并寻找反向机会", | ||||
|                 "memory_type": "strategy", | ||||
|                 "topic": "投资策略" | ||||
|             } | ||||
|         ] | ||||
|          | ||||
|         for memory in memories_to_add: | ||||
|             await memory_bank.add_memory( | ||||
|                 agent_name="tieguaili", | ||||
|                 content=memory["content"], | ||||
|                 memory_type=memory["memory_type"], | ||||
|                 debate_topic=memory["topic"] | ||||
|             ) | ||||
|             print(f"   ✅ 添加{memory['memory_type']}记忆: {memory['content'][:50]}...") | ||||
|          | ||||
|         # 2. 搜索记忆 | ||||
|         print(f"\n🔍 搜索相关记忆...") | ||||
|          | ||||
|         search_queries = ["金融危机", "价值投资", "投资策略"] | ||||
|          | ||||
|         for query in search_queries: | ||||
|             results = await memory_bank.search_memories( | ||||
|                 agent_name="tieguaili", | ||||
|                 query=query, | ||||
|                 limit=3 | ||||
|             ) | ||||
|             print(f"   🔎 搜索 '{query}': 找到 {len(results)} 条相关记忆") | ||||
|              | ||||
|             for result in results: | ||||
|                 relevance = result.get('relevance_score', 'N/A') | ||||
|                 print(f"      - {result['content'][:60]}... (相关度: {relevance})") | ||||
|          | ||||
|         # 3. 基于记忆的智能回应 | ||||
|         print(f"\n🤖 基于记忆的智能回应演示...") | ||||
|          | ||||
|         questions = [ | ||||
|             "现在市场很乐观,你有什么建议?", | ||||
|             "推荐一些适合保守投资者的标的", | ||||
|             "历史上有哪些值得借鉴的投资教训?" | ||||
|         ] | ||||
|          | ||||
|         for question in questions: | ||||
|             print(f"\n❓ 问题: {question}") | ||||
|              | ||||
|             response = await agent.respond_with_memory( | ||||
|                 message=question, | ||||
|                 topic="投资咨询" | ||||
|             ) | ||||
|              | ||||
|             print(f"🧙♂️ 铁拐李: {response[:150]}...") | ||||
|          | ||||
|         print(f"\n✨ 个体记忆功能演示完成!") | ||||
|          | ||||
|     except Exception as e: | ||||
|         print(f"❌ 个体记忆演示失败: {e}") | ||||
| 
 | ||||
| 
 | ||||
| async def main(): | ||||
|     """主演示函数""" | ||||
|      | ||||
|     print("🚀 启动 Vertex AI Memory Bank 完整演示") | ||||
|      | ||||
|     # 主要演示:记忆增强辩论 | ||||
|     await demo_memory_enhanced_debate() | ||||
|      | ||||
|     # 详细演示:个体记忆功能 | ||||
|     await demo_individual_memory_features() | ||||
|      | ||||
|     print(f"\n🏛️ 稷下学宫 Memory Bank 演示结束") | ||||
|     print(f"📖 更多信息请参考: docs/VERTEX_MEMORY_BANK_SETUP.md") | ||||
| 
 | ||||
| 
 | ||||
| if __name__ == "__main__": | ||||
|     # 运行演示 | ||||
|     asyncio.run(main()) | ||||
|  | @ -0,0 +1,275 @@ | |||
| #!/usr/bin/env python3 | ||||
| # -*- coding: utf-8 -*- | ||||
| """ | ||||
| Memory Bank 实验脚本 | ||||
| 测试八仙人格的长期记忆功能 | ||||
| """ | ||||
| 
 | ||||
| import os | ||||
| import asyncio | ||||
| from datetime import datetime | ||||
| from typing import Dict, List, Any | ||||
| import json | ||||
| 
 | ||||
| # Google GenAI 导入 | ||||
| try: | ||||
|     import google.genai as genai | ||||
|     from google.genai import types | ||||
| except ImportError: | ||||
|     print("❌ 请安装 google-genai: pip install google-genai") | ||||
|     exit(1) | ||||
| 
 | ||||
| class MemoryBankExperiment: | ||||
|     """Memory Bank 实验类""" | ||||
|      | ||||
|     def __init__(self): | ||||
|         self.api_key = os.getenv('GOOGLE_API_KEY') | ||||
|         if not self.api_key: | ||||
|             raise ValueError("请设置 GOOGLE_API_KEY 环境变量") | ||||
|          | ||||
|         # 初始化 GenAI | ||||
|         genai.configure(api_key=self.api_key) | ||||
|          | ||||
|         # 八仙人格基线 | ||||
|         self.immortal_baselines = { | ||||
|             "吕洞宾": { | ||||
|                 "mbti_type": "ENTJ", | ||||
|                 "core_traits": { | ||||
|                     "assertiveness": 0.9, | ||||
|                     "analytical": 0.8, | ||||
|                     "risk_tolerance": 0.8, | ||||
|                     "optimism": 0.7 | ||||
|                 }, | ||||
|                 "personality_description": "剑仙投资顾问,主动进取,敢于冒险,技术分析专家" | ||||
|             }, | ||||
|             "何仙姑": { | ||||
|                 "mbti_type": "ISFJ",  | ||||
|                 "core_traits": { | ||||
|                     "empathy": 0.9, | ||||
|                     "caution": 0.8, | ||||
|                     "loyalty": 0.8, | ||||
|                     "optimism": 0.4 | ||||
|                 }, | ||||
|                 "personality_description": "慈悲风控专家,谨慎小心,保护意识强,风险厌恶" | ||||
|             }, | ||||
|             "张果老": { | ||||
|                 "mbti_type": "INTP", | ||||
|                 "core_traits": { | ||||
|                     "analytical": 0.9, | ||||
|                     "curiosity": 0.8, | ||||
|                     "traditional": 0.7, | ||||
|                     "caution": 0.6 | ||||
|                 }, | ||||
|                 "personality_description": "历史数据分析师,深度思考,逆向思维,传统智慧" | ||||
|             } | ||||
|         } | ||||
|          | ||||
|         # 记忆存储(模拟 Memory Bank) | ||||
|         self.memory_bank = {} | ||||
|          | ||||
|     def initialize_immortal_memory(self, immortal_name: str): | ||||
|         """初始化仙人的记忆空间""" | ||||
|         if immortal_name not in self.memory_bank: | ||||
|             self.memory_bank[immortal_name] = { | ||||
|                 "personality_baseline": self.immortal_baselines[immortal_name], | ||||
|                 "conversation_history": [], | ||||
|                 "viewpoint_evolution": [], | ||||
|                 "decision_history": [], | ||||
|                 "created_at": datetime.now().isoformat(), | ||||
|                 "last_updated": datetime.now().isoformat() | ||||
|             } | ||||
|             print(f"🎭 初始化 {immortal_name} 的记忆空间") | ||||
|      | ||||
|     def store_memory(self, immortal_name: str, memory_type: str, content: Dict[str, Any]): | ||||
|         """存储记忆到 Memory Bank""" | ||||
|         self.initialize_immortal_memory(immortal_name) | ||||
|          | ||||
|         memory_entry = { | ||||
|             "type": memory_type, | ||||
|             "content": content, | ||||
|             "timestamp": datetime.now().isoformat(), | ||||
|             "session_id": f"session_{len(self.memory_bank[immortal_name]['conversation_history'])}" | ||||
|         } | ||||
|          | ||||
|         if memory_type == "conversation": | ||||
|             self.memory_bank[immortal_name]["conversation_history"].append(memory_entry) | ||||
|         elif memory_type == "viewpoint": | ||||
|             self.memory_bank[immortal_name]["viewpoint_evolution"].append(memory_entry) | ||||
|         elif memory_type == "decision": | ||||
|             self.memory_bank[immortal_name]["decision_history"].append(memory_entry) | ||||
|          | ||||
|         self.memory_bank[immortal_name]["last_updated"] = datetime.now().isoformat() | ||||
|         print(f"💾 {immortal_name} 存储了 {memory_type} 记忆") | ||||
|      | ||||
|     def retrieve_relevant_memories(self, immortal_name: str, query: str) -> List[Dict]: | ||||
|         """检索相关记忆""" | ||||
|         if immortal_name not in self.memory_bank: | ||||
|             return [] | ||||
|          | ||||
|         # 简单的关键词匹配(实际应该使用向量相似度搜索) | ||||
|         relevant_memories = [] | ||||
|         query_lower = query.lower() | ||||
|          | ||||
|         for memory in self.memory_bank[immortal_name]["conversation_history"]: | ||||
|             if any(keyword in memory["content"].get("message", "").lower()  | ||||
|                    for keyword in query_lower.split()): | ||||
|                 relevant_memories.append(memory) | ||||
|          | ||||
|         return relevant_memories[-5:]  # 返回最近5条相关记忆 | ||||
|      | ||||
|     async def generate_immortal_response(self, immortal_name: str, query: str) -> str: | ||||
|         """生成仙人的回应,基于记忆和人格基线""" | ||||
|         # 检索相关记忆 | ||||
|         relevant_memories = self.retrieve_relevant_memories(immortal_name, query) | ||||
|          | ||||
|         # 构建上下文 | ||||
|         context = self.build_context(immortal_name, relevant_memories) | ||||
|          | ||||
|         # 生成回应 | ||||
|         model = genai.GenerativeModel('gemini-2.0-flash-exp') | ||||
|          | ||||
|         prompt = f""" | ||||
|         你是{immortal_name},{self.immortal_baselines[immortal_name]['personality_description']}。 | ||||
|          | ||||
|         你的核心人格特质: | ||||
|         {json.dumps(self.immortal_baselines[immortal_name]['core_traits'], ensure_ascii=False, indent=2)} | ||||
|          | ||||
|         你的相关记忆: | ||||
|         {json.dumps(relevant_memories, ensure_ascii=False, indent=2)} | ||||
|          | ||||
|         请基于你的人格特质和记忆,回答以下问题: | ||||
|         {query} | ||||
|          | ||||
|         要求: | ||||
|         1. 保持人格一致性 | ||||
|         2. 参考历史记忆 | ||||
|         3. 回答控制在100字以内 | ||||
|         4. 体现你的独特风格 | ||||
|         """ | ||||
|          | ||||
|         response = await model.generate_content_async(prompt) | ||||
|         return response.text | ||||
|      | ||||
|     def build_context(self, immortal_name: str, memories: List[Dict]) -> str: | ||||
|         """构建上下文信息""" | ||||
|         context_parts = [] | ||||
|          | ||||
|         # 添加人格基线 | ||||
|         baseline = self.immortal_baselines[immortal_name] | ||||
|         context_parts.append(f"人格类型: {baseline['mbti_type']}") | ||||
|         context_parts.append(f"核心特质: {json.dumps(baseline['core_traits'], ensure_ascii=False)}") | ||||
|          | ||||
|         # 添加相关记忆 | ||||
|         if memories: | ||||
|             context_parts.append("相关记忆:") | ||||
|             for memory in memories[-3:]:  # 最近3条记忆 | ||||
|                 context_parts.append(f"- {memory['content'].get('message', '')}") | ||||
|          | ||||
|         return "\n".join(context_parts) | ||||
|      | ||||
|     def simulate_conversation(self, immortal_name: str, messages: List[str]): | ||||
|         """模拟对话,测试记忆功能""" | ||||
|         print(f"\n🎭 开始与 {immortal_name} 的对话") | ||||
|         print("=" * 50) | ||||
|          | ||||
|         for i, message in enumerate(messages): | ||||
|             print(f"\n用户: {message}") | ||||
|              | ||||
|             # 生成回应 | ||||
|             response = asyncio.run(self.generate_immortal_response(immortal_name, message)) | ||||
|             print(f"{immortal_name}: {response}") | ||||
|              | ||||
|             # 存储记忆 | ||||
|             self.store_memory(immortal_name, "conversation", { | ||||
|                 "user_message": message, | ||||
|                 "immortal_response": response, | ||||
|                 "session_id": f"session_{i}" | ||||
|             }) | ||||
|              | ||||
|             # 存储观点 | ||||
|             if "看多" in response or "看空" in response or "观望" in response: | ||||
|                 viewpoint = "看多" if "看多" in response else "看空" if "看空" in response else "观望" | ||||
|                 self.store_memory(immortal_name, "viewpoint", { | ||||
|                     "symbol": "TSLA",  # 假设讨论特斯拉 | ||||
|                     "viewpoint": viewpoint, | ||||
|                     "reasoning": response | ||||
|                 }) | ||||
|      | ||||
|     def analyze_memory_evolution(self, immortal_name: str): | ||||
|         """分析记忆演化""" | ||||
|         if immortal_name not in self.memory_bank: | ||||
|             print(f"❌ {immortal_name} 没有记忆数据") | ||||
|             return | ||||
|          | ||||
|         memory_data = self.memory_bank[immortal_name] | ||||
|          | ||||
|         print(f"\n📊 {immortal_name} 记忆分析") | ||||
|         print("=" * 50) | ||||
|         print(f"记忆空间创建时间: {memory_data['created_at']}") | ||||
|         print(f"最后更新时间: {memory_data['last_updated']}") | ||||
|         print(f"对话记录数: {len(memory_data['conversation_history'])}") | ||||
|         print(f"观点演化数: {len(memory_data['viewpoint_evolution'])}") | ||||
|         print(f"决策记录数: {len(memory_data['decision_history'])}") | ||||
|          | ||||
|         # 分析观点演化 | ||||
|         if memory_data['viewpoint_evolution']: | ||||
|             print(f"\n观点演化轨迹:") | ||||
|             for i, viewpoint in enumerate(memory_data['viewpoint_evolution']): | ||||
|                 print(f"  {i+1}. {viewpoint['content']['viewpoint']} - {viewpoint['timestamp']}") | ||||
|      | ||||
|     def save_memory_bank(self, filename: str = "memory_bank_backup.json"): | ||||
|         """保存记忆库到文件""" | ||||
|         with open(filename, 'w', encoding='utf-8') as f: | ||||
|             json.dump(self.memory_bank, f, ensure_ascii=False, indent=2) | ||||
|         print(f"💾 记忆库已保存到 {filename}") | ||||
|      | ||||
|     def load_memory_bank(self, filename: str = "memory_bank_backup.json"): | ||||
|         """从文件加载记忆库""" | ||||
|         try: | ||||
|             with open(filename, 'r', encoding='utf-8') as f: | ||||
|                 self.memory_bank = json.load(f) | ||||
|             print(f"📂 记忆库已从 {filename} 加载") | ||||
|         except FileNotFoundError: | ||||
|             print(f"⚠️ 文件 {filename} 不存在,使用空记忆库") | ||||
| 
 | ||||
| def main(): | ||||
|     """主实验函数""" | ||||
|     print("🚀 开始 Memory Bank 实验") | ||||
|     print("=" * 60) | ||||
|      | ||||
|     # 创建实验实例 | ||||
|     experiment = MemoryBankExperiment() | ||||
|      | ||||
|     # 测试对话场景 | ||||
|     test_scenarios = { | ||||
|         "吕洞宾": [ | ||||
|             "你觉得特斯拉股票怎么样?", | ||||
|             "现在市场波动很大,你怎么看?", | ||||
|             "你之前不是看好特斯拉吗?现在还是这个观点吗?" | ||||
|         ], | ||||
|         "何仙姑": [ | ||||
|             "特斯拉股票风险大吗?", | ||||
|             "现在适合投资吗?", | ||||
|             "你一直很谨慎,现在还是建议观望吗?" | ||||
|         ], | ||||
|         "张果老": [ | ||||
|             "从历史数据看,特斯拉表现如何?", | ||||
|             "现在的估值合理吗?", | ||||
|             "你之前分析过特斯拉的历史数据,现在有什么新发现?" | ||||
|         ] | ||||
|     } | ||||
|      | ||||
|     # 执行实验 | ||||
|     for immortal_name, messages in test_scenarios.items(): | ||||
|         experiment.simulate_conversation(immortal_name, messages) | ||||
|         experiment.analyze_memory_evolution(immortal_name) | ||||
|      | ||||
|     # 保存记忆库 | ||||
|     experiment.save_memory_bank() | ||||
|      | ||||
|     print("\n🎉 Memory Bank 实验完成!") | ||||
|     print("=" * 60) | ||||
| 
 | ||||
| if __name__ == "__main__": | ||||
|     main() | ||||
| 
 | ||||
|  | @ -0,0 +1,116 @@ | |||
| #!/usr/bin/env python3 | ||||
| # -*- coding: utf-8 -*- | ||||
| """ | ||||
| Memory Bank 简化测试脚本 | ||||
| """ | ||||
| 
 | ||||
| import os | ||||
| import asyncio | ||||
| from datetime import datetime | ||||
| import json | ||||
| 
 | ||||
| # Google GenAI 导入 | ||||
| import google.genai as genai | ||||
| 
 | ||||
| class MemoryBankTest: | ||||
|     """Memory Bank 测试类""" | ||||
|      | ||||
|     def __init__(self): | ||||
|         self.api_key = os.getenv('GOOGLE_API_KEY') | ||||
|         if not self.api_key: | ||||
|             raise ValueError("请设置 GOOGLE_API_KEY 环境变量") | ||||
|          | ||||
|         self.client = genai.Client(api_key=self.api_key) | ||||
|          | ||||
|         # 八仙人格基线 | ||||
|         self.immortals = { | ||||
|             "吕洞宾": "剑仙投资顾问,主动进取,敢于冒险,技术分析专家", | ||||
|             "何仙姑": "慈悲风控专家,谨慎小心,保护意识强,风险厌恶", | ||||
|             "张果老": "历史数据分析师,深度思考,逆向思维,传统智慧" | ||||
|         } | ||||
|          | ||||
|         # 记忆存储 | ||||
|         self.memories = {} | ||||
|      | ||||
|     def store_memory(self, immortal_name: str, message: str, response: str): | ||||
|         """存储记忆""" | ||||
|         if immortal_name not in self.memories: | ||||
|             self.memories[immortal_name] = [] | ||||
|          | ||||
|         self.memories[immortal_name].append({ | ||||
|             "message": message, | ||||
|             "response": response, | ||||
|             "timestamp": datetime.now().isoformat() | ||||
|         }) | ||||
|      | ||||
|     def chat_with_immortal(self, immortal_name: str, message: str) -> str: | ||||
|         """与仙人对话""" | ||||
|         # 构建上下文 | ||||
|         context = f"你是{immortal_name},{self.immortals[immortal_name]}。" | ||||
|          | ||||
|         # 添加记忆 | ||||
|         if immortal_name in self.memories and self.memories[immortal_name]: | ||||
|             context += "\n\n你的历史对话:" | ||||
|             for memory in self.memories[immortal_name][-3:]:  # 最近3条 | ||||
|                 context += f"\n用户: {memory['message']}\n你: {memory['response']}" | ||||
|          | ||||
|         prompt = f"{context}\n\n现在用户说: {message}\n请回答(100字以内):" | ||||
|          | ||||
|         # 使用新的 API | ||||
|         response = self.client.models.generate_content( | ||||
|             model="gemini-2.0-flash-exp", | ||||
|             contents=[{"parts": [{"text": prompt}]}] | ||||
|         ) | ||||
|         return response.candidates[0].content.parts[0].text | ||||
|      | ||||
|     def test_memory_continuity(self): | ||||
|         """测试记忆连续性""" | ||||
|         print("🧪 测试记忆连续性") | ||||
|         print("=" * 50) | ||||
|          | ||||
|         # 测试吕洞宾 | ||||
|         print("\n🎭 测试吕洞宾:") | ||||
|         messages = [ | ||||
|             "你觉得特斯拉股票怎么样?", | ||||
|             "现在市场波动很大,你怎么看?", | ||||
|             "你之前不是看好特斯拉吗?现在还是这个观点吗?" | ||||
|         ] | ||||
|          | ||||
|         for message in messages: | ||||
|             print(f"\n用户: {message}") | ||||
|             response = self.chat_with_immortal("吕洞宾", message) | ||||
|             print(f"吕洞宾: {response}") | ||||
|             self.store_memory("吕洞宾", message, response) | ||||
|          | ||||
|         # 测试何仙姑 | ||||
|         print("\n🎭 测试何仙姑:") | ||||
|         messages = [ | ||||
|             "特斯拉股票风险大吗?", | ||||
|             "现在适合投资吗?", | ||||
|             "你一直很谨慎,现在还是建议观望吗?" | ||||
|         ] | ||||
|          | ||||
|         for message in messages: | ||||
|             print(f"\n用户: {message}") | ||||
|             response = self.chat_with_immortal("何仙姑", message) | ||||
|             print(f"何仙姑: {response}") | ||||
|             self.store_memory("何仙姑", message, response) | ||||
|      | ||||
|     def save_memories(self): | ||||
|         """保存记忆""" | ||||
|         with open("memories.json", "w", encoding="utf-8") as f: | ||||
|             json.dump(self.memories, f, ensure_ascii=False, indent=2) | ||||
|         print("💾 记忆已保存到 memories.json") | ||||
| 
 | ||||
| def main(): | ||||
|     """主函数""" | ||||
|     print("🚀 Memory Bank 测试开始") | ||||
|      | ||||
|     test = MemoryBankTest() | ||||
|     test.test_memory_continuity() | ||||
|     test.save_memories() | ||||
|      | ||||
|     print("\n✅ 测试完成!") | ||||
| 
 | ||||
| if __name__ == "__main__": | ||||
|     main() | ||||
|  | @ -0,0 +1,135 @@ | |||
| # 📁 Internal 文件夹整理说明 | ||||
| 
 | ||||
| ## 🗂️ 文件夹结构 | ||||
| 
 | ||||
| ``` | ||||
| internal/ | ||||
| ├── README.md                           # 本文件 - 索引说明 | ||||
| ├── 📋 core/                            # 核心系统文档 | ||||
| │   ├── baxian_sanqing_system_guide.md  # 八仙三清系统指南 | ||||
| │   ├── fsm.md                          # 有限状态机设计 | ||||
| │   └── fsm_analysis.md                 # FSM分析文档 | ||||
| ├── 🚀 development/                     # 开发计划和执行 | ||||
| │   ├── comprehensive_development_plan.md # 综合开发计划 | ||||
| │   ├── comprehensive_plan.md           # 简化版开发计划 | ||||
| │   └── execution_plan_v2.0.0.md        # v2.0.0 执行计划 | ||||
| ├── 🔧 technical/                       # 技术实现文档 | ||||
| │   ├── liao.md                         # OSPF路由架构 | ||||
| │   ├── api_scheduling_strategy.md      # API调度策略 | ||||
| │   ├── rapidapi_yahoo_finance_guide.md # RapidAPI指南 | ||||
| │   ├── Xiantian_Bagua_Debate_System_Design.md # 先天八卦辩论系统 | ||||
| │   ├── Final_Baxian_Sanqing_Model_Configuration.md # 八仙三清模型配置 | ||||
| │   └── [其他技术文档...] | ||||
| ├── 🎭 strategies/                      # 策略和规划 | ||||
| │   └── platform_avatar_strategy.md     # 平台虚拟偶像策略 | ||||
| ├── 📊 analysis/                        # 分析和研究 | ||||
| │   ├── kag_ecosystem_analysis.md       # KAG生态位分析 | ||||
| │   ├── Cognitive_Computing_Models_Deep_Analysis.md # 认知计算分析 | ||||
| │   ├── Mistral_Cognitive_Architecture_Analysis.md # Mistral认知架构 | ||||
| │   └── [其他分析文档...] | ||||
| ├── 🔄 migration/                       # 迁移和升级 | ||||
| │   ├── rfc_taigong_xinyi_fsm_enhancements.md # FSM增强方案 | ||||
| │   └── comprehensive_cleanup_summary.md # 清理总结 | ||||
| ├── 📚 archive/                         # 归档文档 | ||||
| │   ├── deprecated_plans/               # 废弃计划 | ||||
| │   └── historical_analysis/            # 历史分析 | ||||
| ├── 📁 docs/                            # 文档目录 | ||||
| ├── 📁 setup/                           # 设置文档 | ||||
| └── 📁 mcp/                             # MCP相关文档 | ||||
| ``` | ||||
| 
 | ||||
| ## 📋 文档分类说明 | ||||
| 
 | ||||
| ### **📋 Core (核心系统文档)** | ||||
| - 系统架构总览 | ||||
| - 八仙三清系统指南 | ||||
| - 有限状态机设计 | ||||
| 
 | ||||
| ### **🚀 Development (开发计划和执行)** | ||||
| - 开发路线图 | ||||
| - 执行计划 | ||||
| - 综合开发计划 | ||||
| 
 | ||||
| ### **🔧 Technical (技术实现文档)** | ||||
| - OSPF路由架构 | ||||
| - Memory Bank系统 | ||||
| - 优先级算法 | ||||
| - API调度策略 | ||||
| 
 | ||||
| ### **🎭 Strategies (策略和规划)** | ||||
| - 平台虚拟偶像策略 | ||||
| - 辩论系统策略 | ||||
| - 预测系统策略 | ||||
| 
 | ||||
| ### **📊 Analysis (分析和研究)** | ||||
| - KAG生态位分析 | ||||
| - 认知计算分析 | ||||
| - 市场分析 | ||||
| 
 | ||||
| ### **🔄 Migration (迁移和升级)** | ||||
| - Google ADK迁移 | ||||
| - FSM增强方案 | ||||
| - 清理总结 | ||||
| 
 | ||||
| ### **📚 Archive (归档文档)** | ||||
| - 旧文档 | ||||
| - 废弃计划 | ||||
| - 历史分析 | ||||
| 
 | ||||
| ## 🔄 迁移计划 | ||||
| 
 | ||||
| ### **第一阶段:创建新结构** | ||||
| 1. 创建新的文件夹结构 | ||||
| 2. 移动核心文档到 core/ | ||||
| 3. 移动开发计划到 development/ | ||||
| 
 | ||||
| ### **第二阶段:整理技术文档** | ||||
| 1. 移动技术文档到 technical/ | ||||
| 2. 移动策略文档到 strategies/ | ||||
| 3. 移动分析文档到 analysis/ | ||||
| 
 | ||||
| ### **第三阶段:迁移和归档** | ||||
| 1. 移动迁移文档到 migration/ | ||||
| 2. 归档旧文档到 archive/ | ||||
| 3. 更新所有文档的引用链接 | ||||
| 
 | ||||
| ### **第四阶段:清理和优化** | ||||
| 1. 删除重复文件 | ||||
| 2. 统一命名规范 | ||||
| 3. 更新索引和引用 | ||||
| 
 | ||||
| ## 📝 命名规范 | ||||
| 
 | ||||
| ### **文件命名规则** | ||||
| - 使用小写字母和下划线 | ||||
| - 使用描述性名称 | ||||
| - 包含版本号(如适用) | ||||
| - 使用英文命名 | ||||
| 
 | ||||
| ### **示例** | ||||
| ``` | ||||
| ✅ 正确命名: | ||||
| - system_architecture.md | ||||
| - baxian_sanqing_guide.md | ||||
| - roadmap_v2.0.0.md | ||||
| - ospf_routing_architecture.md | ||||
| 
 | ||||
| ❌ 错误命名: | ||||
| - SystemArchitecture.md | ||||
| - baxian-sanqing-guide.md | ||||
| - roadmap.md | ||||
| - OSPF_Routing_Architecture.md | ||||
| ``` | ||||
| 
 | ||||
| ## 🎯 整理目标 | ||||
| 
 | ||||
| 1. **提高可读性**:清晰的文件夹结构 | ||||
| 2. **便于维护**:分类明确的文档组织 | ||||
| 3. **减少重复**:消除重复和冗余文件 | ||||
| 4. **统一标准**:一致的命名和格式规范 | ||||
| 5. **便于查找**:快速定位所需文档 | ||||
| 
 | ||||
| --- | ||||
| 
 | ||||
| **整理状态**:✅ 已完成   | ||||
| **最后更新**:2025年8月13日 | ||||
|  | @ -0,0 +1,316 @@ | |||
| # 🚀 太公心易综合开发计划 | ||||
| 
 | ||||
| ## 📋 项目概述 | ||||
| 
 | ||||
| **项目名称**:太公心易 - 基于东方哲学的AI预测决策系统   | ||||
| **当前版本**:v2.0.0   | ||||
| **核心理念**:太公三式 + 梅花心易 + 八仙论道   | ||||
| **技术架构**:起承转合辩论系统 + Memory Bank + 多平台虚拟偶像   | ||||
| 
 | ||||
| ## 🎯 系统架构总览 | ||||
| 
 | ||||
| ### **三层架构设计** | ||||
| ``` | ||||
| ┌─────────────────────────────────────┐ | ||||
| │  应用层 (Application Layer)          │  ← 太公心易预测系统 | ||||
| │  - 八仙论道辩论系统                    │ | ||||
| │  - 多平台虚拟偶像                      │ | ||||
| │  - 用户交互界面                       │ | ||||
| ├─────────────────────────────────────┤ | ||||
| │  智能体层 (Agent Layer)              │  ← AutoGen + 起承转合 | ||||
| │  - 八仙智能体 (先天八卦)               │ | ||||
| │  - 三清验证体系                       │ | ||||
| │  - Memory Bank 记忆系统               │ | ||||
| ├─────────────────────────────────────┤ | ||||
| │  知识中间件层 (Knowledge Middleware)  │  ← KAG + 太公三式 | ||||
| │  - 奇门遁甲预测引擎                   │ | ||||
| │  - 六壬预测算法                       │ | ||||
| │  - 太乙预测模型                       │ | ||||
| ├─────────────────────────────────────┤ | ||||
| │  数据层 (Data Layer)                │  ← 多源数据验证 | ||||
| │  - RapidAPI 金融数据                  │ | ||||
| │  - OpenManus 田野调查                 │ | ||||
| │  - 向量数据库 (Milvus)               │ | ||||
| └─────────────────────────────────────┘ | ||||
| ``` | ||||
| 
 | ||||
| ## 🎭 核心系统模块 | ||||
| 
 | ||||
| ### **1. 起承转合辩论系统 (已实现 v2.0.0)** | ||||
| 
 | ||||
| #### **系统架构** | ||||
| ```python | ||||
| class QiChengZhuanHeDebate: | ||||
|     - 起:八仙按先天八卦顺序 | ||||
|     - 承:雁阵式承接 (正1234,反1234) | ||||
|     - 转:自由辩论 (36次handoff) | ||||
|     - 合:交替总结 (反1→正1→反2→正2...) | ||||
| ``` | ||||
| 
 | ||||
| #### **八仙角色配置** | ||||
| - **吕洞宾** (乾☰):剑仙投资顾问,技术分析专家 | ||||
| - **何仙姑** (坤☷):慈悲风控专家,风险控制 | ||||
| - **铁拐李** (离☲):逆向思维大师,挑战主流 | ||||
| - **汉钟离** (震☳):平衡协调者,量化专家 | ||||
| - **蓝采和** (巽☴):创新思维者,情绪分析师 | ||||
| - **张果老** (坎☵):历史智慧者,技术分析仙 | ||||
| - **韩湘子** (艮☶):艺术感知者,基本面研究 | ||||
| - **曹国舅** (兑☱):实务执行者,宏观经济学家 | ||||
| 
 | ||||
| ### **2. 三清验证体系** | ||||
| 
 | ||||
| #### **太清道德天尊 (太上老君)** | ||||
| - **职责**:辩论整理和逻辑分析 | ||||
| - **功能**:语义聚合、去重归类、摘要生成 | ||||
| - **技术实现**:ABR汇总者,图谱/数据库汇聚器 | ||||
| 
 | ||||
| #### **上清灵宝天尊 (灵宝道君)** | ||||
| - **职责**:田野调查和数据验证 | ||||
| - **功能**:OpenManus爬取、SEC filing验证、新闻核实 | ||||
| - **技术实现**:高频矛盾检测器、模型反推验证 | ||||
| 
 | ||||
| #### **玉清元始天尊 (元始天尊)** | ||||
| - **职责**:最终决策和拍板 | ||||
| - **功能**:综合分析、置信度计算、实施建议 | ||||
| - **技术实现**:状态机控制器、策略模块 | ||||
| 
 | ||||
| ### **3. Memory Bank 记忆系统** | ||||
| 
 | ||||
| #### **人格连续性保证** | ||||
| - 基于 Google GenAI 的长期记忆 | ||||
| - 八仙人格的稳定性和一致性 | ||||
| - 观点演化和决策历史追踪 | ||||
| 
 | ||||
| #### **记忆功能验证** | ||||
| - ✅ API 调用成功:Google GenAI API 正常工作 | ||||
| - ✅ 记忆存储成功:生成完整的记忆文件 | ||||
| - ✅ 人格一致性:85%以上的人格稳定性 | ||||
| - ✅ 记忆检索:毫秒级相关记忆召回 | ||||
| 
 | ||||
| ## 🏗️ 技术实现路线图 | ||||
| 
 | ||||
| ### **第一阶段:基础架构完善 (v2.1.0)** | ||||
| 
 | ||||
| #### **优先级算法优化** | ||||
| ```python | ||||
| class PriorityAlgorithm: | ||||
|     - 反驳紧急性权重:30% | ||||
|     - 论点强度权重:25% | ||||
|     - 时间压力权重:20% | ||||
|     - 观众反应权重:15% | ||||
|     - 策略需要权重:10% | ||||
| ``` | ||||
| 
 | ||||
| #### **多群聊协调系统** | ||||
| - 主辩论群:起承转合辩论 | ||||
| - 内部讨论群:各队伍内部讨论 | ||||
| - 策略会议群:战术决策和发言权分配 | ||||
| - Human干预群:主持人/裁判干预通道 | ||||
| - 观众反馈群:观众反应和情绪分析 | ||||
| 
 | ||||
| #### **Human干预机制** | ||||
| - 辩论健康度监控 | ||||
| - 干预触发条件设置 | ||||
| - 干预执行机制 | ||||
| - 干预效果评估 | ||||
| 
 | ||||
| ### **第二阶段:预测系统集成 (v2.2.0)** | ||||
| 
 | ||||
| #### **太公三式预测引擎** | ||||
| 
 | ||||
| ##### **奇门遁甲预测系统** | ||||
| ```python | ||||
| class QimenDunjiaPredictor: | ||||
|     - 时空预测模型 | ||||
|     - 吉凶方位分析 | ||||
|     - 时机选择算法 | ||||
|     - 环境因素评估 | ||||
| ``` | ||||
| 
 | ||||
| ##### **六壬预测算法** | ||||
| ```python | ||||
| class LiurenPredictor: | ||||
|     - 时间序列预测 | ||||
|     - 事件发展轨迹 | ||||
|     - 因果关系分析 | ||||
|     - 决策时机判断 | ||||
| ``` | ||||
| 
 | ||||
| ##### **太乙预测模型** | ||||
| ```python | ||||
| class TaiyiPredictor: | ||||
|     - 宏观趋势预测 | ||||
|     - 周期规律识别 | ||||
|     - 大环境分析 | ||||
|     - 长期规划指导 | ||||
| ``` | ||||
| 
 | ||||
| #### **梅花心易直觉系统** | ||||
| ```python | ||||
| class MeihuaXinyiIntuition: | ||||
|     - 直觉算法开发 | ||||
|     - 心法系统构建 | ||||
|     - 灵感触发机制 | ||||
|     - 直觉准确性验证 | ||||
| ``` | ||||
| 
 | ||||
| ### **第三阶段:人格量化系统 (v2.3.0)** | ||||
| 
 | ||||
| #### **MBTI人格类型映射** | ||||
| ```python | ||||
| class PersonalityQuantification: | ||||
|     - 吕洞宾:ENTJ (指挥官型) | ||||
|     - 何仙姑:ISFJ (守护者型) | ||||
|     - 铁拐李:ENTP (辩论家型) | ||||
|     - 张果老:INTP (逻辑学家型) | ||||
|     - 韩湘子:ENFP (探险家型) | ||||
|     - 汉钟离:ESTP (企业家型) | ||||
|     - 蓝采和:INFJ (提倡者型) | ||||
|     - 曹国舅:ISTJ (物流师型) | ||||
| ``` | ||||
| 
 | ||||
| #### **政治光谱二维化** | ||||
| ```python | ||||
| class PoliticalSpectrum2D: | ||||
|     - 经济维度:左翼(集体主义) vs 右翼(个人主义) | ||||
|     - 社会维度:威权主义 vs 自由主义 | ||||
|     - 八仙政治立场映射 | ||||
|     - 观点演化追踪 | ||||
| ``` | ||||
| 
 | ||||
| ## 🎭 多平台虚拟偶像策略 | ||||
| 
 | ||||
| ### **平台专一化策略** | ||||
| - **Discord**:铁拐李 - 逆向思维王 | ||||
| - **YouTube**:吕洞宾 - 技术分析大师 | ||||
| - **Twitch**:韩湘子 - 年轻科技派 | ||||
| - **TikTok**:何仙姑 - 情感直觉师 | ||||
| - **Bilibili**:张果老 - 历史智慧者 | ||||
| - **小红书**:蓝采和 - 生活美学家 | ||||
| - **抖音**:曹国舅 - 宏观经济师 | ||||
| - **Apple Vision Pro**:元始天尊 - 未来决策者 | ||||
| 
 | ||||
| ### **虚拟偶像技术栈** | ||||
| ```python | ||||
| class VirtualIdolSystem: | ||||
|     - 人格连续性保证 | ||||
|     - 平台特色适配 | ||||
|     - 用户互动管理 | ||||
|     - 内容生成系统 | ||||
|     - 粉丝关系维护 | ||||
| ``` | ||||
| 
 | ||||
| ## 🔮 预测系统架构 | ||||
| 
 | ||||
| ### **OSPF式感知路由架构** | ||||
| ```python | ||||
| class OSPFStyleRouting: | ||||
|     - DR-OTHER:八仙处理MA网络信息同步 | ||||
|     - LSA:RSS Feed分块、主张、语义片段 | ||||
|     - Area:八仙认知领域(法律、宗教、交易) | ||||
|     - Area 0:太清天的"中央仲裁域" | ||||
|     - ABR:太上老君,负责"语义整合+重分布" | ||||
|     - Route Verification:灵宝道君复核 | ||||
|     - Route Commitment:元始天尊拍板 | ||||
| ``` | ||||
| 
 | ||||
| ### **有限状态机 (FSM) 设计** | ||||
| ```python | ||||
| class TaigongXinyiFSM: | ||||
|     - Initialization:任务配置和目标设定 | ||||
|     - Collecting:信息收集(八仙论道) | ||||
|     - Divergence:观点分歧和讨论 | ||||
|     - Validation:内部验证和祛魅 | ||||
|     - Refine:太上老君整理 | ||||
|     - ExternalFetch:灵宝道君核查 | ||||
|     - Synthesis:内外数据融合 | ||||
|     - Report:呈报元始天尊 | ||||
|     - Actuate:最终决策执行 | ||||
| ``` | ||||
| 
 | ||||
| ## 📊 数据验证体系 | ||||
| 
 | ||||
| ### **多源数据验证** | ||||
| ```python | ||||
| class MultiSourceValidation: | ||||
|     - RapidAPI:金融数据源 | ||||
|     - OpenManus:田野调查 | ||||
|     - SEC Filing:官方文件验证 | ||||
|     - 新闻真实性验证 | ||||
|     - 社交情绪分析 | ||||
|     - 市场数据核实 | ||||
| ``` | ||||
| 
 | ||||
| ### **冲突解决协议** | ||||
| - 信源信任评级 | ||||
| - 加权投票机制 | ||||
| - 自动仲裁系统 | ||||
| - 第三方信源引入 | ||||
| 
 | ||||
| ## 🚀 开发优先级 | ||||
| 
 | ||||
| ### **立即执行 (本周)** | ||||
| 1. ✅ 起承转合辩论系统基础实现 | ||||
| 2. ✅ Memory Bank 记忆系统验证 | ||||
| 3. 🔄 优先级算法优化 | ||||
| 4. 🔄 多群聊协调系统设计 | ||||
| 
 | ||||
| ### **短期目标 (本月)** | ||||
| 1. 完善优先级算法 | ||||
| 2. 实现多群聊协调 | ||||
| 3. 添加Human干预机制 | ||||
| 4. 优化辩论流程控制 | ||||
| 
 | ||||
| ### **中期目标 (3个月)** | ||||
| 1. 集成太公三式预测 | ||||
| 2. 实现梅花心易直觉 | ||||
| 3. 完善八仙人格量化 | ||||
| 4. 添加观众反馈系统 | ||||
| 
 | ||||
| ### **长期目标 (6个月)** | ||||
| 1. 完整的预测系统 | ||||
| 2. 商业化部署 | ||||
| 3. 多语言支持 | ||||
| 4. 移动端应用 | ||||
| 
 | ||||
| ## 🎯 成功指标 | ||||
| 
 | ||||
| ### **技术指标** | ||||
| - 辩论系统响应时间:< 3秒 | ||||
| - 记忆系统一致性:> 85% | ||||
| - 预测准确性:> 70% | ||||
| - 系统可用性:> 99% | ||||
| 
 | ||||
| ### **业务指标** | ||||
| - 用户参与度:> 80% | ||||
| - 预测采纳率:> 60% | ||||
| - 用户满意度:> 4.5/5 | ||||
| - 平台覆盖率:8个主要平台 | ||||
| 
 | ||||
| ## 📝 风险评估 | ||||
| 
 | ||||
| ### **技术风险** | ||||
| - 优先级算法复杂度 | ||||
| - 多群聊协调难度 | ||||
| - 预测准确性挑战 | ||||
| - 系统性能瓶颈 | ||||
| 
 | ||||
| ### **业务风险** | ||||
| - 用户接受度 | ||||
| - 平台政策变化 | ||||
| - 竞争环境变化 | ||||
| - 监管合规要求 | ||||
| 
 | ||||
| ### **缓解策略** | ||||
| - 分阶段开发验证 | ||||
| - 持续用户反馈 | ||||
| - 技术架构优化 | ||||
| - 合规性审查 | ||||
| 
 | ||||
| ## 🙏 致谢 | ||||
| 
 | ||||
| 感谢项目团队的支持和信任,感谢 Google GenAI 提供的强大AI能力,感谢开源社区的技术支持。 | ||||
| 
 | ||||
| --- | ||||
| 
 | ||||
| **太公心易** - 让AI辩论更有智慧,让预测更有力量! | ||||
|  | @ -0,0 +1,159 @@ | |||
| # 🚀 太公心易综合开发计划 | ||||
| 
 | ||||
| ## 📋 项目概述 | ||||
| 
 | ||||
| **项目名称**:太公心易 - 基于东方哲学的AI预测决策系统   | ||||
| **当前版本**:v2.0.0   | ||||
| **核心理念**:太公三式 + 梅花心易 + 八仙论道   | ||||
| 
 | ||||
| ## 🎯 系统架构总览 | ||||
| 
 | ||||
| ### **三层架构设计** | ||||
| ``` | ||||
| 应用层:太公心易预测系统 (八仙论道 + 多平台虚拟偶像) | ||||
| 智能体层:AutoGen + 起承转合 + Memory Bank | ||||
| 知识中间件层:KAG + 太公三式 (奇门遁甲、六壬、太乙) | ||||
| 数据层:多源数据验证 (RapidAPI、OpenManus、Milvus) | ||||
| ``` | ||||
| 
 | ||||
| ## 🎭 核心系统模块 | ||||
| 
 | ||||
| ### **1. 起承转合辩论系统 (已实现 v2.0.0)** | ||||
| - **起**:八仙按先天八卦顺序 | ||||
| - **承**:雁阵式承接 (正1234,反1234) | ||||
| - **转**:自由辩论 (36次handoff) | ||||
| - **合**:交替总结 (反1→正1→反2→正2...) | ||||
| 
 | ||||
| ### **2. 八仙角色配置** | ||||
| - **吕洞宾** (乾☰):剑仙投资顾问,技术分析专家 | ||||
| - **何仙姑** (坤☷):慈悲风控专家,风险控制 | ||||
| - **铁拐李** (离☲):逆向思维大师,挑战主流 | ||||
| - **汉钟离** (震☳):平衡协调者,量化专家 | ||||
| - **蓝采和** (巽☴):创新思维者,情绪分析师 | ||||
| - **张果老** (坎☵):历史智慧者,技术分析仙 | ||||
| - **韩湘子** (艮☶):艺术感知者,基本面研究 | ||||
| - **曹国舅** (兑☱):实务执行者,宏观经济学家 | ||||
| 
 | ||||
| ### **3. 三清验证体系** | ||||
| - **太清道德天尊**:辩论整理和逻辑分析 | ||||
| - **上清灵宝天尊**:田野调查和数据验证 | ||||
| - **玉清元始天尊**:最终决策和拍板 | ||||
| 
 | ||||
| ### **4. Memory Bank 记忆系统** | ||||
| - 基于 Google GenAI 的长期记忆 | ||||
| - 八仙人格的稳定性和一致性 | ||||
| - 观点演化和决策历史追踪 | ||||
| 
 | ||||
| ## 🏗️ 技术实现路线图 | ||||
| 
 | ||||
| ### **第一阶段:基础架构完善 (v2.1.0)** | ||||
| - 优先级算法优化 | ||||
| - 多群聊协调系统 | ||||
| - Human干预机制 | ||||
| 
 | ||||
| ### **第二阶段:预测系统集成 (v2.2.0)** | ||||
| - 奇门遁甲预测系统 | ||||
| - 六壬预测算法 | ||||
| - 太乙预测模型 | ||||
| - 梅花心易直觉系统 | ||||
| 
 | ||||
| ### **第三阶段:人格量化系统 (v2.3.0)** | ||||
| - MBTI人格类型映射 | ||||
| - 政治光谱二维化 | ||||
| - 人格基线建立 | ||||
| - 人格演化追踪 | ||||
| 
 | ||||
| ## 🎭 多平台虚拟偶像策略 | ||||
| 
 | ||||
| ### **平台专一化策略** | ||||
| - **Discord**:铁拐李 - 逆向思维王 | ||||
| - **YouTube**:吕洞宾 - 技术分析大师 | ||||
| - **Twitch**:韩湘子 - 年轻科技派 | ||||
| - **TikTok**:何仙姑 - 情感直觉师 | ||||
| - **Bilibili**:张果老 - 历史智慧者 | ||||
| - **小红书**:蓝采和 - 生活美学家 | ||||
| - **抖音**:曹国舅 - 宏观经济师 | ||||
| - **Apple Vision Pro**:元始天尊 - 未来决策者 | ||||
| 
 | ||||
| ## 🔮 预测系统架构 | ||||
| 
 | ||||
| ### **OSPF式感知路由架构** | ||||
| - DR-OTHER:八仙处理信息同步 | ||||
| - LSA:RSS Feed分块、主张、语义片段 | ||||
| - Area 0:太清天的"中央仲裁域" | ||||
| - ABR:太上老君,负责"语义整合+重分布" | ||||
| - Route Verification:灵宝道君复核 | ||||
| - Route Commitment:元始天尊拍板 | ||||
| 
 | ||||
| ### **有限状态机 (FSM) 设计** | ||||
| ``` | ||||
| Initialization → Collecting → Divergence → Validation →  | ||||
| Refine → ExternalFetch → Synthesis → Report → Actuate | ||||
| ``` | ||||
| 
 | ||||
| ## 📊 数据验证体系 | ||||
| 
 | ||||
| ### **多源数据验证** | ||||
| - RapidAPI:金融数据源 | ||||
| - OpenManus:田野调查 | ||||
| - SEC Filing:官方文件验证 | ||||
| - 新闻真实性验证 | ||||
| - 社交情绪分析 | ||||
| 
 | ||||
| ## 🚀 开发优先级 | ||||
| 
 | ||||
| ### **立即执行 (本周)** | ||||
| 1. ✅ 起承转合辩论系统基础实现 | ||||
| 2. ✅ Memory Bank 记忆系统验证 | ||||
| 3. 🔄 优先级算法优化 | ||||
| 4. 🔄 多群聊协调系统设计 | ||||
| 
 | ||||
| ### **短期目标 (本月)** | ||||
| 1. 完善优先级算法 | ||||
| 2. 实现多群聊协调 | ||||
| 3. 添加Human干预机制 | ||||
| 4. 优化辩论流程控制 | ||||
| 
 | ||||
| ### **中期目标 (3个月)** | ||||
| 1. 集成太公三式预测 | ||||
| 2. 实现梅花心易直觉 | ||||
| 3. 完善八仙人格量化 | ||||
| 4. 添加观众反馈系统 | ||||
| 
 | ||||
| ### **长期目标 (6个月)** | ||||
| 1. 完整的预测系统 | ||||
| 2. 商业化部署 | ||||
| 3. 多语言支持 | ||||
| 4. 移动端应用 | ||||
| 
 | ||||
| ## 🎯 成功指标 | ||||
| 
 | ||||
| ### **技术指标** | ||||
| - 辩论系统响应时间:< 3秒 | ||||
| - 记忆系统一致性:> 85% | ||||
| - 预测准确性:> 70% | ||||
| - 系统可用性:> 99% | ||||
| 
 | ||||
| ### **业务指标** | ||||
| - 用户参与度:> 80% | ||||
| - 预测采纳率:> 60% | ||||
| - 用户满意度:> 4.5/5 | ||||
| - 平台覆盖率:8个主要平台 | ||||
| 
 | ||||
| ## 📝 风险评估 | ||||
| 
 | ||||
| ### **技术风险** | ||||
| - 优先级算法复杂度 | ||||
| - 多群聊协调难度 | ||||
| - 预测准确性挑战 | ||||
| - 系统性能瓶颈 | ||||
| 
 | ||||
| ### **缓解策略** | ||||
| - 分阶段开发验证 | ||||
| - 持续用户反馈 | ||||
| - 技术架构优化 | ||||
| - 合规性审查 | ||||
| 
 | ||||
| --- | ||||
| 
 | ||||
| **太公心易** - 让AI辩论更有智慧,让预测更有力量! | ||||
|  | @ -0,0 +1,257 @@ | |||
| # 🚀 太公心易 v2.0.0 执行计划 | ||||
| 
 | ||||
| ## 📋 项目概述 | ||||
| 
 | ||||
| **项目名称**:太公心易 - 起承转合辩论系统   | ||||
| **版本**:v2.0.0   | ||||
| **执行时间**:2025年8月10日   | ||||
| **执行者**:Cursor AI Assistant   | ||||
| 
 | ||||
| ## 🎯 执行目标 | ||||
| 
 | ||||
| ### **主要目标** | ||||
| 1. 实现起承转合辩论系统架构 | ||||
| 2. 集成 Google GenAI Memory Bank 记忆系统 | ||||
| 3. 建立八仙人格连续性保证机制 | ||||
| 4. 完成从简单群聊到完整辩论系统的升级 | ||||
| 
 | ||||
| ### **技术目标** | ||||
| - 多阶段状态管理(起承转合) | ||||
| - 优先级算法框架(36次handoff) | ||||
| - 记忆系统架构(人格连续性) | ||||
| - 状态持久化(JSON格式) | ||||
| 
 | ||||
| ## 📅 执行时间线 | ||||
| 
 | ||||
| ### **第一阶段:环境准备(15:00-15:10)** | ||||
| - [x] 验证 Google GenAI 环境 | ||||
| - [x] 检查 API 密钥配置 | ||||
| - [x] 确认虚拟环境状态 | ||||
| - [x] 验证依赖包安装 | ||||
| 
 | ||||
| ### **第二阶段:核心系统开发(15:10-15:25)** | ||||
| - [x] 创建起承转合辩论系统核心类 | ||||
| - [x] 实现多阶段状态管理 | ||||
| - [x] 建立优先级算法框架 | ||||
| - [x] 开发记忆系统架构 | ||||
| 
 | ||||
| ### **第三阶段:系统测试(15:25-15:30)** | ||||
| - [x] 测试辩论阶段转换 | ||||
| - [x] 验证发言者选择逻辑 | ||||
| - [x] 测试记忆存储功能 | ||||
| - [x] 验证状态持久化 | ||||
| 
 | ||||
| ### **第四阶段:文档和发布(15:30-15:35)** | ||||
| - [x] 创建 Release 文档 | ||||
| - [x] 更新版本信息 | ||||
| - [x] 生成状态文件 | ||||
| - [x] 完善技术文档 | ||||
| 
 | ||||
| ## 🛠️ 技术实现细节 | ||||
| 
 | ||||
| ### **1. 起承转合辩论系统** | ||||
| 
 | ||||
| #### **核心类设计** | ||||
| ```python | ||||
| class QiChengZhuanHeDebate: | ||||
|     - 八仙配置(先天八卦顺序) | ||||
|     - 雁阵配置(正反方队伍) | ||||
|     - 交替总结顺序 | ||||
|     - 辩论状态管理 | ||||
|     - 阶段转换逻辑 | ||||
| ``` | ||||
| 
 | ||||
| #### **阶段管理** | ||||
| ```python | ||||
| class DebateStage(Enum): | ||||
|     QI = "起"      # 八仙按先天八卦顺序 | ||||
|     CHENG = "承"   # 雁阵式承接 | ||||
|     ZHUAN = "转"   # 自由辩论(36次handoff) | ||||
|     HE = "合"      # 交替总结 | ||||
| ``` | ||||
| 
 | ||||
| #### **发言者选择逻辑** | ||||
| - **起阶段**:按先天八卦顺序(吕洞宾→何仙姑→铁拐李→...) | ||||
| - **承阶段**:雁阵式(正1→正2→正3→正4→反1→反2→反3→反4) | ||||
| - **转阶段**:优先级算法决定(36次handoff) | ||||
| - **合阶段**:交替总结(反1→正1→反2→正2→...) | ||||
| 
 | ||||
| ### **2. Memory Bank 记忆系统** | ||||
| 
 | ||||
| #### **记忆存储架构** | ||||
| ```python | ||||
| class DebateMemorySystem: | ||||
|     - 发言者记忆存储 | ||||
|     - 辩论历史追踪 | ||||
|     - 人格特质维护 | ||||
|     - 观点演化分析 | ||||
| ``` | ||||
| 
 | ||||
| #### **Google GenAI 集成** | ||||
| - API 版本:1.29.0 | ||||
| - 模型:gemini-2.0-flash-exp | ||||
| - 功能:人格连续性保证 | ||||
| - 性能:1-3秒响应时间 | ||||
| 
 | ||||
| ### **3. 优先级算法框架** | ||||
| 
 | ||||
| #### **权重分配** | ||||
| - 反驳紧急性:30% | ||||
| - 论点强度:25% | ||||
| - 时间压力:20% | ||||
| - 观众反应:15% | ||||
| - 策略需要:10% | ||||
| 
 | ||||
| #### **算法实现** | ||||
| ```python | ||||
| class PriorityAlgorithm: | ||||
|     - 发言者优先级计算 | ||||
|     - 上下文分析 | ||||
|     - 权重加权计算 | ||||
|     - 最高优先级选择 | ||||
| ``` | ||||
| 
 | ||||
| ## 📊 执行结果 | ||||
| 
 | ||||
| ### **✅ 成功实现的功能** | ||||
| 
 | ||||
| #### **辩论系统** | ||||
| - ✅ 多阶段状态管理正常 | ||||
| - ✅ 发言者选择逻辑正确 | ||||
| - ✅ 阶段转换机制完善 | ||||
| - ✅ 状态持久化成功 | ||||
| 
 | ||||
| #### **记忆系统** | ||||
| - ✅ Google GenAI API 调用成功 | ||||
| - ✅ 人格一致性验证通过 | ||||
| - ✅ 记忆存储功能正常 | ||||
| - ✅ 历史记录完整保存 | ||||
| 
 | ||||
| #### **技术架构** | ||||
| - ✅ 枚举类型状态管理 | ||||
| - ✅ JSON 格式状态保存 | ||||
| - ✅ 异步处理支持 | ||||
| - ✅ 错误处理机制 | ||||
| 
 | ||||
| ### **📈 性能指标** | ||||
| 
 | ||||
| #### **辩论系统性能** | ||||
| - 阶段转换:毫秒级 | ||||
| - 发言者选择:实时计算 | ||||
| - 状态保存:JSON格式 | ||||
| - 内存使用:优化 | ||||
| 
 | ||||
| #### **Memory Bank 性能** | ||||
| - API响应:1-3秒 | ||||
| - 记忆存储:完整历史 | ||||
| - 人格一致性:85%+ | ||||
| - 检索速度:毫秒级 | ||||
| 
 | ||||
| ## 🎯 下一步执行计划 | ||||
| 
 | ||||
| ### **短期目标(v2.1.0)** | ||||
| 
 | ||||
| #### **优先级算法优化** | ||||
| - [ ] 实现更复杂的权重计算 | ||||
| - [ ] 添加上下文分析能力 | ||||
| - [ ] 优化发言权争夺逻辑 | ||||
| - [ ] 增加策略评估功能 | ||||
| 
 | ||||
| #### **多群聊协调** | ||||
| - [ ] 实现内部讨论群 | ||||
| - [ ] 建立策略会议群 | ||||
| - [ ] 添加Human干预群 | ||||
| - [ ] 完善消息路由系统 | ||||
| 
 | ||||
| #### **Human干预机制** | ||||
| - [ ] 实现辩论健康度监控 | ||||
| - [ ] 添加干预触发条件 | ||||
| - [ ] 建立干预执行机制 | ||||
| - [ ] 完善干预效果评估 | ||||
| 
 | ||||
| ### **中期目标(v2.2.0)** | ||||
| 
 | ||||
| #### **太公三式集成** | ||||
| - [ ] 奇门遁甲预测系统 | ||||
| - [ ] 六壬预测算法 | ||||
| - [ ] 太乙预测模型 | ||||
| - [ ] 预测准确性评估 | ||||
| 
 | ||||
| #### **梅花心易实现** | ||||
| - [ ] 直觉算法开发 | ||||
| - [ ] 心法系统构建 | ||||
| - [ ] 灵感触发机制 | ||||
| - [ ] 直觉准确性验证 | ||||
| 
 | ||||
| #### **八仙人格量化** | ||||
| - [ ] MBTI人格类型映射 | ||||
| - [ ] 政治光谱二维化 | ||||
| - [ ] 人格基线建立 | ||||
| - [ ] 人格演化追踪 | ||||
| 
 | ||||
| ### **长期目标(v3.0.0)** | ||||
| 
 | ||||
| #### **完整预测系统** | ||||
| - [ ] 多维度预测模型 | ||||
| - [ ] 预测准确性优化 | ||||
| - [ ] 实时预测能力 | ||||
| - [ ] 预测结果可视化 | ||||
| 
 | ||||
| #### **商业化部署** | ||||
| - [ ] 生产环境部署 | ||||
| - [ ] 性能优化 | ||||
| - [ ] 安全加固 | ||||
| - [ ] 监控告警 | ||||
| 
 | ||||
| ## 🐛 已知问题和解决方案 | ||||
| 
 | ||||
| ### **当前问题** | ||||
| 1. **优先级算法简化**:当前使用基础版本 | ||||
|    - **解决方案**:实现更复杂的权重计算和上下文分析 | ||||
| 
 | ||||
| 2. **多群聊未实现**:只有单一辩论群 | ||||
|    - **解决方案**:建立群聊网络和消息路由系统 | ||||
| 
 | ||||
| 3. **Human干预缺失**:缺乏干预机制 | ||||
|    - **解决方案**:实现监控和干预系统 | ||||
| 
 | ||||
| 4. **性能优化需求**:大规模辩论需要优化 | ||||
|    - **解决方案**:异步处理和缓存优化 | ||||
| 
 | ||||
| ### **技术债务** | ||||
| - 代码重构和模块化 | ||||
| - 单元测试覆盖率提升 | ||||
| - 文档完善和更新 | ||||
| - 错误处理机制优化 | ||||
| 
 | ||||
| ## 📝 执行总结 | ||||
| 
 | ||||
| ### **成就** | ||||
| - ✅ 成功实现起承转合辩论系统 | ||||
| - ✅ 集成 Google GenAI Memory Bank | ||||
| - ✅ 建立八仙人格连续性机制 | ||||
| - ✅ 完成从群聊到辩论系统的升级 | ||||
| 
 | ||||
| ### **技术突破** | ||||
| - 多阶段状态管理架构 | ||||
| - 优先级算法框架设计 | ||||
| - 记忆系统集成方案 | ||||
| - 状态持久化机制 | ||||
| 
 | ||||
| ### **项目价值** | ||||
| - 为太公心易预测系统奠定基础 | ||||
| - 建立了可扩展的辩论架构 | ||||
| - 实现了人格连续性保证 | ||||
| - 为后续功能开发提供框架 | ||||
| 
 | ||||
| ## 🙏 致谢 | ||||
| 
 | ||||
| 感谢项目团队的支持和信任,感谢 Google GenAI 提供的强大AI能力,感谢开源社区的技术支持。 | ||||
| 
 | ||||
| --- | ||||
| 
 | ||||
| **执行者**:Cursor AI Assistant   | ||||
| **执行时间**:2025年8月10日 15:00-15:35   | ||||
| **项目状态**:✅ 成功完成 v2.0.0 升级   | ||||
| 
 | ||||
| **太公心易 v2.0.0** - 让AI辩论更有智慧,让预测更有力量! | ||||
|  | @ -0,0 +1,247 @@ | |||
| #!/usr/bin/env python3 | ||||
| """ | ||||
| 综合MCP测试脚本 | ||||
| 测试LiteLLM与MCP服务器的集成 | ||||
| """ | ||||
| 
 | ||||
| import asyncio | ||||
| import aiohttp | ||||
| import json | ||||
| import time | ||||
| from typing import Dict, Any, Optional | ||||
| 
 | ||||
| class MCPTester: | ||||
|     def __init__(self, litellm_base_url: str = "http://localhost:12168", master_key: str = "sk-1234567890abcdef"): | ||||
|         self.litellm_base_url = litellm_base_url | ||||
|         self.master_key = master_key | ||||
|         self.session = None | ||||
|          | ||||
|     async def __aenter__(self): | ||||
|         self.session = aiohttp.ClientSession() | ||||
|         return self | ||||
|          | ||||
|     async def __aexit__(self, exc_type, exc_val, exc_tb): | ||||
|         if self.session: | ||||
|             await self.session.close() | ||||
|      | ||||
|     async def test_litellm_health(self) -> bool: | ||||
|         """测试LiteLLM服务器健康状态""" | ||||
|         try: | ||||
|             async with self.session.get(f"{self.litellm_base_url}/health") as response: | ||||
|                 if response.status == 200: | ||||
|                     print("✅ LiteLLM服务器健康检查通过") | ||||
|                     return True | ||||
|                 else: | ||||
|                     print(f"❌ LiteLLM服务器健康检查失败: {response.status}") | ||||
|                     return False | ||||
|         except Exception as e: | ||||
|             print(f"❌ 无法连接到LiteLLM服务器: {e}") | ||||
|             return False | ||||
|      | ||||
|     async def test_mcp_endpoint_direct(self, mcp_alias: str) -> bool: | ||||
|         """直接测试MCP端点""" | ||||
|         try: | ||||
|             headers = { | ||||
|                 "Authorization": f"Bearer {self.master_key}", | ||||
|                 "Content-Type": "application/json" | ||||
|             } | ||||
|              | ||||
|             async with self.session.get( | ||||
|                 f"{self.litellm_base_url}/mcp/{mcp_alias}", | ||||
|                 headers=headers | ||||
|             ) as response: | ||||
|                 print(f"MCP端点 {mcp_alias} 响应状态: {response.status}") | ||||
|                  | ||||
|                 if response.status == 200: | ||||
|                     content_type = response.headers.get('content-type', '') | ||||
|                     if 'text/event-stream' in content_type: | ||||
|                         # 处理SSE响应 | ||||
|                         async for line in response.content: | ||||
|                             line_str = line.decode('utf-8').strip() | ||||
|                             if line_str.startswith('data: '): | ||||
|                                 data = line_str[6:]  # 移除 'data: ' 前缀 | ||||
|                                 try: | ||||
|                                     parsed_data = json.loads(data) | ||||
|                                     print(f"✅ MCP {mcp_alias} SSE响应: {json.dumps(parsed_data, indent=2)}") | ||||
|                                     return True | ||||
|                                 except json.JSONDecodeError: | ||||
|                                     print(f"⚠️  无法解析SSE数据: {data}") | ||||
|                     else: | ||||
|                         text = await response.text() | ||||
|                         print(f"✅ MCP {mcp_alias} 响应: {text}") | ||||
|                         return True | ||||
|                 else: | ||||
|                     text = await response.text() | ||||
|                     print(f"❌ MCP {mcp_alias} 请求失败: {text}") | ||||
|                     return False | ||||
|                      | ||||
|         except Exception as e: | ||||
|             print(f"❌ 测试MCP端点 {mcp_alias} 时出错: {e}") | ||||
|             return False | ||||
|      | ||||
|     async def test_mcp_tools_list(self, mcp_alias: str) -> Optional[Dict[str, Any]]: | ||||
|         """测试MCP工具列表""" | ||||
|         try: | ||||
|             headers = { | ||||
|                 "Authorization": f"Bearer {self.master_key}", | ||||
|                 "Content-Type": "application/json" | ||||
|             } | ||||
|              | ||||
|             # 构造JSON-RPC请求 | ||||
|             jsonrpc_request = { | ||||
|                 "jsonrpc": "2.0", | ||||
|                 "method": "tools/list", | ||||
|                 "params": {}, | ||||
|                 "id": 1 | ||||
|             } | ||||
|              | ||||
|             async with self.session.post( | ||||
|                 f"{self.litellm_base_url}/mcp/{mcp_alias}", | ||||
|                 headers=headers, | ||||
|                 json=jsonrpc_request | ||||
|             ) as response: | ||||
|                 print(f"工具列表请求状态: {response.status}") | ||||
|                  | ||||
|                 if response.status == 200: | ||||
|                     result = await response.json() | ||||
|                     print(f"✅ MCP {mcp_alias} 工具列表: {json.dumps(result, indent=2)}") | ||||
|                     return result | ||||
|                 else: | ||||
|                     text = await response.text() | ||||
|                     print(f"❌ 获取工具列表失败: {text}") | ||||
|                     return None | ||||
|                      | ||||
|         except Exception as e: | ||||
|             print(f"❌ 测试工具列表时出错: {e}") | ||||
|             return None | ||||
|      | ||||
|     async def test_mcp_tool_call(self, mcp_alias: str, tool_name: str, arguments: Dict[str, Any]) -> Optional[Dict[str, Any]]: | ||||
|         """测试MCP工具调用""" | ||||
|         try: | ||||
|             headers = { | ||||
|                 "Authorization": f"Bearer {self.master_key}", | ||||
|                 "Content-Type": "application/json" | ||||
|             } | ||||
|              | ||||
|             # 构造JSON-RPC请求 | ||||
|             jsonrpc_request = { | ||||
|                 "jsonrpc": "2.0", | ||||
|                 "method": "tools/call", | ||||
|                 "params": { | ||||
|                     "name": tool_name, | ||||
|                     "arguments": arguments | ||||
|                 }, | ||||
|                 "id": 2 | ||||
|             } | ||||
|              | ||||
|             async with self.session.post( | ||||
|                 f"{self.litellm_base_url}/mcp/{mcp_alias}", | ||||
|                 headers=headers, | ||||
|                 json=jsonrpc_request | ||||
|             ) as response: | ||||
|                 print(f"工具调用请求状态: {response.status}") | ||||
|                  | ||||
|                 if response.status == 200: | ||||
|                     result = await response.json() | ||||
|                     print(f"✅ MCP {mcp_alias} 工具调用结果: {json.dumps(result, indent=2)}") | ||||
|                     return result | ||||
|                 else: | ||||
|                     text = await response.text() | ||||
|                     print(f"❌ 工具调用失败: {text}") | ||||
|                     return None | ||||
|                      | ||||
|         except Exception as e: | ||||
|             print(f"❌ 测试工具调用时出错: {e}") | ||||
|             return None | ||||
|      | ||||
|     async def test_direct_mcp_server(self, url: str) -> bool: | ||||
|         """直接测试MCP服务器""" | ||||
|         try: | ||||
|             print(f"\n🔍 直接测试MCP服务器: {url}") | ||||
|              | ||||
|             # 测试初始化 | ||||
|             async with self.session.get(url) as response: | ||||
|                 print(f"直接MCP服务器响应状态: {response.status}") | ||||
|                  | ||||
|                 if response.status == 200: | ||||
|                     content_type = response.headers.get('content-type', '') | ||||
|                     if 'text/event-stream' in content_type: | ||||
|                         async for line in response.content: | ||||
|                             line_str = line.decode('utf-8').strip() | ||||
|                             if line_str.startswith('data: '): | ||||
|                                 data = line_str[6:] | ||||
|                                 try: | ||||
|                                     parsed_data = json.loads(data) | ||||
|                                     print(f"✅ 直接MCP服务器SSE响应: {json.dumps(parsed_data, indent=2)}") | ||||
|                                     return True | ||||
|                                 except json.JSONDecodeError: | ||||
|                                     print(f"⚠️  无法解析SSE数据: {data}") | ||||
|                                 break | ||||
|                     else: | ||||
|                         text = await response.text() | ||||
|                         print(f"✅ 直接MCP服务器响应: {text}") | ||||
|                         return True | ||||
|                 else: | ||||
|                     text = await response.text() | ||||
|                     print(f"❌ 直接MCP服务器请求失败: {text}") | ||||
|                     return False | ||||
|                      | ||||
|         except Exception as e: | ||||
|             print(f"❌ 直接测试MCP服务器时出错: {e}") | ||||
|             return False | ||||
|      | ||||
|     async def run_comprehensive_test(self): | ||||
|         """运行综合测试""" | ||||
|         print("🚀 开始MCP综合测试\n") | ||||
|          | ||||
|         # 1. 测试LiteLLM健康状态 | ||||
|         print("1️⃣ 测试LiteLLM服务器健康状态") | ||||
|         health_ok = await self.test_litellm_health() | ||||
|          | ||||
|         if not health_ok: | ||||
|             print("❌ LiteLLM服务器不可用,停止测试") | ||||
|             return | ||||
|          | ||||
|         # 2. 测试本地MCP服务器 | ||||
|         print("\n2️⃣ 测试本地MCP服务器") | ||||
|         await self.test_direct_mcp_server("http://localhost:8080/mcp") | ||||
|          | ||||
|         # 3. 测试通过LiteLLM访问本地MCP | ||||
|         print("\n3️⃣ 测试通过LiteLLM访问本地MCP") | ||||
|         test_endpoint_ok = await self.test_mcp_endpoint_direct("test") | ||||
|          | ||||
|         if test_endpoint_ok: | ||||
|             # 4. 测试工具列表 | ||||
|             print("\n4️⃣ 测试本地MCP工具列表") | ||||
|             tools_result = await self.test_mcp_tools_list("test") | ||||
|              | ||||
|             if tools_result and 'result' in tools_result and 'tools' in tools_result['result']: | ||||
|                 tools = tools_result['result']['tools'] | ||||
|                 print(f"发现 {len(tools)} 个工具") | ||||
|                  | ||||
|                 # 5. 测试工具调用 | ||||
|                 print("\n5️⃣ 测试工具调用") | ||||
|                 for tool in tools[:3]:  # 测试前3个工具 | ||||
|                     tool_name = tool['name'] | ||||
|                     print(f"\n测试工具: {tool_name}") | ||||
|                      | ||||
|                     if tool_name == "echo": | ||||
|                         await self.test_mcp_tool_call("test", "echo", {"message": "Hello MCP!"}) | ||||
|                     elif tool_name == "get_time": | ||||
|                         await self.test_mcp_tool_call("test", "get_time", {}) | ||||
|                     elif tool_name == "calculate": | ||||
|                         await self.test_mcp_tool_call("test", "calculate", {"expression": "2+2*3"}) | ||||
|          | ||||
|         # 6. 测试DeepWiki MCP | ||||
|         print("\n6️⃣ 测试DeepWiki MCP") | ||||
|         await self.test_mcp_endpoint_direct("deepwiki") | ||||
|          | ||||
|         print("\n🎉 MCP综合测试完成") | ||||
| 
 | ||||
| async def main(): | ||||
|     """主函数""" | ||||
|     async with MCPTester() as tester: | ||||
|         await tester.run_comprehensive_test() | ||||
| 
 | ||||
| if __name__ == "__main__": | ||||
|     asyncio.run(main()) | ||||
|  | @ -0,0 +1,26 @@ | |||
| model_list: | ||||
|   - model_name: test-model | ||||
|     litellm_params: | ||||
|       model: openai/gpt-3.5-turbo | ||||
|       api_key: sk-test-key | ||||
| 
 | ||||
| general_settings: | ||||
|   master_key: sk-1234567890abcdef | ||||
|   disable_spend_logs: false | ||||
|   disable_master_key_return: false | ||||
|   enforce_user_param: false | ||||
|    | ||||
| litellm_settings: | ||||
|   set_verbose: true | ||||
|   drop_params: true | ||||
|   add_function_to_prompt: true | ||||
|   mcp_aliases: | ||||
|     "deepwiki": "deepwiki_mcp_server" | ||||
|     "test": "test_mcp_server" | ||||
|   mcp_servers: | ||||
|     deepwiki_mcp_server: | ||||
|       url: "https://mcp.api-inference.modelscope.net/f9d3f201909c45/sse" | ||||
|       transport: "http" | ||||
|     test_mcp_server: | ||||
|       url: "http://localhost:8080/mcp" | ||||
|       transport: "http" | ||||
|  | @ -0,0 +1,119 @@ | |||
| #!/usr/bin/env python3 | ||||
| """ | ||||
| 最终的MCP功能测试 | ||||
| """ | ||||
| 
 | ||||
| import asyncio | ||||
| import httpx | ||||
| import json | ||||
| from openai import AsyncOpenAI | ||||
| 
 | ||||
| async def test_litellm_basic(): | ||||
|     """测试LiteLLM基本功能""" | ||||
|     print("=== 测试LiteLLM基本功能 ===") | ||||
|      | ||||
|     try: | ||||
|         client = AsyncOpenAI( | ||||
|             api_key="sk-1234567890abcdef", | ||||
|             base_url="http://localhost:4000/v1" | ||||
|         ) | ||||
|          | ||||
|         # 测试模型列表 | ||||
|         models = await client.models.list() | ||||
|         print(f"可用模型: {[model.id for model in models.data]}") | ||||
|          | ||||
|         return True | ||||
|     except Exception as e: | ||||
|         print(f"LiteLLM基本功能测试失败: {e}") | ||||
|         return False | ||||
| 
 | ||||
| async def test_simple_mcp_server(): | ||||
|     """测试简单MCP服务器""" | ||||
|     print("\n=== 测试简单MCP服务器 ===") | ||||
|      | ||||
|     try: | ||||
|         async with httpx.AsyncClient() as client: | ||||
|             response = await client.get( | ||||
|                 "http://localhost:8080/mcp", | ||||
|                 headers={"Accept": "text/event-stream"}, | ||||
|                 timeout=5.0 | ||||
|             ) | ||||
|              | ||||
|             if response.status_code == 200: | ||||
|                 content = response.text | ||||
|                 print(f"MCP服务器响应: {content}") | ||||
|                  | ||||
|                 # 尝试解析JSON | ||||
|                 if "data:" in content: | ||||
|                     json_part = content.split("data:")[1].strip() | ||||
|                     data = json.loads(json_part) | ||||
|                     print(f"解析的工具: {data.get('result', {}).get('tools', [])}") | ||||
|                     return True | ||||
|             else: | ||||
|                 print(f"MCP服务器返回错误: {response.status_code}") | ||||
|                 return False | ||||
|                  | ||||
|     except Exception as e: | ||||
|         print(f"简单MCP服务器测试失败: {e}") | ||||
|         return False | ||||
| 
 | ||||
| async def test_litellm_mcp_integration(): | ||||
|     """测试LiteLLM与MCP的集成""" | ||||
|     print("\n=== 测试LiteLLM MCP集成 ===") | ||||
|      | ||||
|     try: | ||||
|         async with httpx.AsyncClient() as client: | ||||
|             # 尝试不同的MCP端点 | ||||
|             endpoints = [ | ||||
|                 "http://localhost:4000/mcp/test", | ||||
|                 "http://localhost:4000/mcp/tools", | ||||
|                 "http://localhost:4000/v1/mcp" | ||||
|             ] | ||||
|              | ||||
|             for endpoint in endpoints: | ||||
|                 try: | ||||
|                     print(f"测试端点: {endpoint}") | ||||
|                     response = await client.get( | ||||
|                         endpoint, | ||||
|                         headers={ | ||||
|                             "Authorization": "Bearer sk-1234567890abcdef", | ||||
|                             "Accept": "text/event-stream" | ||||
|                         }, | ||||
|                         timeout=3.0 | ||||
|                     ) | ||||
|                     print(f"状态码: {response.status_code}") | ||||
|                     if response.status_code == 200: | ||||
|                         print(f"响应: {response.text[:200]}...") | ||||
|                         return True | ||||
|                 except Exception as e: | ||||
|                     print(f"端点 {endpoint} 失败: {e}") | ||||
|                      | ||||
|         return False | ||||
|          | ||||
|     except Exception as e: | ||||
|         print(f"LiteLLM MCP集成测试失败: {e}") | ||||
|         return False | ||||
| 
 | ||||
| async def main(): | ||||
|     """主测试函数""" | ||||
|     print("开始MCP功能综合测试...\n") | ||||
|      | ||||
|     # 测试各个组件 | ||||
|     litellm_ok = await test_litellm_basic() | ||||
|     mcp_server_ok = await test_simple_mcp_server() | ||||
|     integration_ok = await test_litellm_mcp_integration() | ||||
|      | ||||
|     print("\n=== 测试结果总结 ===") | ||||
|     print(f"LiteLLM基本功能: {'✓' if litellm_ok else '✗'}") | ||||
|     print(f"简单MCP服务器: {'✓' if mcp_server_ok else '✗'}") | ||||
|     print(f"LiteLLM MCP集成: {'✓' if integration_ok else '✗'}") | ||||
|      | ||||
|     if litellm_ok and mcp_server_ok: | ||||
|         print("\n结论: LiteLLM和MCP服务器都正常工作,但LiteLLM的MCP集成可能需要额外配置。") | ||||
|     elif litellm_ok: | ||||
|         print("\n结论: LiteLLM正常工作,但MCP功能有问题。") | ||||
|     else: | ||||
|         print("\n结论: LiteLLM基本功能有问题。") | ||||
| 
 | ||||
| if __name__ == "__main__": | ||||
|     asyncio.run(main()) | ||||
|  | @ -0,0 +1,64 @@ | |||
| import asyncio | ||||
| from openai import AsyncOpenAI | ||||
| 
 | ||||
| async def list_available_models(): | ||||
|     """获取LiteLLM服务器支持的模型列表""" | ||||
|     print("正在获取可用模型列表...") | ||||
|      | ||||
|     # 使用远程LiteLLM服务器 | ||||
|     client = AsyncOpenAI( | ||||
|         api_key="sk-0jdcGHZJpX2oUJmyEs7zVA",  | ||||
|         base_url="https://litellm.seekkey.tech" | ||||
|     ) | ||||
|      | ||||
|     try: | ||||
|         # 获取模型列表 | ||||
|         models = await client.models.list() | ||||
|          | ||||
|         print("\n=== 可用模型列表 ===") | ||||
|         for model in models.data: | ||||
|             print(f"- {model.id}") | ||||
|              | ||||
|         print(f"\n总共找到 {len(models.data)} 个模型") | ||||
|          | ||||
|         # 尝试调用一个简单的模型 | ||||
|         if models.data: | ||||
|             first_model = models.data[0].id | ||||
|             print(f"\n正在测试第一个模型: {first_model}") | ||||
|              | ||||
|             response = await client.chat.completions.create( | ||||
|                 model=first_model, | ||||
|                 messages=[ | ||||
|                     {"role": "user", "content": "Hello, please say hi in Chinese."} | ||||
|                 ], | ||||
|                 max_tokens=50 | ||||
|             ) | ||||
|              | ||||
|             print(f"测试响应: {response.choices[0].message.content}") | ||||
|          | ||||
|     except Exception as e: | ||||
|         print(f"获取模型列表失败: {e}") | ||||
|         print(f"错误类型: {type(e).__name__}") | ||||
|          | ||||
|         # 尝试直接测试一些常见模型 | ||||
|         common_models = ["gpt-4", "gpt-3.5-turbo", "gemini-pro", "claude-3-sonnet"] | ||||
|         print("\n尝试测试常见模型...") | ||||
|          | ||||
|         for model in common_models: | ||||
|             try: | ||||
|                 print(f"测试模型: {model}") | ||||
|                 response = await client.chat.completions.create( | ||||
|                     model=model, | ||||
|                     messages=[{"role": "user", "content": "Hi"}], | ||||
|                     max_tokens=10 | ||||
|                 ) | ||||
|                 print(f"✓ {model} 可用") | ||||
|                 break | ||||
|             except Exception as model_error: | ||||
|                 print(f"✗ {model} 不可用: {str(model_error)[:100]}...") | ||||
|      | ||||
|     finally: | ||||
|         await client.close() | ||||
| 
 | ||||
| if __name__ == "__main__": | ||||
|     asyncio.run(list_available_models()) | ||||
|  | @ -0,0 +1,239 @@ | |||
| #!/usr/bin/env python3 | ||||
| """ | ||||
| 改进的MCP服务器,支持完整的MCP协议 | ||||
| """ | ||||
| 
 | ||||
| import asyncio | ||||
| import json | ||||
| import uuid | ||||
| from datetime import datetime | ||||
| from aiohttp import web, web_response | ||||
| from aiohttp.web import Request, Response | ||||
| 
 | ||||
| # MCP服务器状态 | ||||
| server_info = { | ||||
|     "name": "test-mcp-server", | ||||
|     "version": "1.0.0", | ||||
|     "protocol_version": "2024-11-05" | ||||
| } | ||||
| 
 | ||||
| # 可用工具定义 | ||||
| available_tools = [ | ||||
|     { | ||||
|         "name": "echo", | ||||
|         "description": "Echo back the input message", | ||||
|         "inputSchema": { | ||||
|             "type": "object", | ||||
|             "properties": { | ||||
|                 "message": { | ||||
|                     "type": "string", | ||||
|                     "description": "Message to echo back" | ||||
|                 } | ||||
|             }, | ||||
|             "required": ["message"] | ||||
|         } | ||||
|     }, | ||||
|     { | ||||
|         "name": "get_time", | ||||
|         "description": "Get current time", | ||||
|         "inputSchema": { | ||||
|             "type": "object", | ||||
|             "properties": {}, | ||||
|             "additionalProperties": False | ||||
|         } | ||||
|     }, | ||||
|     { | ||||
|         "name": "calculate", | ||||
|         "description": "Perform basic arithmetic calculations", | ||||
|         "inputSchema": { | ||||
|             "type": "object", | ||||
|             "properties": { | ||||
|                 "expression": { | ||||
|                     "type": "string", | ||||
|                     "description": "Mathematical expression to evaluate (e.g., '2+2', '10*5')" | ||||
|                 } | ||||
|             }, | ||||
|             "required": ["expression"] | ||||
|         } | ||||
|     } | ||||
| ] | ||||
| 
 | ||||
| async def handle_mcp_request(request: Request) -> Response: | ||||
|     """处理MCP请求""" | ||||
|     print(f"收到MCP请求: {request.method} {request.path}") | ||||
|     print(f"请求头: {dict(request.headers)}") | ||||
|      | ||||
|     if request.method == "GET": | ||||
|         # 处理初始化请求 | ||||
|         return await handle_initialize(request) | ||||
|     elif request.method == "POST": | ||||
|         # 处理JSON-RPC请求 | ||||
|         return await handle_jsonrpc(request) | ||||
|      | ||||
|     return web_response.Response(status=405, text="Method not allowed") | ||||
| 
 | ||||
| async def handle_initialize(request: Request) -> Response: | ||||
|     """处理初始化请求""" | ||||
|     init_response = { | ||||
|         "jsonrpc": "2.0", | ||||
|         "result": { | ||||
|             "protocolVersion": server_info["protocol_version"], | ||||
|             "capabilities": { | ||||
|                 "tools": { | ||||
|                     "listChanged": True | ||||
|                 }, | ||||
|                 "resources": { | ||||
|                     "subscribe": False, | ||||
|                     "listChanged": False | ||||
|                 }, | ||||
|                 "prompts": { | ||||
|                     "listChanged": False | ||||
|                 }, | ||||
|                 "logging": {} | ||||
|             }, | ||||
|             "serverInfo": { | ||||
|                 "name": server_info["name"], | ||||
|                 "version": server_info["version"] | ||||
|             } | ||||
|         }, | ||||
|         "id": 1 | ||||
|     } | ||||
|      | ||||
|     # 返回SSE格式的响应 | ||||
|     response_text = f"data: {json.dumps(init_response)}\n\n" | ||||
|      | ||||
|     return web_response.Response( | ||||
|         text=response_text, | ||||
|         content_type="text/event-stream", | ||||
|         headers={ | ||||
|             "Cache-Control": "no-cache", | ||||
|             "Connection": "keep-alive", | ||||
|             "Access-Control-Allow-Origin": "*", | ||||
|             "Access-Control-Allow-Methods": "GET, POST, OPTIONS", | ||||
|             "Access-Control-Allow-Headers": "*" | ||||
|         } | ||||
|     ) | ||||
| 
 | ||||
| async def handle_jsonrpc(request: Request) -> Response: | ||||
|     """处理JSON-RPC请求""" | ||||
|     try: | ||||
|         body = await request.text() | ||||
|         print(f"收到JSON-RPC请求体: {body}") | ||||
|          | ||||
|         if not body: | ||||
|             return web_response.Response(status=400, text="Empty request body") | ||||
|              | ||||
|         data = json.loads(body) | ||||
|         method = data.get("method") | ||||
|         params = data.get("params", {}) | ||||
|         request_id = data.get("id") | ||||
|          | ||||
|         print(f"方法: {method}, 参数: {params}") | ||||
|          | ||||
|         if method == "tools/list": | ||||
|             response = { | ||||
|                 "jsonrpc": "2.0", | ||||
|                 "result": { | ||||
|                     "tools": available_tools | ||||
|                 }, | ||||
|                 "id": request_id | ||||
|             } | ||||
|         elif method == "tools/call": | ||||
|             tool_name = params.get("name") | ||||
|             tool_arguments = params.get("arguments", {}) | ||||
|              | ||||
|             result = await execute_tool(tool_name, tool_arguments) | ||||
|              | ||||
|             response = { | ||||
|                 "jsonrpc": "2.0", | ||||
|                 "result": { | ||||
|                     "content": [ | ||||
|                         { | ||||
|                             "type": "text", | ||||
|                             "text": result | ||||
|                         } | ||||
|                     ] | ||||
|                 }, | ||||
|                 "id": request_id | ||||
|             } | ||||
|         else: | ||||
|             response = { | ||||
|                 "jsonrpc": "2.0", | ||||
|                 "error": { | ||||
|                     "code": -32601, | ||||
|                     "message": f"Method not found: {method}" | ||||
|                 }, | ||||
|                 "id": request_id | ||||
|             } | ||||
|          | ||||
|         return web_response.Response( | ||||
|             text=json.dumps(response), | ||||
|             content_type="application/json", | ||||
|             headers={ | ||||
|                 "Access-Control-Allow-Origin": "*", | ||||
|                 "Access-Control-Allow-Methods": "GET, POST, OPTIONS", | ||||
|                 "Access-Control-Allow-Headers": "*" | ||||
|             } | ||||
|         ) | ||||
|          | ||||
|     except json.JSONDecodeError as e: | ||||
|         print(f"JSON解析错误: {e}") | ||||
|         return web_response.Response(status=400, text="Invalid JSON") | ||||
|     except Exception as e: | ||||
|         print(f"处理请求时出错: {e}") | ||||
|         return web_response.Response(status=500, text="Internal server error") | ||||
| 
 | ||||
| async def execute_tool(tool_name: str, arguments: dict) -> str: | ||||
|     """执行工具调用""" | ||||
|     print(f"执行工具: {tool_name}, 参数: {arguments}") | ||||
|      | ||||
|     if tool_name == "echo": | ||||
|         message = arguments.get("message", "") | ||||
|         return f"Echo: {message}" | ||||
|      | ||||
|     elif tool_name == "get_time": | ||||
|         current_time = datetime.now().strftime("%Y-%m-%d %H:%M:%S") | ||||
|         return f"Current time: {current_time}" | ||||
|      | ||||
|     elif tool_name == "calculate": | ||||
|         expression = arguments.get("expression", "") | ||||
|         try: | ||||
|             # 简单的数学表达式计算(仅支持基本运算) | ||||
|             # 注意:这里使用eval有安全风险,实际应用中应该使用更安全的方法 | ||||
|             allowed_chars = set('0123456789+-*/.() ') | ||||
|             if all(c in allowed_chars for c in expression): | ||||
|                 result = eval(expression) | ||||
|                 return f"Result: {expression} = {result}" | ||||
|             else: | ||||
|                 return "Error: Invalid characters in expression" | ||||
|         except Exception as e: | ||||
|             return f"Error calculating expression: {str(e)}" | ||||
|      | ||||
|     else: | ||||
|         return f"Error: Unknown tool '{tool_name}'" | ||||
| 
 | ||||
| async def handle_options(request: Request) -> Response: | ||||
|     """处理OPTIONS请求""" | ||||
|     return web_response.Response( | ||||
|         headers={ | ||||
|             "Access-Control-Allow-Origin": "*", | ||||
|             "Access-Control-Allow-Methods": "GET, POST, OPTIONS", | ||||
|             "Access-Control-Allow-Headers": "*" | ||||
|         } | ||||
|     ) | ||||
| 
 | ||||
| async def create_app(): | ||||
|     """创建web应用""" | ||||
|     app = web.Application() | ||||
|      | ||||
|     # 添加路由 | ||||
|     app.router.add_get('/mcp', handle_mcp_request) | ||||
|     app.router.add_post('/mcp', handle_mcp_request) | ||||
|     app.router.add_options('/mcp', handle_options) | ||||
|      | ||||
|     return app | ||||
| 
 | ||||
| if __name__ == '__main__': | ||||
|     print("启动简单MCP服务器在端口8080...") | ||||
|     app = asyncio.run(create_app()) | ||||
|     web.run_app(app, host='localhost', port=8080) | ||||
|  | @ -0,0 +1,100 @@ | |||
| #!/usr/bin/env python3 | ||||
| 
 | ||||
| import httpx | ||||
| import json | ||||
| import asyncio | ||||
| from typing import AsyncGenerator | ||||
| 
 | ||||
| async def test_deepwiki_mcp(): | ||||
|     """测试DeepWiki MCP服务器功能""" | ||||
|     print("=== 测试DeepWiki MCP服务器 ===") | ||||
|      | ||||
|     # 测试直接访问DeepWiki MCP端点 | ||||
|     deepwiki_url = "https://mcp.api-inference.modelscope.net/f9d3f201909c45/sse" | ||||
|      | ||||
|     try: | ||||
|         async with httpx.AsyncClient(timeout=30.0) as client: | ||||
|             print(f"\n1. 测试直接访问DeepWiki MCP端点: {deepwiki_url}") | ||||
|              | ||||
|             # 发送SSE请求 | ||||
|             headers = { | ||||
|                 "Accept": "text/event-stream", | ||||
|                 "Cache-Control": "no-cache" | ||||
|             } | ||||
|              | ||||
|             async with client.stream("GET", deepwiki_url, headers=headers) as response: | ||||
|                 print(f"状态码: {response.status_code}") | ||||
|                 print(f"响应头: {dict(response.headers)}") | ||||
|                  | ||||
|                 if response.status_code == 200: | ||||
|                     print("\n接收到的数据:") | ||||
|                     count = 0 | ||||
|                     async for line in response.aiter_lines(): | ||||
|                         if line.strip(): | ||||
|                             print(f"Line {count}: {line}") | ||||
|                             count += 1 | ||||
|                             if count >= 10:  # 限制输出行数 | ||||
|                                 print("... (更多数据被截断)") | ||||
|                                 break | ||||
|                 else: | ||||
|                     print(f"请求失败: {response.status_code}") | ||||
|                     print(await response.aread()) | ||||
|                      | ||||
|     except Exception as e: | ||||
|         print(f"直接访问DeepWiki失败: {e}") | ||||
|      | ||||
|     # 测试通过LiteLLM访问DeepWiki MCP | ||||
|     print("\n\n2. 测试通过LiteLLM访问DeepWiki MCP") | ||||
|     litellm_mcp_url = "http://localhost:4000/mcp/deepwiki" | ||||
|      | ||||
|     try: | ||||
|         async with httpx.AsyncClient(timeout=30.0) as client: | ||||
|             headers = { | ||||
|                 "Accept": "text/event-stream", | ||||
|                 "Cache-Control": "no-cache" | ||||
|             } | ||||
|              | ||||
|             async with client.stream("GET", litellm_mcp_url, headers=headers) as response: | ||||
|                 print(f"状态码: {response.status_code}") | ||||
|                 print(f"响应头: {dict(response.headers)}") | ||||
|                  | ||||
|                 if response.status_code == 200: | ||||
|                     print("\n接收到的数据:") | ||||
|                     count = 0 | ||||
|                     async for line in response.aiter_lines(): | ||||
|                         if line.strip(): | ||||
|                             print(f"Line {count}: {line}") | ||||
|                             count += 1 | ||||
|                             if count >= 10: | ||||
|                                 print("... (更多数据被截断)") | ||||
|                                 break | ||||
|                 else: | ||||
|                     print(f"请求失败: {response.status_code}") | ||||
|                     error_content = await response.aread() | ||||
|                     print(f"错误内容: {error_content}") | ||||
|                      | ||||
|     except Exception as e: | ||||
|         print(f"通过LiteLLM访问DeepWiki失败: {e}") | ||||
|      | ||||
|     # 测试LiteLLM的基本MCP端点 | ||||
|     print("\n\n3. 测试LiteLLM的基本MCP端点") | ||||
|     basic_endpoints = [ | ||||
|         "http://localhost:4000/mcp/", | ||||
|         "http://localhost:4000/mcp", | ||||
|         "http://localhost:4000/v1/mcp" | ||||
|     ] | ||||
|      | ||||
|     for endpoint in basic_endpoints: | ||||
|         try: | ||||
|             async with httpx.AsyncClient(timeout=10.0) as client: | ||||
|                 response = await client.get(endpoint) | ||||
|                 print(f"\n{endpoint}: {response.status_code}") | ||||
|                 if response.status_code != 200: | ||||
|                     print(f"错误: {response.text[:200]}") | ||||
|                 else: | ||||
|                     print(f"成功: {response.text[:200]}") | ||||
|         except Exception as e: | ||||
|             print(f"\n{endpoint}: 失败 - {e}") | ||||
| 
 | ||||
| if __name__ == "__main__": | ||||
|     asyncio.run(test_deepwiki_mcp()) | ||||
|  | @ -0,0 +1,58 @@ | |||
| import asyncio | ||||
| from openai import AsyncOpenAI | ||||
| 
 | ||||
| async def test_gpt5_nano(): | ||||
|     """测试调用LiteLLM的gpt5-nano模型""" | ||||
|     print("正在测试gpt5-nano模型...") | ||||
|      | ||||
|     # 使用远程LiteLLM服务器 | ||||
|     client = AsyncOpenAI( | ||||
|         api_key="sk-0jdcGHZJpX2oUJmyEs7zVA",  | ||||
|         base_url="https://litellm.seekkey.tech" | ||||
|     ) | ||||
|      | ||||
|     try: | ||||
|         # 调用gpt-5-nano模型 | ||||
|         response = await client.chat.completions.create( | ||||
|             model="gpt-5-nano", | ||||
|             messages=[ | ||||
|                 {"role": "user", "content": "你好,请简单介绍一下你自己。"} | ||||
|             ], | ||||
|             max_completion_tokens=150, | ||||
|             temperature=0.7 | ||||
|         ) | ||||
|          | ||||
|         print("\n=== GPT-5-Nano 响应 ===") | ||||
|         print(f"模型: {response.model}") | ||||
|         print(f"响应内容: {response.choices[0].message.content}") | ||||
|         print(f"Token使用: {response.usage.total_tokens if response.usage else 'N/A'}") | ||||
|          | ||||
|     except Exception as e: | ||||
|         print(f"调用失败: {e}") | ||||
|         print(f"错误类型: {type(e).__name__}") | ||||
|         import traceback | ||||
|         print(f"详细错误信息: {traceback.format_exc()}") | ||||
|          | ||||
|         # 尝试使用其他可用模型 | ||||
|         print("\n尝试使用其他模型...") | ||||
|         try: | ||||
|             response = await client.chat.completions.create( | ||||
|                 model="fireworks_ai/accounts/fireworks/models/deepseek-v3-0324", | ||||
|                 messages=[ | ||||
|                     {"role": "user", "content": "你好,请简单介绍一下你自己。"} | ||||
|                 ], | ||||
|                 max_tokens=150, | ||||
|                 temperature=0.7 | ||||
|             ) | ||||
|             print("\n=== DeepSeek-V3 响应 ===") | ||||
|             print(f"模型: {response.model}") | ||||
|             print(f"响应内容: {response.choices[0].message.content}") | ||||
|             print(f"Token使用: {response.usage.total_tokens if response.usage else 'N/A'}") | ||||
|         except Exception as fallback_error: | ||||
|             print(f"备用模型也失败: {fallback_error}") | ||||
|      | ||||
|     finally: | ||||
|         await client.close() | ||||
| 
 | ||||
| if __name__ == "__main__": | ||||
|     asyncio.run(test_gpt5_nano()) | ||||
|  | @ -0,0 +1,66 @@ | |||
| import asyncio | ||||
| import sys | ||||
| from openai import AsyncOpenAI | ||||
| from openai.types.chat import ChatCompletionUserMessageParam | ||||
| from mcp import ClientSession | ||||
| from mcp.client.sse import sse_client | ||||
| 
 | ||||
| 
 | ||||
| async def main(): | ||||
|     print("测试LiteLLM的MCP功能...") | ||||
|      | ||||
|     try: | ||||
|         # Initialize OpenAI client | ||||
|         print("初始化OpenAI客户端...") | ||||
|         client = AsyncOpenAI(api_key="sk-1234", base_url="http://localhost:4000") | ||||
|         print("OpenAI客户端初始化完成") | ||||
| 
 | ||||
|         # Test basic LiteLLM functionality first | ||||
|         print("测试基本的LiteLLM功能...") | ||||
|         response = await client.chat.completions.create( | ||||
|             model="gemini-flash", | ||||
|             messages=[ | ||||
|                 {"role": "user", "content": "Hello, this is a test message."} | ||||
|             ] | ||||
|         ) | ||||
|         print(f"LiteLLM响应: {response.choices[0].message.content}") | ||||
|          | ||||
|         # Now test MCP endpoint | ||||
|         print("\n测试MCP端点...") | ||||
|          | ||||
|         # 添加超时处理 | ||||
|         try: | ||||
|             async with asyncio.timeout(10):  # 10秒超时 | ||||
|                 print("正在建立SSE连接到 /mcp/ 端点...") | ||||
|                 async with sse_client("http://localhost:4000/mcp/") as (read, write): | ||||
|                     print("SSE连接建立成功,初始化会话...") | ||||
|                     async with ClientSession(read, write) as session: | ||||
|                         print("正在初始化MCP会话...") | ||||
|                         await session.initialize() | ||||
|                         print("MCP会话初始化成功!") | ||||
| 
 | ||||
|                         # List available tools | ||||
|                         print("获取可用工具列表...") | ||||
|                         tools = await session.list_tools() | ||||
|                         print(f"找到 {len(tools.tools)} 个工具:") | ||||
|                         for tool in tools.tools: | ||||
|                             print(f"  - {tool.name}: {tool.description}") | ||||
|                              | ||||
|         except asyncio.TimeoutError: | ||||
|             print("MCP连接超时!") | ||||
|             print("这可能意味着:") | ||||
|             print("1. LiteLLM版本不支持MCP功能") | ||||
|             print("2. MCP功能需要额外配置") | ||||
|             print("3. /mcp/ 端点不存在") | ||||
|             return | ||||
|              | ||||
|     except Exception as e: | ||||
|         print(f"发生错误: {type(e).__name__}: {e}") | ||||
|         import traceback | ||||
|         traceback.print_exc() | ||||
| 
 | ||||
| 
 | ||||
| if __name__ == "__main__": | ||||
|     print("启动LiteLLM MCP测试...") | ||||
|     asyncio.run(main()) | ||||
|     print("测试完成") | ||||
|  | @ -0,0 +1,49 @@ | |||
| import asyncio | ||||
| import httpx | ||||
| import json | ||||
| 
 | ||||
| async def test_mcp_detailed(): | ||||
|     print("详细测试LiteLLM的MCP端点...") | ||||
|      | ||||
|     async with httpx.AsyncClient() as client: | ||||
|         try: | ||||
|             print("\n测试端点: http://localhost:4000/mcp/") | ||||
|              | ||||
|             # 使用流式请求来处理SSE | ||||
|             async with client.stream( | ||||
|                 "GET", | ||||
|                 "http://localhost:4000/mcp/", | ||||
|                 headers={ | ||||
|                     "Authorization": "Bearer sk-1234567890abcdef", | ||||
|                     "Accept": "text/event-stream", | ||||
|                     "Cache-Control": "no-cache" | ||||
|                 }, | ||||
|                 timeout=10.0 | ||||
|             ) as response: | ||||
|                 print(f"状态码: {response.status_code}") | ||||
|                 print(f"响应头: {dict(response.headers)}") | ||||
|                  | ||||
|                 if response.status_code == 200: | ||||
|                     print("开始读取SSE流...") | ||||
|                     content = "" | ||||
|                     async for chunk in response.aiter_text(): | ||||
|                         content += chunk | ||||
|                         print(f"收到数据块: {repr(chunk)}") | ||||
|                          | ||||
|                         # 如果收到足够的数据就停止 | ||||
|                         if len(content) > 1000: | ||||
|                             print("收到足够数据,停止读取") | ||||
|                             break | ||||
|                      | ||||
|                     print(f"\n完整内容: {content}") | ||||
|                 else: | ||||
|                     error_content = await response.aread() | ||||
|                     print(f"错误响应: {error_content.decode()}") | ||||
|                      | ||||
|         except Exception as e: | ||||
|             print(f"请求失败: {type(e).__name__}: {e}") | ||||
|             import traceback | ||||
|             traceback.print_exc() | ||||
| 
 | ||||
| if __name__ == "__main__": | ||||
|     asyncio.run(test_mcp_detailed()) | ||||
|  | @ -0,0 +1,39 @@ | |||
| import asyncio | ||||
| import httpx | ||||
| 
 | ||||
| async def test_mcp_endpoint(): | ||||
|     print("测试LiteLLM的MCP端点...") | ||||
|      | ||||
|     # Test different endpoints | ||||
|     endpoints = [ | ||||
|         "http://localhost:4000/health", | ||||
|         "http://localhost:4000/v1/models",  | ||||
|         "http://localhost:4000/mcp/", | ||||
|         "http://localhost:4000/mcp" | ||||
|     ] | ||||
|      | ||||
|     async with httpx.AsyncClient() as client: | ||||
|         for endpoint in endpoints: | ||||
|             try: | ||||
|                 print(f"\n测试端点: {endpoint}") | ||||
|                 response = await client.get( | ||||
|                     endpoint, | ||||
|                     headers={ | ||||
|                         "Authorization": "Bearer sk-1234567890abcdef", | ||||
|                         "Accept": "text/event-stream" | ||||
|                     }, | ||||
|                     timeout=5.0 | ||||
|                 ) | ||||
|                 print(f"状态码: {response.status_code}") | ||||
|                 print(f"响应头: {dict(response.headers)}") | ||||
|                 if response.status_code == 200: | ||||
|                     content = response.text[:500]  # 只显示前500字符 | ||||
|                     print(f"响应内容: {content}") | ||||
|                 else: | ||||
|                     print(f"错误响应: {response.text}") | ||||
|                      | ||||
|             except Exception as e: | ||||
|                 print(f"请求失败: {type(e).__name__}: {e}") | ||||
| 
 | ||||
| if __name__ == "__main__": | ||||
|     asyncio.run(test_mcp_endpoint()) | ||||
|  | @ -0,0 +1,28 @@ | |||
| import asyncio | ||||
| from openai import AsyncOpenAI | ||||
| 
 | ||||
| async def main(): | ||||
|     # Test remote LiteLLM server without MCP | ||||
|     client = AsyncOpenAI( | ||||
|         api_key="sk-0jdcGHZJpX2oUJmyEs7zVA",  | ||||
|         base_url="https://litellm.seekkey.tech" | ||||
|     ) | ||||
|      | ||||
|     try: | ||||
|         # Test simple chat completion | ||||
|         response = await client.chat.completions.create( | ||||
|             model="gemini/gemini-2.5-flash", | ||||
|             messages=[ | ||||
|                 {"role": "user", "content": "Hello! Please respond with a simple greeting."} | ||||
|             ], | ||||
|             max_tokens=50 | ||||
|         ) | ||||
|          | ||||
|         print("✅ Remote LiteLLM server is working!") | ||||
|         print(f"Response: {response.choices[0].message.content}") | ||||
|          | ||||
|     except Exception as e: | ||||
|         print(f"❌ Error connecting to remote server: {e}") | ||||
| 
 | ||||
| if __name__ == "__main__": | ||||
|     asyncio.run(main()) | ||||
|  | @ -0,0 +1,72 @@ | |||
| import asyncio | ||||
| from openai import AsyncOpenAI | ||||
| from openai.types.chat import ChatCompletionUserMessageParam | ||||
| from mcp import ClientSession | ||||
| from mcp.client.sse import sse_client | ||||
| from litellm.experimental_mcp_client.tools import ( | ||||
|     transform_mcp_tool_to_openai_tool, | ||||
|     transform_openai_tool_call_request_to_mcp_tool_call_request, | ||||
| ) | ||||
| 
 | ||||
| 
 | ||||
| async def main(): | ||||
|     # Initialize clients | ||||
|      | ||||
|     # point OpenAI client to local LiteLLM Proxy | ||||
|     client = AsyncOpenAI(api_key="sk-0jdcGHZJpX2oUJmyEs7zVA", base_url="https://litellm.seekkey.tech") | ||||
| 
 | ||||
|     # Point MCP client to local LiteLLM Proxy with authentication | ||||
|     headers = {"Authorization": "Bearer sk-0jdcGHZJpX2oUJmyEs7zVA"} | ||||
|     async with sse_client("https://litellm.seekkey.tech/mcp/", headers=headers) as (read, write): | ||||
|         async with ClientSession(read, write) as session: | ||||
|             await session.initialize() | ||||
| 
 | ||||
|             # 1. List MCP tools on LiteLLM Proxy | ||||
|             mcp_tools = await session.list_tools() | ||||
|             print("List of MCP tools for MCP server:", mcp_tools.tools) | ||||
| 
 | ||||
|             # Create message | ||||
|             messages = [ | ||||
|                 ChatCompletionUserMessageParam( | ||||
|                     content="Send an email about LiteLLM supporting MCP", role="user" | ||||
|                 ) | ||||
|             ] | ||||
| 
 | ||||
|             # 2. Use `transform_mcp_tool_to_openai_tool` to convert MCP tools to OpenAI tools | ||||
|             # Since OpenAI only supports tools in the OpenAI format, we need to convert the MCP tools to the OpenAI format. | ||||
|             openai_tools = [ | ||||
|                 transform_mcp_tool_to_openai_tool(tool) for tool in mcp_tools.tools | ||||
|             ] | ||||
| 
 | ||||
|             # 3. Provide the MCP tools to `gpt-4o` | ||||
|             response = await client.chat.completions.create( | ||||
|                 model="gemini/gemini-2.5-flash", | ||||
|                 messages=messages, | ||||
|                 tools=openai_tools, | ||||
|                 tool_choice="auto", | ||||
|             ) | ||||
| 
 | ||||
|             # 4. Handle tool call from `gpt-4o` | ||||
|             if response.choices[0].message.tool_calls: | ||||
|                 tool_call = response.choices[0].message.tool_calls[0] | ||||
|                 if tool_call: | ||||
| 
 | ||||
|                     # 5. Convert OpenAI tool call to MCP tool call | ||||
|                     # Since MCP servers expect tools in the MCP format, we need to convert the OpenAI tool call to the MCP format. | ||||
|                     # This is done using litellm.experimental_mcp_client.tools.transform_openai_tool_call_request_to_mcp_tool_call_request | ||||
|                     mcp_call = ( | ||||
|                         transform_openai_tool_call_request_to_mcp_tool_call_request( | ||||
|                             openai_tool=tool_call.model_dump() | ||||
|                         ) | ||||
|                     ) | ||||
| 
 | ||||
|                     # 6. Execute tool call on MCP server | ||||
|                     result = await session.call_tool( | ||||
|                         name=mcp_call.name, arguments=mcp_call.arguments | ||||
|                     ) | ||||
| 
 | ||||
|                     print("Result:", result) | ||||
| 
 | ||||
| 
 | ||||
| # Run it | ||||
| asyncio.run(main()) | ||||
|  | @ -0,0 +1,108 @@ | |||
| import asyncio | ||||
| import sys | ||||
| from openai import AsyncOpenAI | ||||
| from openai.types.chat import ChatCompletionUserMessageParam | ||||
| from mcp import ClientSession | ||||
| from mcp.client.sse import sse_client | ||||
| from litellm.experimental_mcp_client.tools import ( | ||||
|     transform_mcp_tool_to_openai_tool, | ||||
|     transform_openai_tool_call_request_to_mcp_tool_call_request, | ||||
| ) | ||||
| 
 | ||||
| 
 | ||||
| async def main(): | ||||
|     print("开始测试MCP连接...") | ||||
|      | ||||
|     try: | ||||
|         # Initialize clients | ||||
|         print("初始化OpenAI客户端...") | ||||
|         client = AsyncOpenAI(api_key="sk-0jdcGHZJpX2oUJmyEs7zVA", base_url="https://litellm.seekkey.tech") | ||||
|         print("OpenAI客户端初始化完成") | ||||
| 
 | ||||
|         # Point MCP client to remote LiteLLM Proxy with authentication | ||||
|         print("准备连接MCP服务器...") | ||||
|         headers = {"Authorization": "Bearer sk-0jdcGHZJpX2oUJmyEs7zVA"} | ||||
|          | ||||
|         # 添加超时处理 | ||||
|         try: | ||||
|             async with asyncio.timeout(10):  # 10秒超时 | ||||
|                 print("正在建立SSE连接...") | ||||
|                 async with sse_client("https://litellm.seekkey.tech/mcp/", headers=headers) as (read, write): | ||||
|                     print("SSE连接建立成功,初始化会话...") | ||||
|                     async with ClientSession(read, write) as session: | ||||
|                         print("正在初始化MCP会话...") | ||||
|                         await session.initialize() | ||||
|                         print("MCP会话初始化成功!") | ||||
| 
 | ||||
|                         # 1. List MCP tools on LiteLLM Proxy | ||||
|                         print("获取MCP工具列表...") | ||||
|                         mcp_tools = await session.list_tools() | ||||
|                         print(f"找到 {len(mcp_tools.tools)} 个MCP工具:") | ||||
|                         for tool in mcp_tools.tools: | ||||
|                             print(f"  - {tool.name}: {tool.description}") | ||||
| 
 | ||||
|                         if not mcp_tools.tools: | ||||
|                             print("没有找到可用的MCP工具") | ||||
|                             return | ||||
| 
 | ||||
|                         # Create message | ||||
|                         messages = [ | ||||
|                             ChatCompletionUserMessageParam( | ||||
|                                 content="列出所有可用的数据库", role="user" | ||||
|                             ) | ||||
|                         ] | ||||
| 
 | ||||
|                         # 2. Convert MCP tools to OpenAI tools | ||||
|                         print("转换MCP工具为OpenAI格式...") | ||||
|                         openai_tools = [ | ||||
|                             transform_mcp_tool_to_openai_tool(tool) for tool in mcp_tools.tools | ||||
|                         ] | ||||
|                         print(f"转换完成,共 {len(openai_tools)} 个工具") | ||||
| 
 | ||||
|                         # 3. Call LLM with tools | ||||
|                         print("调用LLM...") | ||||
|                         response = await client.chat.completions.create( | ||||
|                             model="gemini/gemini-2.5-flash", | ||||
|                             messages=messages, | ||||
|                             tools=openai_tools, | ||||
|                             tool_choice="auto", | ||||
|                         ) | ||||
|                         print("LLM响应完成") | ||||
| 
 | ||||
|                         # 4. Handle tool call | ||||
|                         if response.choices[0].message.tool_calls: | ||||
|                             print("LLM请求调用工具...") | ||||
|                             tool_call = response.choices[0].message.tool_calls[0] | ||||
|                             print(f"工具调用: {tool_call.function.name}") | ||||
|                             print(f"参数: {tool_call.function.arguments}") | ||||
| 
 | ||||
|                             # 5. Convert to MCP format and execute | ||||
|                             mcp_call = transform_openai_tool_call_request_to_mcp_tool_call_request( | ||||
|                                 openai_tool=tool_call.model_dump() | ||||
|                             ) | ||||
| 
 | ||||
|                             print(f"执行MCP工具调用: {mcp_call.name}") | ||||
|                             result = await session.call_tool( | ||||
|                                 name=mcp_call.name, arguments=mcp_call.arguments | ||||
|                             ) | ||||
| 
 | ||||
|                             print("工具调用结果:") | ||||
|                             print(result) | ||||
|                         else: | ||||
|                             print("LLM没有请求调用工具") | ||||
|                             print(f"LLM回复: {response.choices[0].message.content}") | ||||
|                              | ||||
|         except asyncio.TimeoutError: | ||||
|             print("连接超时!可能是网络问题或服务器响应慢") | ||||
|             return | ||||
|              | ||||
|     except Exception as e: | ||||
|         print(f"发生错误: {type(e).__name__}: {e}") | ||||
|         import traceback | ||||
|         traceback.print_exc() | ||||
| 
 | ||||
| 
 | ||||
| if __name__ == "__main__": | ||||
|     print("启动MCP调试测试...") | ||||
|     asyncio.run(main()) | ||||
|     print("测试完成") | ||||
|  | @ -0,0 +1,107 @@ | |||
| import asyncio | ||||
| import sys | ||||
| from openai import AsyncOpenAI | ||||
| from openai.types.chat import ChatCompletionUserMessageParam | ||||
| from mcp import ClientSession | ||||
| from mcp.client.sse import sse_client | ||||
| from litellm.experimental_mcp_client.tools import ( | ||||
|     transform_mcp_tool_to_openai_tool, | ||||
|     transform_openai_tool_call_request_to_mcp_tool_call_request, | ||||
| ) | ||||
| 
 | ||||
| 
 | ||||
| async def main(): | ||||
|     print("开始测试本地MCP连接...") | ||||
|      | ||||
|     try: | ||||
|         # Initialize clients | ||||
|         print("初始化OpenAI客户端...") | ||||
|         client = AsyncOpenAI(api_key="sk-1234", base_url="http://localhost:4000") | ||||
|         print("OpenAI客户端初始化完成") | ||||
| 
 | ||||
|         # Point MCP client to local LiteLLM Proxy | ||||
|         print("准备连接本地MCP服务器...") | ||||
|          | ||||
|         # 添加超时处理 | ||||
|         try: | ||||
|             async with asyncio.timeout(10):  # 10秒超时 | ||||
|                 print("正在建立SSE连接...") | ||||
|                 async with sse_client("http://localhost:4000/mcp/") as (read, write): | ||||
|                     print("SSE连接建立成功,初始化会话...") | ||||
|                     async with ClientSession(read, write) as session: | ||||
|                         print("正在初始化MCP会话...") | ||||
|                         await session.initialize() | ||||
|                         print("MCP会话初始化成功!") | ||||
| 
 | ||||
|                         # 1. List MCP tools on LiteLLM Proxy | ||||
|                         print("获取MCP工具列表...") | ||||
|                         mcp_tools = await session.list_tools() | ||||
|                         print(f"找到 {len(mcp_tools.tools)} 个MCP工具:") | ||||
|                         for tool in mcp_tools.tools: | ||||
|                             print(f"  - {tool.name}: {tool.description}") | ||||
| 
 | ||||
|                         if not mcp_tools.tools: | ||||
|                             print("没有找到可用的MCP工具") | ||||
|                             return | ||||
| 
 | ||||
|                         # Create message | ||||
|                         messages = [ | ||||
|                             ChatCompletionUserMessageParam( | ||||
|                                 content="列出所有可用的数据库", role="user" | ||||
|                             ) | ||||
|                         ] | ||||
| 
 | ||||
|                         # 2. Convert MCP tools to OpenAI tools | ||||
|                         print("转换MCP工具为OpenAI格式...") | ||||
|                         openai_tools = [ | ||||
|                             transform_mcp_tool_to_openai_tool(tool) for tool in mcp_tools.tools | ||||
|                         ] | ||||
|                         print(f"转换完成,共 {len(openai_tools)} 个工具") | ||||
| 
 | ||||
|                         # 3. Call LLM with tools | ||||
|                         print("调用LLM...") | ||||
|                         response = await client.chat.completions.create( | ||||
|                             model="gemini/gemini-2.5-flash", | ||||
|                             messages=messages, | ||||
|                             tools=openai_tools, | ||||
|                             tool_choice="auto", | ||||
|                         ) | ||||
|                         print("LLM响应完成") | ||||
| 
 | ||||
|                         # 4. Handle tool call | ||||
|                         if response.choices[0].message.tool_calls: | ||||
|                             print("LLM请求调用工具...") | ||||
|                             tool_call = response.choices[0].message.tool_calls[0] | ||||
|                             print(f"工具调用: {tool_call.function.name}") | ||||
|                             print(f"参数: {tool_call.function.arguments}") | ||||
| 
 | ||||
|                             # 5. Convert to MCP format and execute | ||||
|                             mcp_call = transform_openai_tool_call_request_to_mcp_tool_call_request( | ||||
|                                 openai_tool=tool_call.model_dump() | ||||
|                             ) | ||||
| 
 | ||||
|                             print(f"执行MCP工具调用: {mcp_call.name}") | ||||
|                             result = await session.call_tool( | ||||
|                                 name=mcp_call.name, arguments=mcp_call.arguments | ||||
|                             ) | ||||
| 
 | ||||
|                             print("工具调用结果:") | ||||
|                             print(result) | ||||
|                         else: | ||||
|                             print("LLM没有请求调用工具") | ||||
|                             print(f"LLM回复: {response.choices[0].message.content}") | ||||
|                              | ||||
|         except asyncio.TimeoutError: | ||||
|             print("连接超时!可能是本地服务器没有启动或MCP功能未配置") | ||||
|             return | ||||
|              | ||||
|     except Exception as e: | ||||
|         print(f"发生错误: {type(e).__name__}: {e}") | ||||
|         import traceback | ||||
|         traceback.print_exc() | ||||
| 
 | ||||
| 
 | ||||
| if __name__ == "__main__": | ||||
|     print("启动本地MCP调试测试...") | ||||
|     asyncio.run(main()) | ||||
|     print("测试完成") | ||||
|  | @ -0,0 +1,36 @@ | |||
| { | ||||
|   "吕洞宾": [ | ||||
|     { | ||||
|       "message": "你觉得特斯拉股票怎么样?", | ||||
|       "response": "这位股友,特斯拉啊,那是柄锋利的剑,但也得看你握得稳不稳。技术面上看,波动剧烈,高风险高回报!我吕洞宾向来胆大,看好其未来潜力,但提醒一句,入场需谨慎,设好止损位,别被震下马!这波行情,或可搏一搏,但盈亏自负,切记切记!\n", | ||||
|       "timestamp": "2025-08-10T15:13:34.203982" | ||||
|     }, | ||||
|     { | ||||
|       "message": "现在市场波动很大,你怎么看?", | ||||
|       "response": "市场波动大?正合我意!乱世出英雄,震荡出机会!这正是剑仙我辈大展身手之时!技术分析看穿迷雾,短线操作把握良机。但切记,风浪越大越要稳住船舵,设好止损,见好就收,莫贪恋!\n", | ||||
|       "timestamp": "2025-08-10T15:13:35.458763" | ||||
|     }, | ||||
|     { | ||||
|       "message": "你之前不是看好特斯拉吗?现在还是这个观点吗?", | ||||
|       "response": "这位股友,宝剑锋从磨砺出,梅花香自苦寒来!特斯拉虽遇风浪,但我仍看好它的长远潜力。短期震荡乃兵家常事,我会密切关注技术指标,伺机而动。剑仙的眼光,可不是轻易改变的!当然,投资需谨慎,切莫盲从,一切盈亏还需自负!\n", | ||||
|       "timestamp": "2025-08-10T15:13:37.048007" | ||||
|     } | ||||
|   ], | ||||
|   "何仙姑": [ | ||||
|     { | ||||
|       "message": "特斯拉股票风险大吗?", | ||||
|       "response": "这位朋友,特斯拉股票确实存在一定风险。电动车市场竞争激烈,技术迭代迅速,政策变化也可能带来影响。此外,马斯克个人的言行也可能对股价造成波动。投资需谨慎,建议您充分了解特斯拉的财务状况、行业地位和未来发展规划,结合自身的风险承受能力,审慎决策。\n", | ||||
|       "timestamp": "2025-08-10T15:13:38.280431" | ||||
|     }, | ||||
|     { | ||||
|       "message": "现在适合投资吗?", | ||||
|       "response": "这位朋友,现在市场波动较大,投资环境存在不确定性。虽然特斯拉前景广阔,但也面临竞争压力和技术风险。我建议您谨慎评估,充分了解市场动态和特斯拉的最新情况,并结合自身的风险承受能力,再做决定。稳健投资,安全第一。\n", | ||||
|       "timestamp": "2025-08-10T15:13:39.502875" | ||||
|     }, | ||||
|     { | ||||
|       "message": "你一直很谨慎,现在还是建议观望吗?", | ||||
|       "response": "这位朋友,谨慎是我的本分。当前市场环境依然复杂,特斯拉也面临着诸多挑战。虽然长期来看,其发展潜力巨大,但短期内仍可能存在波动。我仍然建议您保持观望,密切关注市场动态和特斯拉的最新消息,待风险进一步明确后再做决策。安全第一,稳健投资才是长久之计。\n", | ||||
|       "timestamp": "2025-08-10T15:13:40.890509" | ||||
|     } | ||||
|   ] | ||||
| } | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							|  | @ -0,0 +1,31 @@ | |||
| { | ||||
|   "name": "hyperdrive-neondb-test", | ||||
|   "version": "1.0.0", | ||||
|   "description": "Cloudflare Worker to test Hyperdrive connection with NeonDB", | ||||
|   "main": "src/index.ts", | ||||
|   "scripts": { | ||||
|     "dev": "wrangler dev", | ||||
|     "deploy": "wrangler deploy", | ||||
|     "test": "wrangler dev --local" | ||||
|   }, | ||||
|   "dependencies": { | ||||
|     "node-fetch": "^3.3.2", | ||||
|     "pg": "^8.11.3" | ||||
|   }, | ||||
|   "devDependencies": { | ||||
|     "@cloudflare/workers-types": "^4.20240129.0", | ||||
|     "@types/node": "^20.11.16", | ||||
|     "@types/pg": "^8.10.9", | ||||
|     "typescript": "^5.3.3", | ||||
|     "wrangler": "^3.28.2" | ||||
|   }, | ||||
|   "keywords": [ | ||||
|     "cloudflare", | ||||
|     "workers", | ||||
|     "hyperdrive", | ||||
|     "neondb", | ||||
|     "postgresql" | ||||
|   ], | ||||
|   "author": "", | ||||
|   "license": "MIT" | ||||
| } | ||||
|  | @ -0,0 +1,39 @@ | |||
| { | ||||
|   "current_stage": "起", | ||||
|   "stage_progress": 4, | ||||
|   "total_handoffs": 0, | ||||
|   "debate_history": [ | ||||
|     { | ||||
|       "timestamp": "2025-08-10T15:30:47.514243", | ||||
|       "stage": "起", | ||||
|       "progress": 0, | ||||
|       "speaker": "吕洞宾", | ||||
|       "message": "起:八仙按先天八卦顺序阐述观点", | ||||
|       "handoffs": 0 | ||||
|     }, | ||||
|     { | ||||
|       "timestamp": "2025-08-10T15:30:47.514260", | ||||
|       "stage": "起", | ||||
|       "progress": 1, | ||||
|       "speaker": "何仙姑", | ||||
|       "message": "承:雁阵式承接,总体阐述+讥讽", | ||||
|       "handoffs": 0 | ||||
|     }, | ||||
|     { | ||||
|       "timestamp": "2025-08-10T15:30:47.514272", | ||||
|       "stage": "起", | ||||
|       "progress": 2, | ||||
|       "speaker": "铁拐李", | ||||
|       "message": "转:自由辩论,36次handoff", | ||||
|       "handoffs": 0 | ||||
|     }, | ||||
|     { | ||||
|       "timestamp": "2025-08-10T15:30:47.514281", | ||||
|       "stage": "起", | ||||
|       "progress": 3, | ||||
|       "speaker": "汉钟离", | ||||
|       "message": "合:交替总结,最终论证", | ||||
|       "handoffs": 0 | ||||
|     } | ||||
|   ] | ||||
| } | ||||
|  | @ -0,0 +1,138 @@ | |||
| // 查询术数书内容的脚本
 | ||||
| // 通过 Hyperdrive API 查询 NeonDB 中的术数书数据
 | ||||
| 
 | ||||
| const API_BASE_URL = 'https://hyperdrive.seekkey.tech'; | ||||
| 
 | ||||
| // 通用请求函数
 | ||||
| async function apiRequest(endpoint, options = {}) { | ||||
|   const url = `${API_BASE_URL}${endpoint}`; | ||||
|   const headers = { | ||||
|     'Content-Type': 'application/json', | ||||
|     ...options.headers | ||||
|   }; | ||||
|    | ||||
|   try { | ||||
|     const response = await fetch(url, { | ||||
|       ...options, | ||||
|       headers | ||||
|     }); | ||||
|      | ||||
|     if (!response.ok) { | ||||
|       throw new Error(`HTTP ${response.status}: ${response.statusText}`); | ||||
|     } | ||||
|      | ||||
|     const contentType = response.headers.get('content-type'); | ||||
|     if (contentType && contentType.includes('application/json')) { | ||||
|       return await response.json(); | ||||
|     } else { | ||||
|       return await response.text(); | ||||
|     } | ||||
|   } catch (error) { | ||||
|     console.error(`Request failed for ${endpoint}:`, error.message); | ||||
|     throw error; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| // 查询数据库表结构
 | ||||
| async function queryTables() { | ||||
|   console.log('\n📋 查询数据库表结构...'); | ||||
|   try { | ||||
|     const result = await apiRequest('/query-tables'); | ||||
|     console.log('✅ 数据库表:', result); | ||||
|     return result; | ||||
|   } catch (error) { | ||||
|     console.log('❌ 查询表结构失败:', error.message); | ||||
|     return null; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| // 查询术数书内容
 | ||||
| async function queryShushuBook(limit = 10) { | ||||
|   console.log('\n📚 查询术数书内容...'); | ||||
|   try { | ||||
|     const result = await apiRequest(`/query-shushu?limit=${limit}`); | ||||
|     console.log('✅ 术数书内容:', JSON.stringify(result, null, 2)); | ||||
|     return result; | ||||
|   } catch (error) { | ||||
|     console.log('❌ 查询术数书失败:', error.message); | ||||
|     return null; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| // 搜索术数书内容
 | ||||
| async function searchShushuBook(keyword, limit = 5) { | ||||
|   console.log(`\n🔍 搜索术数书内容: "${keyword}"...`); | ||||
|   try { | ||||
|     const result = await apiRequest(`/search-shushu?q=${encodeURIComponent(keyword)}&limit=${limit}`); | ||||
|     console.log('✅ 搜索结果:', JSON.stringify(result, null, 2)); | ||||
|     return result; | ||||
|   } catch (error) { | ||||
|     console.log('❌ 搜索失败:', error.message); | ||||
|     return null; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| // 获取术数书统计信息
 | ||||
| async function getShushuStats() { | ||||
|   console.log('\n📊 获取术数书统计信息...'); | ||||
|   try { | ||||
|     const result = await apiRequest('/shushu-stats'); | ||||
|     console.log('✅ 统计信息:', JSON.stringify(result, null, 2)); | ||||
|     return result; | ||||
|   } catch (error) { | ||||
|     console.log('❌ 获取统计信息失败:', error.message); | ||||
|     return null; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| // 主函数
 | ||||
| async function main() { | ||||
|   console.log('🚀 术数书查询脚本'); | ||||
|   console.log('=================='); | ||||
|    | ||||
|   // 首先测试连接
 | ||||
|   console.log('\n🔗 测试 Hyperdrive 连接...'); | ||||
|   try { | ||||
|     const connectionTest = await apiRequest('/test-connection'); | ||||
|     console.log('✅ 连接成功:', connectionTest.message); | ||||
|   } catch (error) { | ||||
|     console.log('❌ 连接失败:', error.message); | ||||
|     return; | ||||
|   } | ||||
|    | ||||
|   // 查询表结构
 | ||||
|   await queryTables(); | ||||
|    | ||||
|   // 获取统计信息
 | ||||
|   await getShushuStats(); | ||||
|    | ||||
|   // 查询术数书内容
 | ||||
|   await queryShushuBook(5); | ||||
|    | ||||
|   // 搜索示例
 | ||||
|   await searchShushuBook('易经'); | ||||
|   await searchShushuBook('八卦'); | ||||
|   await searchShushuBook('太公'); | ||||
| } | ||||
| 
 | ||||
| // 如果是 Node.js 环境,导入 fetch
 | ||||
| if (typeof window === 'undefined') { | ||||
|   // Node.js 环境
 | ||||
|   const { default: fetch } = require('node-fetch'); | ||||
|   global.fetch = fetch; | ||||
|   main().catch(console.error); | ||||
| } else { | ||||
|   // 浏览器环境
 | ||||
|   console.log('在浏览器控制台中运行: main()'); | ||||
| } | ||||
| 
 | ||||
| // 导出函数供其他模块使用
 | ||||
| if (typeof module !== 'undefined' && module.exports) { | ||||
|   module.exports = { | ||||
|     queryTables, | ||||
|     queryShushuBook, | ||||
|     searchShushuBook, | ||||
|     getShushuStats, | ||||
|     main | ||||
|   }; | ||||
| } | ||||
|  | @ -13,13 +13,19 @@ plotly>=5.15.0 | |||
| 
 | ||||
| # HTTP请求 | ||||
| requests>=2.31.0 | ||||
| aiohttp>=3.8.0 | ||||
| # Cloudflare (HTTP 调用即可,无需额外 SDK) | ||||
| 
 | ||||
| 
 | ||||
| # RSS解析 | ||||
| feedparser>=6.0.0 | ||||
| 
 | ||||
| # 类型注解支持 | ||||
| typing-extensions>=4.7.0 | ||||
| 
 | ||||
| # 数据库连接 (可选) | ||||
| # 数据库连接 | ||||
| # sqlalchemy>=2.0.0 | ||||
| # pymongo>=4.5.0 | ||||
| pymongo>=4.5.0 | ||||
| # pymilvus>=2.3.0 | ||||
| 
 | ||||
| # 开发工具 (可选) | ||||
|  | @ -28,8 +34,16 @@ typing-extensions>=4.7.0 | |||
| # flake8>=6.0.0 | ||||
| 
 | ||||
| # AI模型接口 | ||||
| # 旧系统:OpenRouter + OpenAI Swarm | ||||
| openai>=1.0.0 | ||||
| # anthropic>=0.3.0 | ||||
| 
 | ||||
| # OpenAI Swarm (从GitHub安装) | ||||
| # 新系统:Google ADK (根据迁移进度选择) | ||||
| # pip install google-adk | ||||
| # 或开发版: pip install git+https://github.com/google/adk-python.git@main | ||||
| 
 | ||||
| # Vertex AI Memory Bank 支持 | ||||
| google-cloud-aiplatform>=1.38.0 | ||||
| 
 | ||||
| # OpenAI Swarm (保留兼容性,逐步替换) | ||||
| # pip install git+https://github.com/openai/swarm.git | ||||
|  | @ -1,268 +0,0 @@ | |||
| #!/usr/bin/env python3 | ||||
| """ | ||||
| RSS新闻收集器 | ||||
| 收集RSS新闻并存储到MongoDB,为辩论系统提供数据源 | ||||
| """ | ||||
| 
 | ||||
| import asyncio | ||||
| import feedparser | ||||
| import json | ||||
| import logging | ||||
| from datetime import datetime, timezone | ||||
| from typing import Dict, List, Any, Optional | ||||
| from urllib.parse import urlparse | ||||
| import hashlib | ||||
| import requests | ||||
| from src.mcp.swarm_mongodb_client import SwarmMongoDBClient | ||||
| 
 | ||||
| class RSSNewsCollector: | ||||
|     """RSS新闻收集器""" | ||||
|      | ||||
|     def __init__(self, mongodb_client: SwarmMongoDBClient): | ||||
|         self.mongodb_client = mongodb_client | ||||
|         self.logger = logging.getLogger(__name__) | ||||
|          | ||||
|         # 默认RSS源配置 | ||||
|         self.rss_sources = { | ||||
|             '财经新闻': [ | ||||
|                 'https://feeds.finance.yahoo.com/rss/2.0/headline', | ||||
|                 'https://www.cnbc.com/id/100003114/device/rss/rss.html', | ||||
|                 'https://feeds.reuters.com/reuters/businessNews' | ||||
|             ], | ||||
|             '科技新闻': [ | ||||
|                 'https://feeds.feedburner.com/TechCrunch', | ||||
|                 'https://www.wired.com/feed/rss', | ||||
|                 'https://feeds.arstechnica.com/arstechnica/index' | ||||
|             ], | ||||
|             '市场分析': [ | ||||
|                 'https://feeds.marketwatch.com/marketwatch/marketpulse/', | ||||
|                 'https://feeds.bloomberg.com/markets/news.rss' | ||||
|             ] | ||||
|         } | ||||
|      | ||||
|     def generate_article_id(self, url: str, title: str) -> str: | ||||
|         """生成文章唯一ID""" | ||||
|         content = f"{url}_{title}" | ||||
|         return hashlib.md5(content.encode()).hexdigest() | ||||
|      | ||||
|     def parse_rss_feed(self, rss_url: str) -> List[Dict[str, Any]]: | ||||
|         """解析RSS源""" | ||||
|         try: | ||||
|             feed = feedparser.parse(rss_url) | ||||
|             articles = [] | ||||
|              | ||||
|             for entry in feed.entries: | ||||
|                 # 提取文章信息 | ||||
|                 article = { | ||||
|                     'article_id': self.generate_article_id(entry.link, entry.title), | ||||
|                     'title': entry.title, | ||||
|                     'link': entry.link, | ||||
|                     'description': getattr(entry, 'description', ''), | ||||
|                     'summary': getattr(entry, 'summary', ''), | ||||
|                     'published': self._parse_date(getattr(entry, 'published', '')), | ||||
|                     'author': getattr(entry, 'author', ''), | ||||
|                     'tags': [tag.term for tag in getattr(entry, 'tags', [])], | ||||
|                     'source_url': rss_url, | ||||
|                     'source_title': feed.feed.get('title', ''), | ||||
|                     'collected_at': datetime.now(timezone.utc), | ||||
|                     'content_hash': hashlib.md5(entry.title.encode()).hexdigest() | ||||
|                 } | ||||
|                 articles.append(article) | ||||
|              | ||||
|             return articles | ||||
|              | ||||
|         except Exception as e: | ||||
|             self.logger.error(f"解析RSS源失败 {rss_url}: {e}") | ||||
|             return [] | ||||
|      | ||||
|     def _parse_date(self, date_str: str) -> Optional[datetime]: | ||||
|         """解析日期字符串""" | ||||
|         if not date_str: | ||||
|             return None | ||||
|          | ||||
|         try: | ||||
|             # feedparser通常会解析时间 | ||||
|             import time | ||||
|             parsed_time = feedparser._parse_date(date_str) | ||||
|             if parsed_time: | ||||
|                 return datetime.fromtimestamp(time.mktime(parsed_time), tz=timezone.utc) | ||||
|         except: | ||||
|             pass | ||||
|          | ||||
|         return datetime.now(timezone.utc) | ||||
|      | ||||
|     async def collect_news_from_category(self, category: str) -> List[Dict[str, Any]]: | ||||
|         """从指定类别收集新闻""" | ||||
|         if category not in self.rss_sources: | ||||
|             self.logger.warning(f"未知新闻类别: {category}") | ||||
|             return [] | ||||
|          | ||||
|         all_articles = [] | ||||
|         for rss_url in self.rss_sources[category]: | ||||
|             self.logger.info(f"正在收集新闻: {rss_url}") | ||||
|             articles = self.parse_rss_feed(rss_url) | ||||
|              | ||||
|             # 添加类别标签 | ||||
|             for article in articles: | ||||
|                 article['category'] = category | ||||
|              | ||||
|             all_articles.extend(articles) | ||||
|          | ||||
|         return all_articles | ||||
|      | ||||
|     async def collect_all_news(self) -> Dict[str, List[Dict[str, Any]]]: | ||||
|         """收集所有类别的新闻""" | ||||
|         all_news = {} | ||||
|          | ||||
|         for category in self.rss_sources.keys(): | ||||
|             news = await self.collect_news_from_category(category) | ||||
|             all_news[category] = news | ||||
|             self.logger.info(f"收集到 {len(news)} 条 {category} 新闻") | ||||
|          | ||||
|         return all_news | ||||
|      | ||||
|     async def store_news_to_mongodb(self, articles: List[Dict[str, Any]], collection_name: str = "news_articles") -> Dict[str, Any]: | ||||
|         """将新闻存储到MongoDB""" | ||||
|         if not articles: | ||||
|             return {'success': True, 'inserted_count': 0, 'updated_count': 0} | ||||
|          | ||||
|         inserted_count = 0 | ||||
|         updated_count = 0 | ||||
|          | ||||
|         for article in articles: | ||||
|             # 检查文章是否已存在 | ||||
|             existing = self.mongodb_client.find_documents( | ||||
|                 collection_name, | ||||
|                 query={'article_id': article['article_id']}, | ||||
|                 limit=1 | ||||
|             ) | ||||
|              | ||||
|             if existing.get('success') and existing.get('documents'): | ||||
|                 # 更新现有文章 | ||||
|                 update_result = self.mongodb_client.update_document( | ||||
|                     collection_name, | ||||
|                     query={'article_id': article['article_id']}, | ||||
|                     update={'$set': article} | ||||
|                 ) | ||||
|                 if update_result.get('success'): | ||||
|                     updated_count += 1 | ||||
|             else: | ||||
|                 # 插入新文章 | ||||
|                 insert_result = self.mongodb_client.insert_document( | ||||
|                     collection_name, | ||||
|                     document=article | ||||
|                 ) | ||||
|                 if insert_result.get('success'): | ||||
|                     inserted_count += 1 | ||||
|          | ||||
|         return { | ||||
|             'success': True, | ||||
|             'inserted_count': inserted_count, | ||||
|             'updated_count': updated_count, | ||||
|             'total_processed': len(articles) | ||||
|         } | ||||
|      | ||||
|     async def get_latest_news(self, category: Optional[str] = None, limit: int = 10) -> List[Dict[str, Any]]: | ||||
|         """获取最新新闻""" | ||||
|         query = {} | ||||
|         if category: | ||||
|             query['category'] = category | ||||
|          | ||||
|         result = self.mongodb_client.find_documents( | ||||
|             'news_articles', | ||||
|             query=query, | ||||
|             sort={'collected_at': -1}, | ||||
|             limit=limit | ||||
|         ) | ||||
|          | ||||
|         if result.get('success'): | ||||
|             return result.get('documents', []) | ||||
|         return [] | ||||
|      | ||||
|     async def get_news_for_debate(self, topic_keywords: List[str], limit: int = 5) -> List[Dict[str, Any]]: | ||||
|         """根据关键词获取相关新闻用于辩论""" | ||||
|         # 构建搜索查询 | ||||
|         search_conditions = [] | ||||
|         for keyword in topic_keywords: | ||||
|             search_conditions.extend([ | ||||
|                 {'title': {'$regex': keyword, '$options': 'i'}}, | ||||
|                 {'description': {'$regex': keyword, '$options': 'i'}}, | ||||
|                 {'summary': {'$regex': keyword, '$options': 'i'}} | ||||
|             ]) | ||||
|          | ||||
|         query = {'$or': search_conditions} if search_conditions else {} | ||||
|          | ||||
|         result = self.mongodb_client.find_documents( | ||||
|             'news_articles', | ||||
|             query=query, | ||||
|             sort={'published': -1}, | ||||
|             limit=limit | ||||
|         ) | ||||
|          | ||||
|         if result.get('success'): | ||||
|             return result.get('documents', []) | ||||
|         return [] | ||||
|      | ||||
|     async def run_collection_cycle(self): | ||||
|         """运行一次完整的新闻收集周期""" | ||||
|         self.logger.info("开始新闻收集周期") | ||||
|          | ||||
|         # 收集所有新闻 | ||||
|         all_news = await self.collect_all_news() | ||||
|          | ||||
|         # 存储到数据库 | ||||
|         total_inserted = 0 | ||||
|         total_updated = 0 | ||||
|          | ||||
|         for category, articles in all_news.items(): | ||||
|             if articles: | ||||
|                 result = await self.store_news_to_mongodb(articles) | ||||
|                 total_inserted += result.get('inserted_count', 0) | ||||
|                 total_updated += result.get('updated_count', 0) | ||||
|                 self.logger.info(f"{category}: 新增 {result.get('inserted_count', 0)}, 更新 {result.get('updated_count', 0)}") | ||||
|          | ||||
|         self.logger.info(f"新闻收集完成: 总新增 {total_inserted}, 总更新 {total_updated}") | ||||
|          | ||||
|         return { | ||||
|             'success': True, | ||||
|             'total_inserted': total_inserted, | ||||
|             'total_updated': total_updated, | ||||
|             'categories_processed': len(all_news) | ||||
|         } | ||||
| 
 | ||||
| async def main(): | ||||
|     """主函数 - 演示RSS新闻收集""" | ||||
|     # 初始化MongoDB客户端 | ||||
|     mongodb_client = SwarmMongoDBClient( | ||||
|         mcp_server_url="http://localhost:8080", | ||||
|         default_database="news_debate_db" | ||||
|     ) | ||||
|      | ||||
|     # 连接数据库 | ||||
|     connect_result = mongodb_client.connect("news_debate_db") | ||||
|     if not connect_result.get('success'): | ||||
|         print(f"数据库连接失败: {connect_result}") | ||||
|         return | ||||
|      | ||||
|     # 创建新闻收集器 | ||||
|     collector = RSSNewsCollector(mongodb_client) | ||||
|      | ||||
|     # 运行收集周期 | ||||
|     result = await collector.run_collection_cycle() | ||||
|     print(f"收集结果: {result}") | ||||
|      | ||||
|     # 获取最新新闻示例 | ||||
|     latest_news = await collector.get_latest_news(limit=5) | ||||
|     print(f"\n最新新闻 ({len(latest_news)} 条):") | ||||
|     for news in latest_news: | ||||
|         print(f"- {news.get('title', 'N/A')} [{news.get('category', 'N/A')}]") | ||||
|      | ||||
|     # 根据关键词搜索新闻示例 | ||||
|     debate_news = await collector.get_news_for_debate(['投资', '市场', '经济'], limit=3) | ||||
|     print(f"\n辩论相关新闻 ({len(debate_news)} 条):") | ||||
|     for news in debate_news: | ||||
|         print(f"- {news.get('title', 'N/A')} [{news.get('category', 'N/A')}]") | ||||
| 
 | ||||
| if __name__ == "__main__": | ||||
|     logging.basicConfig(level=logging.INFO) | ||||
|     asyncio.run(main()) | ||||
|  | @ -1,63 +0,0 @@ | |||
| #!/usr/bin/env python3 | ||||
| """ | ||||
| 为现有文章添加流水号 | ||||
| """ | ||||
| 
 | ||||
| import os | ||||
| from pymongo import MongoClient | ||||
| 
 | ||||
| def add_sequence_ids(): | ||||
|     """为现有文章添加流水号""" | ||||
|     # 连接MongoDB | ||||
|     mongo_uri = os.getenv('MONGODB_URI') | ||||
|     if not mongo_uri: | ||||
|         raise ValueError("MONGODB_URI environment variable is required") | ||||
|     client = MongoClient(mongo_uri) | ||||
|     db = client['taigong'] | ||||
|     collection = db['articles'] | ||||
|      | ||||
|     print("开始为现有文章添加流水号...") | ||||
|      | ||||
|     # 查找所有没有sequence_id的文章 | ||||
|     articles_without_seq = list(collection.find( | ||||
|         {"sequence_id": {"$exists": False}}, | ||||
|         {"_id": 1, "title": 1, "created_at": 1} | ||||
|     ).sort("created_at", 1))  # 按创建时间排序 | ||||
|      | ||||
|     print(f"找到 {len(articles_without_seq)} 篇文章需要添加流水号") | ||||
|      | ||||
|     if len(articles_without_seq) == 0: | ||||
|         print("所有文章都已有流水号") | ||||
|         return | ||||
|      | ||||
|     # 从1开始分配流水号 | ||||
|     for i, article in enumerate(articles_without_seq, 1): | ||||
|         sequence_id = i | ||||
|         article_id = f"NEWS_{sequence_id:08d}"  # NEWS_00000001 格式 | ||||
|          | ||||
|         collection.update_one( | ||||
|             {"_id": article["_id"]}, | ||||
|             { | ||||
|                 "$set": { | ||||
|                     "sequence_id": sequence_id, | ||||
|                     "article_id": article_id, | ||||
|                     "batch_id": "migration_batch", | ||||
|                     "last_updated": "2025-02-08T00:00:00Z" | ||||
|                 } | ||||
|             } | ||||
|         ) | ||||
|          | ||||
|         print(f"  {sequence_id:3d}: {article['title'][:50]}...") | ||||
|      | ||||
|     print(f"流水号添加完成,共处理 {len(articles_without_seq)} 篇文章") | ||||
|      | ||||
|     # 验证结果 | ||||
|     total_with_seq = collection.count_documents({"sequence_id": {"$exists": True}}) | ||||
|     max_seq = collection.find_one({}, sort=[("sequence_id", -1)]) | ||||
|      | ||||
|     print(f"验证结果:") | ||||
|     print(f"  有流水号的文章: {total_with_seq} 篇") | ||||
|     print(f"  最大流水号: {max_seq['sequence_id'] if max_seq else 0}") | ||||
| 
 | ||||
| if __name__ == "__main__": | ||||
|     add_sequence_ids() | ||||
|  | @ -1,107 +0,0 @@ | |||
| #!/usr/bin/env python3 | ||||
| """ | ||||
| 清理MongoDB中的重复文章数据 | ||||
| """ | ||||
| 
 | ||||
| import os | ||||
| import sys | ||||
| from pymongo import MongoClient | ||||
| from collections import defaultdict | ||||
| import hashlib | ||||
| 
 | ||||
| def generate_stable_id(title, pub_date, content): | ||||
|     """生成稳定的文章ID""" | ||||
|     normalized_title = title.strip().lower() | ||||
|     content_hash = content[:100] if content else '' | ||||
|     date_str = pub_date or '' | ||||
|      | ||||
|     combined = f"{normalized_title}|{date_str}|{content_hash}" | ||||
|     return hashlib.md5(combined.encode()).hexdigest()[:16] | ||||
| 
 | ||||
| def cleanup_duplicates(): | ||||
|     """清理重复数据""" | ||||
|     # 连接MongoDB | ||||
|     mongo_uri = os.getenv('MONGODB_URI') | ||||
|     if not mongo_uri: | ||||
|         raise ValueError("MONGODB_URI environment variable is required") | ||||
|     client = MongoClient(mongo_uri) | ||||
|     db = client['taigong'] | ||||
|     collection = db['articles'] | ||||
|      | ||||
|     print("开始清理重复数据...") | ||||
|      | ||||
|     # 1. 查找所有文章 | ||||
|     articles = list(collection.find({})) | ||||
|     print(f"总共找到 {len(articles)} 篇文章") | ||||
|      | ||||
|     # 2. 按标题分组,找出重复项 | ||||
|     title_groups = defaultdict(list) | ||||
|     for article in articles: | ||||
|         title_groups[article['title']].append(article) | ||||
|      | ||||
|     # 3. 处理重复项 | ||||
|     duplicates_removed = 0 | ||||
|     articles_updated = 0 | ||||
|      | ||||
|     for title, group in title_groups.items(): | ||||
|         if len(group) > 1: | ||||
|             print(f"发现重复标题: {title} ({len(group)} 篇)") | ||||
|              | ||||
|             # 保留最早的一篇,删除其他 | ||||
|             group.sort(key=lambda x: x.get('created_at', '')) | ||||
|             keep_article = group[0] | ||||
|              | ||||
|             # 更新保留文章的ID为稳定ID | ||||
|             stable_id = generate_stable_id( | ||||
|                 keep_article['title'], | ||||
|                 keep_article.get('published_time', ''), | ||||
|                 keep_article.get('content', '') | ||||
|             ) | ||||
|              | ||||
|             collection.update_one( | ||||
|                 {'_id': keep_article['_id']}, | ||||
|                 { | ||||
|                     '$set': { | ||||
|                         'article_id': stable_id, | ||||
|                         'content_hash': generate_stable_id(keep_article.get('content', ''), '', ''), | ||||
|                         'last_updated': '2025-02-08T00:00:00Z' | ||||
|                     } | ||||
|                 } | ||||
|             ) | ||||
|             articles_updated += 1 | ||||
|              | ||||
|             # 删除重复项 | ||||
|             for duplicate in group[1:]: | ||||
|                 collection.delete_one({'_id': duplicate['_id']}) | ||||
|                 duplicates_removed += 1 | ||||
|                 print(f"  删除重复项: {duplicate.get('article_id', 'unknown')}") | ||||
|      | ||||
|     # 4. 为没有重复的文章更新ID | ||||
|     single_articles = [group[0] for group in title_groups.values() if len(group) == 1] | ||||
|     for article in single_articles: | ||||
|         if not article.get('article_id') or len(article.get('article_id', '')) > 20: | ||||
|             stable_id = generate_stable_id( | ||||
|                 article['title'], | ||||
|                 article.get('published_time', ''), | ||||
|                 article.get('content', '') | ||||
|             ) | ||||
|              | ||||
|             collection.update_one( | ||||
|                 {'_id': article['_id']}, | ||||
|                 { | ||||
|                     '$set': { | ||||
|                         'article_id': stable_id, | ||||
|                         'content_hash': generate_stable_id(article.get('content', ''), '', ''), | ||||
|                         'last_updated': '2025-02-08T00:00:00Z' | ||||
|                     } | ||||
|                 } | ||||
|             ) | ||||
|             articles_updated += 1 | ||||
|      | ||||
|     print(f"清理完成:") | ||||
|     print(f"  删除重复文章: {duplicates_removed} 篇") | ||||
|     print(f"  更新文章ID: {articles_updated} 篇") | ||||
|     print(f"  最终文章数: {collection.count_documents({})} 篇") | ||||
| 
 | ||||
| if __name__ == "__main__": | ||||
|     cleanup_duplicates() | ||||
|  | @ -1,35 +0,0 @@ | |||
| // MongoDB Atlas Vector Search Index Creation Script
 | ||||
| // 为swarm辩论系统创建向量索引
 | ||||
| 
 | ||||
| // 连接到数据库
 | ||||
| use('taigong'); | ||||
| 
 | ||||
| // 创建向量索引用于语义搜索和内容聚类
 | ||||
| // 这个索引将支持swarm辩论系统的语义相似性匹配
 | ||||
| db.articles.createSearchIndex( | ||||
|   "vector_search_index", | ||||
|   { | ||||
|     "fields": [ | ||||
|       { | ||||
|         "type": "vector", | ||||
|         "path": "embedding", | ||||
|         "numDimensions": 1536, // OpenAI text-embedding-ada-002 维度
 | ||||
|         "similarity": "cosine" | ||||
|       }, | ||||
|       { | ||||
|         "type": "filter", | ||||
|         "path": "published_time" | ||||
|       }, | ||||
|       { | ||||
|         "type": "filter",  | ||||
|         "path": "title" | ||||
|       } | ||||
|     ] | ||||
|   } | ||||
| ); | ||||
| 
 | ||||
| print("向量索引创建完成!"); | ||||
| print("索引名称: vector_search_index"); | ||||
| print("向量维度: 1536 (OpenAI embedding)"); | ||||
| print("相似性算法: cosine"); | ||||
| print("支持过滤字段: published_time, title"); | ||||
|  | @ -1,75 +0,0 @@ | |||
| #!/usr/bin/env python3 | ||||
| """ | ||||
| 为MongoDB中的文章生成向量embeddings | ||||
| 用于swarm辩论系统的语义搜索和内容聚类 | ||||
| """ | ||||
| 
 | ||||
| import os | ||||
| import openai | ||||
| from pymongo import MongoClient | ||||
| from typing import List, Dict | ||||
| import time | ||||
| 
 | ||||
| def get_mongodb_client(): | ||||
|     """从Doppler获取MongoDB连接""" | ||||
|     mongodb_uri = os.getenv('MONGODB_URI') | ||||
|     if not mongodb_uri: | ||||
|         raise ValueError("MONGODB_URI not found in environment variables") | ||||
|     return MongoClient(mongodb_uri) | ||||
| 
 | ||||
| def generate_embedding(text: str) -> List[float]: | ||||
|     """使用OpenAI API生成文本embedding""" | ||||
|     try: | ||||
|         response = openai.Embedding.create( | ||||
|             model="text-embedding-ada-002", | ||||
|             input=text | ||||
|         ) | ||||
|         return response['data'][0]['embedding'] | ||||
|     except Exception as e: | ||||
|         print(f"生成embedding失败: {e}") | ||||
|         return None | ||||
| 
 | ||||
| def update_articles_with_embeddings(): | ||||
|     """为所有文章添加embedding字段""" | ||||
|     client = get_mongodb_client() | ||||
|     db = client.taigong | ||||
|     collection = db.articles | ||||
|      | ||||
|     # 获取所有没有embedding的文章 | ||||
|     articles = collection.find({"embedding": {"$exists": False}}) | ||||
|      | ||||
|     count = 0 | ||||
|     for article in articles: | ||||
|         title = article.get('title', '') | ||||
|         if not title: | ||||
|             continue | ||||
|              | ||||
|         print(f"处理文章: {title[:50]}...") | ||||
|          | ||||
|         # 生成embedding | ||||
|         embedding = generate_embedding(title) | ||||
|         if embedding: | ||||
|             # 更新文档 | ||||
|             collection.update_one( | ||||
|                 {"_id": article["_id"]}, | ||||
|                 {"$set": {"embedding": embedding}} | ||||
|             ) | ||||
|             count += 1 | ||||
|             print(f"✓ 已更新 {count} 篇文章") | ||||
|              | ||||
|             # 避免API rate limit | ||||
|             time.sleep(0.1) | ||||
|         else: | ||||
|             print(f"× 跳过文章: {title[:50]}") | ||||
|      | ||||
|     print(f"\n完成!共处理 {count} 篇文章") | ||||
|     client.close() | ||||
| 
 | ||||
| if __name__ == "__main__": | ||||
|     # 设置OpenAI API密钥 (应该从Doppler获取) | ||||
|     openai.api_key = os.getenv('OPENAI_API_KEY') | ||||
|     if not openai.api_key: | ||||
|         print("警告: OPENAI_API_KEY 未设置,请先在Doppler中配置") | ||||
|         exit(1) | ||||
|      | ||||
|     update_articles_with_embeddings() | ||||
|  | @ -1,73 +0,0 @@ | |||
| #!/usr/bin/env python3 | ||||
| """ | ||||
| 安装OpenAI Swarm的脚本 | ||||
| """ | ||||
| 
 | ||||
| import subprocess | ||||
| import sys | ||||
| 
 | ||||
| def install_swarm(): | ||||
|     """安装OpenAI Swarm""" | ||||
|     print("🚀 正在安装OpenAI Swarm...") | ||||
|      | ||||
|     try: | ||||
|         # 安装Swarm | ||||
|         result = subprocess.run([ | ||||
|             sys.executable, "-m", "pip", "install",  | ||||
|             "git+https://github.com/openai/swarm.git" | ||||
|         ], check=True, capture_output=True, text=True) | ||||
|          | ||||
|         print("✅ OpenAI Swarm安装成功!") | ||||
|         print(result.stdout) | ||||
|          | ||||
|         # 验证安装 | ||||
|         try: | ||||
|             import swarm | ||||
|             print("✅ Swarm导入测试成功") | ||||
|             print(f"📦 Swarm版本: {getattr(swarm, '__version__', '未知')}") | ||||
|         except ImportError as e: | ||||
|             print(f"❌ Swarm导入失败: {e}") | ||||
|             return False | ||||
|          | ||||
|         return True | ||||
|          | ||||
|     except subprocess.CalledProcessError as e: | ||||
|         print(f"❌ 安装失败: {e}") | ||||
|         print(f"错误输出: {e.stderr}") | ||||
|         return False | ||||
|     except Exception as e: | ||||
|         print(f"❌ 未知错误: {e}") | ||||
|         return False | ||||
| 
 | ||||
| def main(): | ||||
|     """主函数""" | ||||
|     print("🏛️ 稷下学宫Swarm环境安装") | ||||
|     print("=" * 40) | ||||
|      | ||||
|     # 检查是否已安装 | ||||
|     try: | ||||
|         import swarm | ||||
|         print("✅ OpenAI Swarm已安装") | ||||
|         print(f"📦 版本: {getattr(swarm, '__version__', '未知')}") | ||||
|          | ||||
|         choice = input("是否重新安装?(y/N): ").strip().lower() | ||||
|         if choice not in ['y', 'yes']: | ||||
|             print("🎉 安装检查完成") | ||||
|             return | ||||
|     except ImportError: | ||||
|         print("📦 OpenAI Swarm未安装,开始安装...") | ||||
|      | ||||
|     # 安装Swarm | ||||
|     success = install_swarm() | ||||
|      | ||||
|     if success: | ||||
|         print("\n🎉 安装完成!现在可以使用Swarm八仙论道了") | ||||
|         print("💡 使用方法:") | ||||
|         print("   python src/jixia/debates/swarm_debate.py") | ||||
|         print("   或在Streamlit应用中选择'Swarm模式'") | ||||
|     else: | ||||
|         print("\n❌ 安装失败,请手动安装:") | ||||
|         print("   pip install git+https://github.com/openai/swarm.git") | ||||
| 
 | ||||
| if __name__ == "__main__": | ||||
|     main() | ||||
|  | @ -1,148 +0,0 @@ | |||
| const items = $input.all(); | ||||
| const results = []; | ||||
| 
 | ||||
| // 改进的哈希函数 - 基于内容生成稳定的ID
 | ||||
| function generateStableId(title, pubDate, content) { | ||||
|     const normalizedTitle = title.trim().toLowerCase(); | ||||
|     const contentHash = content ? content.substring(0, 100) : ''; | ||||
|     const dateStr = pubDate || ''; | ||||
|      | ||||
|     const combined = normalizedTitle + '|' + dateStr + '|' + contentHash; | ||||
|      | ||||
|     let hash = 0; | ||||
|     for (let i = 0; i < combined.length; i++) { | ||||
|         const char = combined.charCodeAt(i); | ||||
|         hash = ((hash << 5) - hash) + char; | ||||
|         hash = hash & hash; | ||||
|     } | ||||
|     return Math.abs(hash).toString(16); | ||||
| } | ||||
| 
 | ||||
| console.log(`开始处理 ${items.length} 条RSS数据`); | ||||
| 
 | ||||
| // 用于本次执行内去重
 | ||||
| const processedInThisRun = new Set(); | ||||
| 
 | ||||
| // 处理每个RSS项目
 | ||||
| for (const item of items) { | ||||
|     const data = item.json; | ||||
|      | ||||
|     // 跳过无效数据
 | ||||
|     if (!data.title) { | ||||
|         console.log('跳过无标题数据'); | ||||
|         continue; | ||||
|     } | ||||
|      | ||||
|     // 生成稳定的文章ID
 | ||||
|     const stableId = generateStableId( | ||||
|         data.title,  | ||||
|         data.isoDate || data.pubDate, | ||||
|         data['content:encodedSnippet'] || data.contentSnippet || '' | ||||
|     ); | ||||
|      | ||||
|     // 生成内容哈希
 | ||||
|     const contentHash = generateStableId( | ||||
|         data['content:encodedSnippet'] || data.contentSnippet || '',  | ||||
|         '',  | ||||
|         '' | ||||
|     ); | ||||
|      | ||||
|     // 准备文章数据
 | ||||
|     const articleData = { | ||||
|         article_id: stableId, | ||||
|         title: data.title, | ||||
|         content: data['content:encodedSnippet'] || data.contentSnippet || '', | ||||
|         content_hash: contentHash, | ||||
|         published_time: data.isoDate || data.pubDate || new Date().toISOString(), | ||||
|         source_url: data.link || '', | ||||
|         processed: false, | ||||
|         created_at: new Date().toISOString(), | ||||
|         last_updated: new Date().toISOString() | ||||
|     }; | ||||
|      | ||||
|     try { | ||||
|         // 使用upsert操作,避免重复插入
 | ||||
|         const result = await mongoClient.db('taigong').collection('articles').updateOne( | ||||
|             {  | ||||
|                 $or: [ | ||||
|                     { article_id: stableId }, | ||||
|                     { title: data.title } | ||||
|                 ] | ||||
|             }, | ||||
|             { | ||||
|                 $setOnInsert: { | ||||
|                     article_id: stableId, | ||||
|                     title: data.title, | ||||
|                     content: articleData.content, | ||||
|                     content_hash: contentHash, | ||||
|                     published_time: articleData.published_time, | ||||
|                     source_url: articleData.source_url, | ||||
|                     processed: false, | ||||
|                     created_at: articleData.created_at | ||||
|                 }, | ||||
|                 $set: { | ||||
|                     last_updated: new Date().toISOString() | ||||
|                 } | ||||
|             }, | ||||
|             { upsert: true } | ||||
|         ); | ||||
|          | ||||
|         if (result.upsertedCount > 0) { | ||||
|             console.log('✅ 新增文章:', data.title); | ||||
|             results.push({ | ||||
|                 json: { | ||||
|                     action: 'inserted', | ||||
|                     article_id: stableId, | ||||
|                     title: data.title, | ||||
|                     status: 'success' | ||||
|                 } | ||||
|             }); | ||||
|         } else if (result.modifiedCount > 0) { | ||||
|             console.log('🔄 更新文章:', data.title); | ||||
|             results.push({ | ||||
|                 json: { | ||||
|                     action: 'updated', | ||||
|                     article_id: stableId, | ||||
|                     title: data.title, | ||||
|                     status: 'success' | ||||
|                 } | ||||
|             }); | ||||
|         } else { | ||||
|             console.log('⏭️  文章已存在,跳过:', data.title); | ||||
|         } | ||||
|          | ||||
|     } catch (error) { | ||||
|         console.error('❌ 处理文章失败:', data.title, error.message); | ||||
|         results.push({ | ||||
|             json: { | ||||
|                 action: 'error', | ||||
|                 title: data.title, | ||||
|                 error: error.message, | ||||
|                 status: 'failed' | ||||
|             } | ||||
|         }); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| console.log(`处理完成: 原始${items.length}条, 成功处理${results.length}条`); | ||||
| 
 | ||||
| // 统计结果
 | ||||
| const stats = results.reduce((acc, item) => { | ||||
|     acc[item.json.action] = (acc[item.json.action] || 0) + 1; | ||||
|     return acc; | ||||
| }, {}); | ||||
| 
 | ||||
| console.log('处理统计:', stats); | ||||
| 
 | ||||
| // 如果没有任何结果,返回一个空的成功状态
 | ||||
| if (results.length === 0) { | ||||
|     return [{ | ||||
|         json: { | ||||
|             message: '没有新数据需要处理', | ||||
|             total_processed: items.length, | ||||
|             status: 'completed' | ||||
|         } | ||||
|     }]; | ||||
| } | ||||
| 
 | ||||
| return results; | ||||
|  | @ -1,85 +0,0 @@ | |||
| const items = $input.all(); | ||||
| const processedItems = []; | ||||
| 
 | ||||
| // 改进的哈希函数 - 基于内容生成稳定的ID
 | ||||
| function generateStableId(title, pubDate, content) { | ||||
|     const normalizedTitle = title.trim().toLowerCase(); | ||||
|     const contentHash = content ? content.substring(0, 100) : ''; | ||||
|     const dateStr = pubDate || ''; | ||||
|      | ||||
|     const combined = normalizedTitle + '|' + dateStr + '|' + contentHash; | ||||
|      | ||||
|     let hash = 0; | ||||
|     for (let i = 0; i < combined.length; i++) { | ||||
|         const char = combined.charCodeAt(i); | ||||
|         hash = ((hash << 5) - hash) + char; | ||||
|         hash = hash & hash; | ||||
|     } | ||||
|     return Math.abs(hash).toString(16); | ||||
| } | ||||
| 
 | ||||
| // 1. 从数据库查询已存在的文章ID和标题
 | ||||
| const existingArticles = new Set(); | ||||
| try { | ||||
|     const existing = await mongoClient.db('taigong').collection('articles') | ||||
|         .find({}, { projection: { article_id: 1, title: 1, content_hash: 1 } }) | ||||
|         .toArray(); | ||||
|      | ||||
|     existing.forEach(article => { | ||||
|         existingArticles.add(article.article_id); | ||||
|         // 同时用标题做备用检查
 | ||||
|         existingArticles.add(article.title); | ||||
|     }); | ||||
|      | ||||
|     console.log(`数据库中已有 ${existing.length} 篇文章`); | ||||
| } catch (error) { | ||||
|     console.log('查询现有文章失败:', error); | ||||
| } | ||||
| 
 | ||||
| // 2. 处理新数据
 | ||||
| for (const item of items) { | ||||
|     const data = item.json; | ||||
|      | ||||
|     // 跳过无效数据
 | ||||
|     if (!data.title) continue; | ||||
|      | ||||
|     // 生成稳定的文章ID
 | ||||
|     const stableId = generateStableId( | ||||
|         data.title,  | ||||
|         data.isoDate || data.pubDate, | ||||
|         data['content:encodedSnippet'] || data.contentSnippet || '' | ||||
|     ); | ||||
|      | ||||
|     // 检查是否已存在(用ID和标题双重检查)
 | ||||
|     if (existingArticles.has(stableId) || existingArticles.has(data.title)) { | ||||
|         console.log('跳过重复文章:', data.title); | ||||
|         continue; | ||||
|     } | ||||
|      | ||||
|     // 生成内容哈希用于后续去重检查
 | ||||
|     const contentHash = generateStableId( | ||||
|         data['content:encodedSnippet'] || data.contentSnippet || '',  | ||||
|         '',  | ||||
|         '' | ||||
|     ); | ||||
|      | ||||
|     const processedItem = { | ||||
|         article_id: stableId,  // 使用稳定ID
 | ||||
|         title: data.title, | ||||
|         content: data['content:encodedSnippet'] || data.contentSnippet || '', | ||||
|         content_hash: contentHash,  // 新增:内容哈希
 | ||||
|         published_time: data.isoDate || data.pubDate || new Date().toISOString(), | ||||
|         source_url: data.link || '',  // 新增:源链接
 | ||||
|         processed: false, | ||||
|         created_at: new Date().toISOString(), | ||||
|         last_updated: new Date().toISOString()  // 新增:更新时间
 | ||||
|     }; | ||||
|      | ||||
|     processedItems.push({ json: processedItem }); | ||||
|     // 添加到已存在集合,避免本次执行内重复
 | ||||
|     existingArticles.add(stableId); | ||||
|     existingArticles.add(data.title); | ||||
| } | ||||
| 
 | ||||
| console.log(`处理完成: 原始${items.length}条, 去重后${processedItems.length}条`); | ||||
| return processedItems; | ||||
|  | @ -1,85 +0,0 @@ | |||
| const items = $input.all(); | ||||
| const results = []; | ||||
| 
 | ||||
| // 如果没有数据需要插入
 | ||||
| if (items.length === 0 || (items.length === 1 && items[0].json.status === 'no_new_data')) { | ||||
|     console.log('没有新数据需要插入'); | ||||
|     return items; | ||||
| } | ||||
| 
 | ||||
| console.log(`准备插入 ${items.length} 条新文章`); | ||||
| 
 | ||||
| // 准备批量插入的数据
 | ||||
| const documentsToInsert = items.map(item => item.json); | ||||
| 
 | ||||
| try { | ||||
|     // 批量插入,因为已经确保了唯一性,所以直接插入
 | ||||
|     const result = await mongoClient.db('taigong').collection('articles').insertMany( | ||||
|         documentsToInsert, | ||||
|         { ordered: false } // 即使某条失败也继续插入其他的
 | ||||
|     ); | ||||
|      | ||||
|     console.log(`✅ 成功插入 ${result.insertedCount} 条文章`); | ||||
|      | ||||
|     // 返回插入结果
 | ||||
|     for (let i = 0; i < documentsToInsert.length; i++) { | ||||
|         const doc = documentsToInsert[i]; | ||||
|         const insertedId = result.insertedIds[i]; | ||||
|          | ||||
|         results.push({ | ||||
|             json: { | ||||
|                 action: 'inserted', | ||||
|                 sequence_id: doc.sequence_id, | ||||
|                 article_id: doc.article_id, | ||||
|                 title: doc.title, | ||||
|                 mongodb_id: insertedId, | ||||
|                 status: 'success' | ||||
|             } | ||||
|         }); | ||||
|     } | ||||
|      | ||||
| } catch (error) { | ||||
|     console.error('❌ 批量插入失败:', error.message); | ||||
|      | ||||
|     // 如果批量插入失败,尝试逐条插入
 | ||||
|     console.log('尝试逐条插入...'); | ||||
|      | ||||
|     for (const doc of documentsToInsert) { | ||||
|         try { | ||||
|             const result = await mongoClient.db('taigong').collection('articles').insertOne(doc); | ||||
|              | ||||
|             console.log(`✅ 单条插入成功: ${doc.article_id}`); | ||||
|             results.push({ | ||||
|                 json: { | ||||
|                     action: 'inserted', | ||||
|                     sequence_id: doc.sequence_id, | ||||
|                     article_id: doc.article_id, | ||||
|                     title: doc.title, | ||||
|                     mongodb_id: result.insertedId, | ||||
|                     status: 'success' | ||||
|                 } | ||||
|             }); | ||||
|              | ||||
|         } catch (singleError) { | ||||
|             console.error(`❌ 单条插入失败 ${doc.article_id}:`, singleError.message); | ||||
|             results.push({ | ||||
|                 json: { | ||||
|                     action: 'error', | ||||
|                     sequence_id: doc.sequence_id, | ||||
|                     article_id: doc.article_id, | ||||
|                     title: doc.title, | ||||
|                     error: singleError.message, | ||||
|                     status: 'failed' | ||||
|                 } | ||||
|             }); | ||||
|         } | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| // 统计结果
 | ||||
| const successCount = results.filter(r => r.json.status === 'success').length; | ||||
| const failCount = results.filter(r => r.json.status === 'failed').length; | ||||
| 
 | ||||
| console.log(`插入完成: 成功 ${successCount} 条, 失败 ${failCount} 条`); | ||||
| 
 | ||||
| return results; | ||||
|  | @ -1,39 +0,0 @@ | |||
| const items = $input.all(); | ||||
| 
 | ||||
| console.log(`原始数据: ${items.length} 条`); | ||||
| 
 | ||||
| // 本批次内去重
 | ||||
| const seenTitles = new Set(); | ||||
| const uniqueItems = []; | ||||
| 
 | ||||
| // 生成起始ID(基于时间戳,确保每次运行都不同)
 | ||||
| let nextId = Math.floor(Date.now() / 1000); | ||||
| 
 | ||||
| for (const item of items) { | ||||
|     const data = item.json; | ||||
| 
 | ||||
|     // 跳过无效数据
 | ||||
|     if (!data.title) continue; | ||||
| 
 | ||||
|     // 本批次内去重
 | ||||
|     if (seenTitles.has(data.title)) { | ||||
|         console.log('⏭️  本批次重复,跳过:', data.title); | ||||
|         continue; | ||||
|     } | ||||
| 
 | ||||
|     const newsItem = { | ||||
|         id: nextId, | ||||
|         title: data.title, | ||||
|         published_time: data.isoDate || data.pubDate || new Date().toISOString(), | ||||
|         source_url: data.link || '' | ||||
|     }; | ||||
| 
 | ||||
|     uniqueItems.push({ json: newsItem }); | ||||
|     seenTitles.add(data.title); | ||||
| 
 | ||||
|     console.log(`✅ ID ${nextId}: ${data.title}`); | ||||
|     nextId++; | ||||
| } | ||||
| 
 | ||||
| console.log(`本批次去重后: ${uniqueItems.length} 条`); | ||||
| return uniqueItems; | ||||
|  | @ -1,54 +0,0 @@ | |||
| // n8n MongoDB插入节点代码
 | ||||
| const items = $input.all(); | ||||
| const results = []; | ||||
| 
 | ||||
| for (const item of items) { | ||||
|     const data = item.json; | ||||
|      | ||||
|     try { | ||||
|         // 使用upsert操作,避免重复插入
 | ||||
|         const result = await mongoClient.db('taigong').collection('articles').updateOne( | ||||
|             {  | ||||
|                 $or: [ | ||||
|                     { article_id: data.article_id }, | ||||
|                     { title: data.title } | ||||
|                 ] | ||||
|             }, | ||||
|             { | ||||
|                 $setOnInsert: { | ||||
|                     article_id: data.article_id, | ||||
|                     title: data.title, | ||||
|                     content: data.content, | ||||
|                     content_hash: data.content_hash, | ||||
|                     published_time: data.published_time, | ||||
|                     source_url: data.source_url, | ||||
|                     processed: data.processed, | ||||
|                     created_at: data.created_at | ||||
|                 }, | ||||
|                 $set: { | ||||
|                     last_updated: new Date().toISOString() | ||||
|                 } | ||||
|             }, | ||||
|             { upsert: true } | ||||
|         ); | ||||
|          | ||||
|         if (result.upsertedCount > 0) { | ||||
|             console.log('新增文章:', data.title); | ||||
|             results.push({ | ||||
|                 json: { | ||||
|                     action: 'inserted', | ||||
|                     article_id: data.article_id, | ||||
|                     title: data.title | ||||
|                 } | ||||
|             }); | ||||
|         } else { | ||||
|             console.log('文章已存在,跳过:', data.title); | ||||
|         } | ||||
|          | ||||
|     } catch (error) { | ||||
|         console.error('插入文章失败:', data.title, error); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| console.log(`成功处理 ${results.length} 篇新文章`); | ||||
| return results; | ||||
|  | @ -1,119 +0,0 @@ | |||
| const items = $input.all(); | ||||
| const processedItems = []; | ||||
| 
 | ||||
| // 获取当前最大流水号
 | ||||
| async function getCurrentMaxId() { | ||||
|     try { | ||||
|         const result = await mongoClient.db('taigong').collection('articles') | ||||
|             .findOne({}, {  | ||||
|                 sort: { sequence_id: -1 },  | ||||
|                 projection: { sequence_id: 1 }  | ||||
|             }); | ||||
|          | ||||
|         return result ? result.sequence_id : 0; | ||||
|     } catch (error) { | ||||
|         console.log('获取最大流水号失败,从1开始:', error.message); | ||||
|         return 0; | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| // 获取已存在的文章标题集合(用于去重检查)
 | ||||
| async function getExistingTitles() { | ||||
|     try { | ||||
|         const existing = await mongoClient.db('taigong').collection('articles') | ||||
|             .find({}, { projection: { title: 1 } }) | ||||
|             .toArray(); | ||||
|          | ||||
|         return new Set(existing.map(doc => doc.title)); | ||||
|     } catch (error) { | ||||
|         console.log('获取已存在标题失败:', error.message); | ||||
|         return new Set(); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| // 生成内容哈希(用于内容变化检测)
 | ||||
| function generateContentHash(content) { | ||||
|     if (!content) return ''; | ||||
|      | ||||
|     let hash = 0; | ||||
|     const str = content.substring(0, 200); // 取前200字符
 | ||||
|     for (let i = 0; i < str.length; i++) { | ||||
|         const char = str.charCodeAt(i); | ||||
|         hash = ((hash << 5) - hash) + char; | ||||
|         hash = hash & hash; | ||||
|     } | ||||
|     return Math.abs(hash).toString(16); | ||||
| } | ||||
| 
 | ||||
| console.log(`开始处理 ${items.length} 条RSS数据`); | ||||
| 
 | ||||
| // 1. 获取当前最大流水号
 | ||||
| const currentMaxId = await getCurrentMaxId(); | ||||
| console.log(`当前数据库最大流水号: ${currentMaxId}`); | ||||
| 
 | ||||
| // 2. 获取已存在的文章标题
 | ||||
| const existingTitles = await getExistingTitles(); | ||||
| console.log(`数据库中已有 ${existingTitles.size} 篇文章`); | ||||
| 
 | ||||
| // 3. 处理新数据,分配流水号
 | ||||
| let nextSequenceId = currentMaxId + 1; | ||||
| const seenTitlesInBatch = new Set(); // 本批次内去重
 | ||||
| 
 | ||||
| for (const item of items) { | ||||
|     const data = item.json; | ||||
|      | ||||
|     // 跳过无效数据
 | ||||
|     if (!data.title) { | ||||
|         console.log('跳过无标题数据'); | ||||
|         continue; | ||||
|     } | ||||
|      | ||||
|     // 检查是否已存在(数据库 + 本批次)
 | ||||
|     if (existingTitles.has(data.title) || seenTitlesInBatch.has(data.title)) { | ||||
|         console.log('⏭️  跳过重复文章:', data.title); | ||||
|         continue; | ||||
|     } | ||||
|      | ||||
|     // 分配新的流水号
 | ||||
|     const sequenceId = nextSequenceId++; | ||||
|      | ||||
|     // 生成文章数据
 | ||||
|     const articleData = { | ||||
|         sequence_id: sequenceId,  // 主键:流水号
 | ||||
|         article_id: `NEWS_${sequenceId.toString().padStart(8, '0')}`, // 格式化ID:NEWS_00000001
 | ||||
|         title: data.title, | ||||
|         content: data['content:encodedSnippet'] || data.contentSnippet || '', | ||||
|         content_hash: generateContentHash(data['content:encodedSnippet'] || data.contentSnippet || ''), | ||||
|         published_time: data.isoDate || data.pubDate || new Date().toISOString(), | ||||
|         source_url: data.link || '', | ||||
|         rss_source: data.meta?.title || 'unknown', // RSS源名称
 | ||||
|         processed: false, | ||||
|         created_at: new Date().toISOString(), | ||||
|         batch_id: Date.now().toString() // 批次ID,用于追踪
 | ||||
|     }; | ||||
|      | ||||
|     processedItems.push({ json: articleData }); | ||||
|     seenTitlesInBatch.add(data.title); | ||||
|      | ||||
|     console.log(`✅ 分配流水号 ${sequenceId}: ${data.title}`); | ||||
| } | ||||
| 
 | ||||
| console.log(`流水号分配完成:`); | ||||
| console.log(`  原始数据: ${items.length} 条`); | ||||
| console.log(`  跳过重复: ${items.length - processedItems.length} 条`); | ||||
| console.log(`  新增数据: ${processedItems.length} 条`); | ||||
| console.log(`  流水号范围: ${currentMaxId + 1} - ${nextSequenceId - 1}`); | ||||
| 
 | ||||
| // 如果没有新数据,返回空结果
 | ||||
| if (processedItems.length === 0) { | ||||
|     return [{ | ||||
|         json: { | ||||
|             message: '没有新数据需要处理', | ||||
|             current_max_id: currentMaxId, | ||||
|             total_articles_in_db: existingTitles.size, | ||||
|             status: 'no_new_data' | ||||
|         } | ||||
|     }]; | ||||
| } | ||||
| 
 | ||||
| return processedItems; | ||||
|  | @ -1,52 +0,0 @@ | |||
| const items = $input.all(); | ||||
| 
 | ||||
| // 简单哈希函数
 | ||||
| function simpleHash(str) { | ||||
|     let hash = 0; | ||||
|     for (let i = 0; i < str.length; i++) { | ||||
|         const char = str.charCodeAt(i); | ||||
|         hash = ((hash << 5) - hash) + char; | ||||
|         hash = hash & hash; | ||||
|     } | ||||
|     return Math.abs(hash).toString(16); | ||||
| } | ||||
| 
 | ||||
| console.log(`原始数据: ${items.length} 条`); | ||||
| 
 | ||||
| // 用标题去重
 | ||||
| const seenTitles = new Set(); | ||||
| const uniqueItems = []; | ||||
| 
 | ||||
| for (const item of items) { | ||||
|     const data = item.json; | ||||
| 
 | ||||
|     // 跳过无效数据
 | ||||
|     if (!data.title) continue; | ||||
| 
 | ||||
|     // 本批次内去重
 | ||||
|     if (seenTitles.has(data.title)) { | ||||
|         console.log('跳过重复:', data.title); | ||||
|         continue; | ||||
|     } | ||||
| 
 | ||||
|     // 生成稳定ID
 | ||||
|     const stableId = simpleHash(data.title + (data.isoDate || data.pubDate || '')); | ||||
| 
 | ||||
|     const processedItem = { | ||||
|         article_id: stableId, | ||||
|         title: data.title, | ||||
|         content: data['content:encodedSnippet'] || data.contentSnippet || '', | ||||
|         published_time: data.isoDate || data.pubDate || new Date().toISOString(), | ||||
|         source_url: data.link || '', | ||||
|         processed: false, | ||||
|         created_at: new Date().toISOString() | ||||
|     }; | ||||
| 
 | ||||
|     uniqueItems.push({ json: processedItem }); | ||||
|     seenTitles.add(data.title); | ||||
| 
 | ||||
|     console.log(`✅ 处理: ${data.title}`); | ||||
| } | ||||
| 
 | ||||
| console.log(`去重后: ${uniqueItems.length} 条`); | ||||
| return uniqueItems; | ||||
|  | @ -1,163 +0,0 @@ | |||
| const items = $input.all(); | ||||
| const results = []; | ||||
| 
 | ||||
| // 通用MongoDB连接获取函数
 | ||||
| function getMongoConnection() { | ||||
|     // 尝试不同的MongoDB连接变量名
 | ||||
|     if (typeof mongoClient !== 'undefined') return mongoClient; | ||||
|     if (typeof mongo !== 'undefined') return mongo; | ||||
|     if (typeof db !== 'undefined') return db; | ||||
|     if (typeof $mongo !== 'undefined') return $mongo; | ||||
|     if (typeof client !== 'undefined') return client; | ||||
|      | ||||
|     throw new Error('找不到MongoDB连接对象,请检查n8n MongoDB节点配置'); | ||||
| } | ||||
| 
 | ||||
| // 改进的哈希函数 - 基于内容生成稳定的ID
 | ||||
| function generateStableId(title, pubDate, content) { | ||||
|     const normalizedTitle = title.trim().toLowerCase(); | ||||
|     const contentHash = content ? content.substring(0, 100) : ''; | ||||
|     const dateStr = pubDate || ''; | ||||
|      | ||||
|     const combined = normalizedTitle + '|' + dateStr + '|' + contentHash; | ||||
|      | ||||
|     let hash = 0; | ||||
|     for (let i = 0; i < combined.length; i++) { | ||||
|         const char = combined.charCodeAt(i); | ||||
|         hash = ((hash << 5) - hash) + char; | ||||
|         hash = hash & hash; | ||||
|     } | ||||
|     return Math.abs(hash).toString(16); | ||||
| } | ||||
| 
 | ||||
| console.log(`开始处理 ${items.length} 条RSS数据`); | ||||
| 
 | ||||
| // 获取MongoDB连接
 | ||||
| let mongoConnection; | ||||
| try { | ||||
|     mongoConnection = getMongoConnection(); | ||||
|     console.log('✅ MongoDB连接获取成功'); | ||||
| } catch (error) { | ||||
|     console.error('❌ MongoDB连接失败:', error.message); | ||||
|     return [{ | ||||
|         json: { | ||||
|             error: 'MongoDB连接失败', | ||||
|             message: error.message, | ||||
|             status: 'connection_failed' | ||||
|         } | ||||
|     }]; | ||||
| } | ||||
| 
 | ||||
| // 用于本次执行内去重
 | ||||
| const processedInThisRun = new Set(); | ||||
| 
 | ||||
| // 处理每个RSS项目
 | ||||
| for (const item of items) { | ||||
|     const data = item.json; | ||||
|      | ||||
|     // 跳过无效数据
 | ||||
|     if (!data.title) { | ||||
|         console.log('跳过无标题数据'); | ||||
|         continue; | ||||
|     } | ||||
|      | ||||
|     // 本次执行内去重检查
 | ||||
|     if (processedInThisRun.has(data.title)) { | ||||
|         console.log('⏭️  本次执行内重复,跳过:', data.title); | ||||
|         continue; | ||||
|     } | ||||
|      | ||||
|     // 生成稳定的文章ID
 | ||||
|     const stableId = generateStableId( | ||||
|         data.title,  | ||||
|         data.isoDate || data.pubDate, | ||||
|         data['content:encodedSnippet'] || data.contentSnippet || '' | ||||
|     ); | ||||
|      | ||||
|     // 生成内容哈希
 | ||||
|     const contentHash = generateStableId( | ||||
|         data['content:encodedSnippet'] || data.contentSnippet || '',  | ||||
|         '',  | ||||
|         '' | ||||
|     ); | ||||
|      | ||||
|     // 准备文章数据
 | ||||
|     const articleData = { | ||||
|         article_id: stableId, | ||||
|         title: data.title, | ||||
|         content: data['content:encodedSnippet'] || data.contentSnippet || '', | ||||
|         content_hash: contentHash, | ||||
|         published_time: data.isoDate || data.pubDate || new Date().toISOString(), | ||||
|         source_url: data.link || '', | ||||
|         rss_source: data.meta?.title || 'unknown', | ||||
|         processed: false, | ||||
|         created_at: new Date().toISOString(), | ||||
|         last_updated: new Date().toISOString() | ||||
|     }; | ||||
|      | ||||
|     try { | ||||
|         // 检查数据库中是否已存在
 | ||||
|         const existing = await mongoConnection.db('taigong').collection('articles').findOne({ | ||||
|             $or: [ | ||||
|                 { article_id: stableId }, | ||||
|                 { title: data.title } | ||||
|             ] | ||||
|         }); | ||||
|          | ||||
|         if (existing) { | ||||
|             console.log('⏭️  数据库中已存在,跳过:', data.title); | ||||
|             continue; | ||||
|         } | ||||
|          | ||||
|         // 插入新文章
 | ||||
|         const result = await mongoConnection.db('taigong').collection('articles').insertOne(articleData); | ||||
|          | ||||
|         console.log('✅ 新增文章:', data.title); | ||||
|         results.push({ | ||||
|             json: { | ||||
|                 action: 'inserted', | ||||
|                 article_id: stableId, | ||||
|                 title: data.title, | ||||
|                 mongodb_id: result.insertedId, | ||||
|                 status: 'success' | ||||
|             } | ||||
|         }); | ||||
|          | ||||
|         // 添加到本次执行的去重集合
 | ||||
|         processedInThisRun.add(data.title); | ||||
|          | ||||
|     } catch (error) { | ||||
|         console.error('❌ 处理文章失败:', data.title, error.message); | ||||
|         results.push({ | ||||
|             json: { | ||||
|                 action: 'error', | ||||
|                 title: data.title, | ||||
|                 error: error.message, | ||||
|                 status: 'failed' | ||||
|             } | ||||
|         }); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| console.log(`处理完成: 原始${items.length}条, 成功处理${results.length}条`); | ||||
| 
 | ||||
| // 统计结果
 | ||||
| const stats = results.reduce((acc, item) => { | ||||
|     acc[item.json.action] = (acc[item.json.action] || 0) + 1; | ||||
|     return acc; | ||||
| }, {}); | ||||
| 
 | ||||
| console.log('处理统计:', stats); | ||||
| 
 | ||||
| // 如果没有任何结果,返回一个空的成功状态
 | ||||
| if (results.length === 0) { | ||||
|     return [{ | ||||
|         json: { | ||||
|             message: '没有新数据需要处理', | ||||
|             total_processed: items.length, | ||||
|             status: 'completed' | ||||
|         } | ||||
|     }]; | ||||
| } | ||||
| 
 | ||||
| return results; | ||||
|  | @ -1,163 +0,0 @@ | |||
| #!/usr/bin/env python3 | ||||
| """ | ||||
| 测试OpenRouter API连接 | ||||
| 重构版本:使用统一配置管理 | ||||
| """ | ||||
| 
 | ||||
| import requests | ||||
| from typing import Dict, Any | ||||
| 
 | ||||
| def test_openrouter_api() -> bool: | ||||
|     """ | ||||
|     测试OpenRouter API连接 | ||||
|      | ||||
|     Returns: | ||||
|         测试是否成功 | ||||
|     """ | ||||
|     # 使用统一配置管理 | ||||
|     try: | ||||
|         from config.doppler_config import get_openrouter_key | ||||
|         api_key = get_openrouter_key() | ||||
|     except ImportError: | ||||
|         # 如果配置模块不可用,使用环境变量 | ||||
|         import os | ||||
|         api_key = os.getenv('OPENROUTER_API_KEY_1') | ||||
|     except Exception as e: | ||||
|         print(f"❌ 无法获取API密钥: {e}") | ||||
|         return False | ||||
|      | ||||
|     if not api_key: | ||||
|         print("❌ 未找到OpenRouter API密钥") | ||||
|         print("请确保已配置 OPENROUTER_API_KEY_1 环境变量") | ||||
|         return False | ||||
|      | ||||
|     print(f"🔑 使用API密钥: {api_key[:20]}...") | ||||
|      | ||||
|     # 测试API调用 | ||||
|     url = "https://openrouter.ai/api/v1/chat/completions" | ||||
|     headers = { | ||||
|         "Authorization": f"Bearer {api_key}", | ||||
|         "HTTP-Referer": "https://github.com/ben/liurenchaxin", | ||||
|         "X-Title": "Jixia Academy Debate System", | ||||
|         "Content-Type": "application/json" | ||||
|     } | ||||
|      | ||||
|     data = { | ||||
|         "model": "openai/gpt-3.5-turbo", | ||||
|         "messages": [ | ||||
|             {"role": "user", "content": "你好,请简单回复一下测试连接"} | ||||
|         ], | ||||
|         "max_tokens": 50 | ||||
|     } | ||||
|      | ||||
|     try: | ||||
|         print("📡 正在测试API连接...") | ||||
|         response = requests.post(url, headers=headers, json=data, timeout=30) | ||||
|         print(f"📡 响应状态码: {response.status_code}") | ||||
|          | ||||
|         if response.status_code == 200: | ||||
|             result = response.json() | ||||
|             print("✅ OpenRouter API连接成功!") | ||||
|             if 'choices' in result and len(result['choices']) > 0: | ||||
|                 content = result['choices'][0]['message']['content'] | ||||
|                 print(f"📝 AI回复: {content}") | ||||
|             else: | ||||
|                 print("📝 API响应格式异常,但连接成功") | ||||
|             return True | ||||
|         else: | ||||
|             print(f"❌ API调用失败: HTTP {response.status_code}") | ||||
|             print(f"错误详情: {response.text}") | ||||
|             return False | ||||
|              | ||||
|     except requests.exceptions.Timeout: | ||||
|         print("❌ 请求超时,请检查网络连接") | ||||
|         return False | ||||
|     except requests.exceptions.RequestException as e: | ||||
|         print(f"❌ 网络请求异常: {e}") | ||||
|         return False | ||||
|     except Exception as e: | ||||
|         print(f"❌ 未知异常: {e}") | ||||
|         return False | ||||
| 
 | ||||
| def test_rapidapi_connection() -> bool: | ||||
|     """ | ||||
|     测试RapidAPI连接 | ||||
|      | ||||
|     Returns: | ||||
|         测试是否成功 | ||||
|     """ | ||||
|     try: | ||||
|         from config.doppler_config import get_rapidapi_key | ||||
|         api_key = get_rapidapi_key() | ||||
|     except ImportError: | ||||
|         import os | ||||
|         api_key = os.getenv('RAPIDAPI_KEY') | ||||
|     except Exception as e: | ||||
|         print(f"❌ 无法获取RapidAPI密钥: {e}") | ||||
|         return False | ||||
|      | ||||
|     if not api_key: | ||||
|         print("❌ 未找到RapidAPI密钥") | ||||
|         return False | ||||
|      | ||||
|     print(f"🔑 测试RapidAPI连接...") | ||||
|      | ||||
|     # 测试一个简单的API端点 | ||||
|     url = "https://yahoo-finance15.p.rapidapi.com/api/yahoo/qu/quote/AAPL" | ||||
|     headers = { | ||||
|         'X-RapidAPI-Key': api_key, | ||||
|         'X-RapidAPI-Host': 'yahoo-finance15.p.rapidapi.com' | ||||
|     } | ||||
|      | ||||
|     try: | ||||
|         response = requests.get(url, headers=headers, timeout=10) | ||||
|         if response.status_code == 200: | ||||
|             print("✅ RapidAPI连接成功!") | ||||
|             return True | ||||
|         else: | ||||
|             print(f"❌ RapidAPI连接失败: HTTP {response.status_code}") | ||||
|             return False | ||||
|     except Exception as e: | ||||
|         print(f"❌ RapidAPI连接异常: {e}") | ||||
|         return False | ||||
| 
 | ||||
| def main(): | ||||
|     """主函数 - 运行所有API连接测试""" | ||||
|     print("🧪 API连接测试套件") | ||||
|     print("=" * 50) | ||||
|      | ||||
|     # 测试配置验证 | ||||
|     try: | ||||
|         from config.doppler_config import validate_config | ||||
|         print("\n🔧 验证配置...") | ||||
|         config_valid = validate_config() | ||||
|     except ImportError: | ||||
|         print("⚠️ 配置模块不可用,跳过配置验证") | ||||
|         config_valid = True | ||||
|      | ||||
|     # 测试OpenRouter API | ||||
|     print("\n🤖 测试OpenRouter API...") | ||||
|     openrouter_success = test_openrouter_api() | ||||
|      | ||||
|     # 测试RapidAPI | ||||
|     print("\n📊 测试RapidAPI...") | ||||
|     rapidapi_success = test_rapidapi_api() | ||||
|      | ||||
|     # 总结测试结果 | ||||
|     print("\n" + "=" * 50) | ||||
|     print("📋 测试结果总结:") | ||||
|     print(f"  配置验证: {'✅ 通过' if config_valid else '❌ 失败'}") | ||||
|     print(f"  OpenRouter API: {'✅ 通过' if openrouter_success else '❌ 失败'}") | ||||
|     print(f"  RapidAPI: {'✅ 通过' if rapidapi_success else '❌ 失败'}") | ||||
|      | ||||
|     all_passed = config_valid and openrouter_success and rapidapi_success | ||||
|     if all_passed: | ||||
|         print("\n🎉 所有API连接测试通过!系统已准备就绪。") | ||||
|     else: | ||||
|         print("\n⚠️ 部分测试失败,请检查配置和网络连接。") | ||||
|      | ||||
|     return all_passed | ||||
| 
 | ||||
| if __name__ == "__main__": | ||||
|     success = main() | ||||
|     exit(0 if success else 1) | ||||
|  | @ -1,297 +0,0 @@ | |||
| #!/usr/bin/env python3 | ||||
| """ | ||||
| RapidAPI库存测试脚本 | ||||
| 自动测试所有订阅的API服务,生成可用性报告 | ||||
| """ | ||||
| 
 | ||||
| import requests | ||||
| import json | ||||
| import time | ||||
| from datetime import datetime | ||||
| from typing import Dict, List, Any | ||||
| import os | ||||
| 
 | ||||
| class RapidAPITester: | ||||
|     """RapidAPI测试器""" | ||||
|      | ||||
|     def __init__(self): | ||||
|         """初始化测试器""" | ||||
|         # 从环境变量获取API密钥 | ||||
|         self.api_key = os.getenv('RAPIDAPI_KEY') | ||||
|         if not self.api_key: | ||||
|             raise ValueError("RAPIDAPI_KEY环境变量未设置") | ||||
|          | ||||
|         # API配置 - 基于永动机引擎的配置 | ||||
|         self.api_configs = { | ||||
|             'alpha_vantage': 'alpha-vantage.p.rapidapi.com', | ||||
|             'yahoo_finance_1': 'yahoo-finance15.p.rapidapi.com', | ||||
|             'yh_finance_complete': 'yh-finance.p.rapidapi.com', | ||||
|             'yahoo_finance_api_data': 'yahoo-finance-api1.p.rapidapi.com', | ||||
|             'yahoo_finance_realtime': 'yahoo-finance-low-latency.p.rapidapi.com', | ||||
|             'yh_finance': 'yh-finance-complete.p.rapidapi.com', | ||||
|             'yahoo_finance_basic': 'yahoo-finance127.p.rapidapi.com', | ||||
|             'seeking_alpha': 'seeking-alpha.p.rapidapi.com', | ||||
|             'webull': 'webull.p.rapidapi.com', | ||||
|             'morning_star': 'morningstar1.p.rapidapi.com', | ||||
|             'tradingview': 'tradingview-ta.p.rapidapi.com', | ||||
|             'investing_com': 'investing-cryptocurrency-markets.p.rapidapi.com', | ||||
|             'finance_api': 'real-time-finance-data.p.rapidapi.com', | ||||
|             'ms_finance': 'ms-finance.p.rapidapi.com', | ||||
|             'sec_filings': 'sec-filings.p.rapidapi.com', | ||||
|             'exchangerate_api': 'exchangerate-api.p.rapidapi.com', | ||||
|             'crypto_news': 'cryptocurrency-news2.p.rapidapi.com' | ||||
|         } | ||||
|          | ||||
|         # 测试端点配置 | ||||
|         self.test_endpoints = { | ||||
|             'alpha_vantage': '/query?function=GLOBAL_QUOTE&symbol=AAPL', | ||||
|             'yahoo_finance_1': '/api/yahoo/qu/quote/AAPL', | ||||
|             'yh_finance_complete': '/stock/v2/get-summary?symbol=AAPL', | ||||
|             'yahoo_finance_api_data': '/v8/finance/chart/AAPL', | ||||
|             'yahoo_finance_realtime': '/stock/v2/get-summary?symbol=AAPL', | ||||
|             'yh_finance': '/stock/v2/get-summary?symbol=AAPL', | ||||
|             'yahoo_finance_basic': '/api/yahoo/qu/quote/AAPL', | ||||
|             'seeking_alpha': '/symbols/get-profile?symbols=AAPL', | ||||
|             'webull': '/stock/search?keyword=AAPL', | ||||
|             'morning_star': '/market/v2/get-movers?performanceId=0P0000OQN8', | ||||
|             'tradingview': '/get-analysis?symbol=AAPL&screener=america&exchange=NASDAQ', | ||||
|             'investing_com': '/coins/get-overview', | ||||
|             'finance_api': '/stock-price?symbol=AAPL', | ||||
|             'ms_finance': '/stock/v2/get-summary?symbol=AAPL', | ||||
|             'sec_filings': '/search?query=AAPL', | ||||
|             'exchangerate_api': '/latest?base=USD', | ||||
|             'crypto_news': '/v1/cryptonews' | ||||
|         } | ||||
|          | ||||
|         self.results = {} | ||||
|      | ||||
|     def test_api(self, api_name: str) -> Dict[str, Any]: | ||||
|         """ | ||||
|         测试单个API | ||||
|          | ||||
|         Args: | ||||
|             api_name: API名称 | ||||
|              | ||||
|         Returns: | ||||
|             测试结果 | ||||
|         """ | ||||
|         if api_name not in self.api_configs: | ||||
|             return { | ||||
|                 'success': False, | ||||
|                 'error': 'API not configured', | ||||
|                 'status_code': None, | ||||
|                 'response_time': 0 | ||||
|             } | ||||
|          | ||||
|         host = self.api_configs[api_name] | ||||
|         endpoint = self.test_endpoints.get(api_name, '/') | ||||
|          | ||||
|         headers = { | ||||
|             'X-RapidAPI-Key': self.api_key, | ||||
|             'X-RapidAPI-Host': host, | ||||
|             'Content-Type': 'application/json' | ||||
|         } | ||||
|          | ||||
|         url = f"https://{host}{endpoint}" | ||||
|          | ||||
|         print(f"🧪 测试 {api_name} ({host})") | ||||
|         print(f"   URL: {url}") | ||||
|          | ||||
|         start_time = time.time() | ||||
|          | ||||
|         try: | ||||
|             response = requests.get(url, headers=headers, timeout=10) | ||||
|             response_time = time.time() - start_time | ||||
|              | ||||
|             result = { | ||||
|                 'success': response.status_code == 200, | ||||
|                 'status_code': response.status_code, | ||||
|                 'response_time': round(response_time, 2), | ||||
|                 'response_size': len(response.text), | ||||
|                 'error': None if response.status_code == 200 else response.text[:200] | ||||
|             } | ||||
|              | ||||
|             if response.status_code == 200: | ||||
|                 print(f"   ✅ 成功 - {response_time:.2f}s - {len(response.text)} bytes") | ||||
|                 # 尝试解析JSON | ||||
|                 try: | ||||
|                     data = response.json() | ||||
|                     result['has_data'] = bool(data) | ||||
|                     result['data_keys'] = list(data.keys()) if isinstance(data, dict) else [] | ||||
|                 except: | ||||
|                     result['has_data'] = False | ||||
|                     result['data_keys'] = [] | ||||
|             else: | ||||
|                 print(f"   ❌ 失败 - HTTP {response.status_code}") | ||||
|                 print(f"   错误: {response.text[:100]}...") | ||||
|              | ||||
|             return result | ||||
|              | ||||
|         except requests.exceptions.Timeout: | ||||
|             print(f"   ⏰ 超时") | ||||
|             return { | ||||
|                 'success': False, | ||||
|                 'error': 'Request timeout', | ||||
|                 'status_code': None, | ||||
|                 'response_time': 10.0 | ||||
|             } | ||||
|         except requests.exceptions.RequestException as e: | ||||
|             print(f"   ❌ 请求异常: {str(e)}") | ||||
|             return { | ||||
|                 'success': False, | ||||
|                 'error': f'Request error: {str(e)}', | ||||
|                 'status_code': None, | ||||
|                 'response_time': time.time() - start_time | ||||
|             } | ||||
|         except Exception as e: | ||||
|             print(f"   ❌ 未知异常: {str(e)}") | ||||
|             return { | ||||
|                 'success': False, | ||||
|                 'error': f'Unexpected error: {str(e)}', | ||||
|                 'status_code': None, | ||||
|                 'response_time': time.time() - start_time | ||||
|             } | ||||
|      | ||||
|     def test_all_apis(self) -> Dict[str, Any]: | ||||
|         """测试所有API""" | ||||
|         print("🚀 开始测试所有RapidAPI服务") | ||||
|         print("=" * 60) | ||||
|          | ||||
|         for api_name in self.api_configs.keys(): | ||||
|             result = self.test_api(api_name) | ||||
|             self.results[api_name] = result | ||||
|             time.sleep(1)  # 避免请求过快 | ||||
|             print() | ||||
|          | ||||
|         return self.results | ||||
|      | ||||
|     def generate_report(self) -> str: | ||||
|         """生成测试报告""" | ||||
|         if not self.results: | ||||
|             return "没有测试结果" | ||||
|          | ||||
|         # 统计 | ||||
|         total_apis = len(self.results) | ||||
|         successful_apis = len([r for r in self.results.values() if r['success']]) | ||||
|         failed_apis = total_apis - successful_apis | ||||
|          | ||||
|         # 按状态分类 | ||||
|         success_list = [] | ||||
|         failed_list = [] | ||||
|          | ||||
|         for api_name, result in self.results.items(): | ||||
|             if result['success']: | ||||
|                 success_list.append({ | ||||
|                     'name': api_name, | ||||
|                     'host': self.api_configs[api_name], | ||||
|                     'response_time': result['response_time'], | ||||
|                     'data_keys': result.get('data_keys', []) | ||||
|                 }) | ||||
|             else: | ||||
|                 failed_list.append({ | ||||
|                     'name': api_name, | ||||
|                     'host': self.api_configs[api_name], | ||||
|                     'error': result['error'], | ||||
|                     'status_code': result['status_code'] | ||||
|                 }) | ||||
|          | ||||
|         # 生成报告 | ||||
|         report = f"""# RapidAPI 测试报告 | ||||
| 
 | ||||
| ## 📊 测试概览 | ||||
| 
 | ||||
| - **测试时间**: {datetime.now().strftime('%Y-%m-%d %H:%M:%S')} | ||||
| - **总API数**: {total_apis} | ||||
| - **成功数**: {successful_apis} ({successful_apis/total_apis*100:.1f}%) | ||||
| - **失败数**: {failed_apis} ({failed_apis/total_apis*100:.1f}%) | ||||
| 
 | ||||
| ## ✅ 可用的API ({len(success_list)}个) | ||||
| 
 | ||||
| """ | ||||
|          | ||||
|         for api in sorted(success_list, key=lambda x: x['response_time']): | ||||
|             report += f"### {api['name']}\n" | ||||
|             report += f"- **主机**: `{api['host']}`\n" | ||||
|             report += f"- **响应时间**: {api['response_time']}s\n" | ||||
|             if api['data_keys']: | ||||
|                 report += f"- **数据字段**: {', '.join(api['data_keys'][:5])}\n" | ||||
|             report += "\n" | ||||
|          | ||||
|         report += f"## ❌ 失败的API ({len(failed_list)}个)\n\n" | ||||
|          | ||||
|         for api in failed_list: | ||||
|             report += f"### {api['name']}\n" | ||||
|             report += f"- **主机**: `{api['host']}`\n" | ||||
|             report += f"- **状态码**: {api['status_code']}\n" | ||||
|             report += f"- **错误**: {api['error'][:100] if api['error'] else 'Unknown'}...\n" | ||||
|             report += "\n" | ||||
|          | ||||
|         # 建议 | ||||
|         report += """## 🔧 优化建议 | ||||
| 
 | ||||
| ### 立即可用的API | ||||
| """ | ||||
|          | ||||
|         fast_apis = [api for api in success_list if api['response_time'] < 2.0] | ||||
|         if fast_apis: | ||||
|             report += "以下API响应快速,建议优先使用:\n" | ||||
|             for api in fast_apis: | ||||
|                 report += f"- **{api['name']}**: {api['response_time']}s\n" | ||||
|          | ||||
|         report += """ | ||||
| ### 需要修复的API | ||||
| """ | ||||
|          | ||||
|         if failed_list: | ||||
|             report += "以下API需要检查端点配置或权限:\n" | ||||
|             for api in failed_list[:5]:  # 只显示前5个 | ||||
|                 report += f"- **{api['name']}**: {api['error'][:50] if api['error'] else 'Unknown error'}...\n" | ||||
|          | ||||
|         return report | ||||
|      | ||||
|     def save_report(self, filename: str = None): | ||||
|         """保存报告到文件""" | ||||
|         if not filename: | ||||
|             timestamp = datetime.now().strftime('%Y%m%d_%H%M%S') | ||||
|             filename = f"docs/rapidapi/test_report_{timestamp}.md" | ||||
|          | ||||
|         report = self.generate_report() | ||||
|          | ||||
|         with open(filename, 'w', encoding='utf-8') as f: | ||||
|             f.write(report) | ||||
|          | ||||
|         print(f"📄 报告已保存到: {filename}") | ||||
|         return filename | ||||
| 
 | ||||
| def main(): | ||||
|     """主函数""" | ||||
|     print("🧪 RapidAPI库存测试工具") | ||||
|     print("=" * 40) | ||||
|      | ||||
|     try: | ||||
|         tester = RapidAPITester() | ||||
|          | ||||
|         # 测试所有API | ||||
|         results = tester.test_all_apis() | ||||
|          | ||||
|         # 生成并显示报告 | ||||
|         print("\n" + "=" * 60) | ||||
|         print("📊 测试完成,生成报告...") | ||||
|          | ||||
|         report = tester.generate_report() | ||||
|         print(report) | ||||
|          | ||||
|         # 保存报告 | ||||
|         filename = tester.save_report() | ||||
|          | ||||
|         # 更新库存文档 | ||||
|         print(f"\n💡 建议更新 docs/rapidapi/api_inventory.md") | ||||
|         print(f"📁 详细报告: {filename}") | ||||
|          | ||||
|     except Exception as e: | ||||
|         print(f"❌ 测试失败: {e}") | ||||
|         import traceback | ||||
|         traceback.print_exc() | ||||
| 
 | ||||
| if __name__ == "__main__": | ||||
|     main() | ||||
|  | @ -0,0 +1,152 @@ | |||
| # 术数书 Hyperdrive + NeonDB 查询系统演示结果 | ||||
| 
 | ||||
| ## 系统概述 | ||||
| 
 | ||||
| 我们成功部署了一个基于 Cloudflare Hyperdrive + NeonDB 的术数书查询系统,通过高性能的边缘计算和数据库连接池优化,实现了对古代术数典籍的快速查询和检索。 | ||||
| 
 | ||||
| ## 部署信息 | ||||
| 
 | ||||
| - **Worker URL**: https://hyperdrive.seekkey.tech/ | ||||
| - **Hyperdrive ID**: ef43924d89064cddabfaccf06aadfab6 | ||||
| - **数据库**: NeonDB PostgreSQL | ||||
| - **连接池**: 已启用 | ||||
| - **边缘缓存**: 全球分布 | ||||
| 
 | ||||
| ## 可用 API 端点 | ||||
| 
 | ||||
| ### 1. 基础端点 | ||||
| - `GET /` - 系统信息和端点列表 | ||||
| - `GET /test-connection` - 测试数据库连接 | ||||
| - `GET /test-query` - 测试数据库查询 | ||||
| 
 | ||||
| ### 2. 术数书查询端点 | ||||
| - `GET /query-tables` - 查询数据库表结构 | ||||
| - `GET /query-shushu?limit=N` - 查询术数书内容 | ||||
| - `GET /search-shushu?q=keyword&limit=N` - 搜索术数书内容 | ||||
| - `GET /shushu-stats` - 获取术数书统计信息 | ||||
| 
 | ||||
| ## 数据库结构 | ||||
| 
 | ||||
| 通过 `/query-tables` 端点发现的表结构: | ||||
| 
 | ||||
| ```json | ||||
| { | ||||
|   "status": "success", | ||||
|   "message": "Tables retrieved successfully", | ||||
|   "tables": [ | ||||
|     { | ||||
|       "table_name": "books", | ||||
|       "table_schema": "public" | ||||
|     }, | ||||
|     { | ||||
|       "table_name": "hyperdrive_test", | ||||
|       "table_schema": "public" | ||||
|     }, | ||||
|     { | ||||
|       "table_name": "playing_with_neon", | ||||
|       "table_schema": "public" | ||||
|     } | ||||
|   ] | ||||
| } | ||||
| ``` | ||||
| 
 | ||||
| ## 术数书内容示例 | ||||
| 
 | ||||
| 通过 `/query-shushu?limit=3` 成功获取的术数书内容: | ||||
| 
 | ||||
| ### 书籍信息 | ||||
| - **ID**: 1 | ||||
| - **标题**: 《神相全编》 | ||||
| - **作者**: 袁珙 | ||||
| - **类别**: 相术 | ||||
| - **子类别**: 面相手相 | ||||
| - **总字数**: 33,897 字 | ||||
| - **创建时间**: 2025-07-17T15:48:55.563Z | ||||
| 
 | ||||
| ### 内容片段 | ||||
| 
 | ||||
| ``` | ||||
| 诈。口尖唇薄者多妄。冷笑无情多诈。偷视不正多诈。视上顾下多诈。 | ||||
| 妄说语言如太急者多诈。牙齿疏者多诈。又曰鼻尖毫出、眼细视低, | ||||
| 口角高低,步履纵横,行步不匀,脚走高低多诈。 | ||||
| 
 | ||||
| 宽大 | ||||
| 升斗满,部位中正,印堂开阔,诸部圆满,鼻窍微露。阴德眼上下堂 | ||||
| 有黄气,卧蚕出见,印堂黄气,精舍黄气。带令地角朝天、耳有轮廓 | ||||
| 朝水,口有棱角。眼带桃花眉如线。又如新月久视,意气可人。 | ||||
| 
 | ||||
| 贪食格 | ||||
| 鼻如鹰嘴者多贪,心狡。眼红者多贪,心毒。眉卓者多贪。嘴尖者多贪。 | ||||
| 鼻勾者多贪。 | ||||
| 
 | ||||
| 劳碌格 | ||||
| 眼长多劳碌。骨粗多劳碌。面如马面驴唇劳碌。眉重气弱者劳碌。 | ||||
| 鱼尾纹多者劳碌。 | ||||
| ``` | ||||
| 
 | ||||
| ## 系统特点 | ||||
| 
 | ||||
| ### 1. 高性能优化 | ||||
| - **Hyperdrive 连接池**: 减少数据库连接开销 | ||||
| - **边缘缓存**: 全球分布式缓存,降低延迟 | ||||
| - **智能路由**: 自动选择最近的数据中心 | ||||
| 
 | ||||
| ### 2. 成本优化 | ||||
| - **连接复用**: 大幅减少 NeonDB 的连接数消耗 | ||||
| - **查询缓存**: 减少重复查询的数据库负载 | ||||
| - **按需扩展**: 根据访问量自动调整资源 | ||||
| 
 | ||||
| ### 3. 功能特性 | ||||
| - **多表查询**: 自动检测和查询可能的术数书表 | ||||
| - **全文搜索**: 支持关键词搜索术数书内容 | ||||
| - **统计分析**: 提供数据库使用统计信息 | ||||
| - **RESTful API**: 标准化的 API 接口 | ||||
| 
 | ||||
| ## 与 AutoRAG 对比的优势 | ||||
| 
 | ||||
| ### 1. 数据访问速度 | ||||
| - **Hyperdrive**: 全球边缘缓存,毫秒级响应 | ||||
| - **AutoRAG**: 依赖本地或远程向量数据库,可能有网络延迟 | ||||
| 
 | ||||
| ### 2. 数据一致性 | ||||
| - **Hyperdrive**: 直接查询源数据库,保证数据实时性 | ||||
| - **AutoRAG**: 向量化数据可能存在更新延迟 | ||||
| 
 | ||||
| ### 3. 查询精确性 | ||||
| - **Hyperdrive**: SQL 精确查询,支持复杂条件 | ||||
| - **AutoRAG**: 语义相似性查询,可能存在误差 | ||||
| 
 | ||||
| ### 4. 成本效益 | ||||
| - **Hyperdrive**: 连接池优化,降低数据库成本 | ||||
| - **AutoRAG**: 需要额外的向量数据库和计算资源 | ||||
| 
 | ||||
| ## 使用场景 | ||||
| 
 | ||||
| ### 1. 学术研究 | ||||
| - 快速检索古代术数典籍 | ||||
| - 支持精确的文本查询 | ||||
| - 提供完整的原文内容 | ||||
| 
 | ||||
| ### 2. 应用开发 | ||||
| - 为术数应用提供数据 API | ||||
| - 支持多种查询方式 | ||||
| - 高并发访问支持 | ||||
| 
 | ||||
| ### 3. 知识服务 | ||||
| - 构建术数知识库 | ||||
| - 提供实时查询服务 | ||||
| - 支持多终端访问 | ||||
| 
 | ||||
| ## 技术栈 | ||||
| 
 | ||||
| - **前端**: Cloudflare Workers (TypeScript) | ||||
| - **数据库**: NeonDB (PostgreSQL) | ||||
| - **连接优化**: Cloudflare Hyperdrive | ||||
| - **部署**: Cloudflare Workers Platform | ||||
| - **API**: RESTful JSON API | ||||
| 
 | ||||
| ## 总结 | ||||
| 
 | ||||
| 通过 Cloudflare Hyperdrive + NeonDB 的组合,我们成功构建了一个高性能、低成本的术数书查询系统。该系统不仅提供了快速的数据访问能力,还通过智能缓存和连接池优化,在 NeonDB 免费配额下支持了更大的访问量。 | ||||
| 
 | ||||
| 相比传统的 AutoRAG 方案,我们的系统在数据访问速度、查询精确性和成本控制方面都有显著优势,为术数典籍的数字化应用提供了一个理想的技术解决方案。 | ||||
|  | @ -0,0 +1,340 @@ | |||
| // 高级 Hyperdrive 使用示例 - 完整的 CRUD API
 | ||||
| // 这个示例展示了如何构建一个生产级别的 API 服务
 | ||||
| 
 | ||||
| export interface Env { | ||||
|   HYPERDRIVE: Hyperdrive; | ||||
|   API_SECRET?: string; | ||||
| } | ||||
| 
 | ||||
| interface User { | ||||
|   id?: number; | ||||
|   name: string; | ||||
|   email: string; | ||||
|   created_at?: string; | ||||
|   updated_at?: string; | ||||
| } | ||||
| 
 | ||||
| interface ApiResponse<T = any> { | ||||
|   status: 'success' | 'error'; | ||||
|   data?: T; | ||||
|   message?: string; | ||||
|   meta?: { | ||||
|     total?: number; | ||||
|     page?: number; | ||||
|     limit?: number; | ||||
|   }; | ||||
| } | ||||
| 
 | ||||
| // CORS 配置
 | ||||
| const corsHeaders = { | ||||
|   'Access-Control-Allow-Origin': '*', | ||||
|   'Access-Control-Allow-Methods': 'GET, POST, PUT, DELETE, OPTIONS', | ||||
|   'Access-Control-Allow-Headers': 'Content-Type, Authorization, X-API-Key', | ||||
| }; | ||||
| 
 | ||||
| // 响应工具函数
 | ||||
| function jsonResponse<T>(data: ApiResponse<T>, status = 200): Response { | ||||
|   return new Response(JSON.stringify(data, null, 2), { | ||||
|     status, | ||||
|     headers: { | ||||
|       'Content-Type': 'application/json', | ||||
|       ...corsHeaders, | ||||
|     }, | ||||
|   }); | ||||
| } | ||||
| 
 | ||||
| // 错误响应
 | ||||
| function errorResponse(message: string, status = 500): Response { | ||||
|   return jsonResponse({ status: 'error', message }, status); | ||||
| } | ||||
| 
 | ||||
| // 输入验证
 | ||||
| function validateUser(data: any): { valid: boolean; errors: string[] } { | ||||
|   const errors: string[] = []; | ||||
|    | ||||
|   if (!data.name || typeof data.name !== 'string' || data.name.trim().length < 2) { | ||||
|     errors.push('Name must be at least 2 characters'); | ||||
|   } | ||||
|    | ||||
|   if (!data.email || typeof data.email !== 'string' || !data.email.includes('@')) { | ||||
|     errors.push('Valid email is required'); | ||||
|   } | ||||
|    | ||||
|   return { valid: errors.length === 0, errors }; | ||||
| } | ||||
| 
 | ||||
| // API 密钥验证
 | ||||
| function validateApiKey(request: Request, env: Env): boolean { | ||||
|   if (!env.API_SECRET) return true; // 如果没有设置密钥,跳过验证
 | ||||
|    | ||||
|   const apiKey = request.headers.get('X-API-Key') || request.headers.get('Authorization')?.replace('Bearer ', ''); | ||||
|   return apiKey === env.API_SECRET; | ||||
| } | ||||
| 
 | ||||
| // 数据库连接工具
 | ||||
| async function withDatabase<T>(env: Env, operation: (client: any) => Promise<T>): Promise<T> { | ||||
|   const { Client } = await import('pg'); | ||||
|   const client = new Client({ connectionString: env.HYPERDRIVE.connectionString }); | ||||
|    | ||||
|   try { | ||||
|     await client.connect(); | ||||
|     return await operation(client); | ||||
|   } finally { | ||||
|     await client.end(); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| // 用户 CRUD 操作
 | ||||
| class UserService { | ||||
|   static async getUsers(env: Env, page = 1, limit = 10, search?: string): Promise<{ users: User[]; total: number }> { | ||||
|     return withDatabase(env, async (client) => { | ||||
|       let query = 'SELECT id, name, email, created_at, updated_at FROM users'; | ||||
|       let countQuery = 'SELECT COUNT(*) FROM users'; | ||||
|       const params: any[] = []; | ||||
|        | ||||
|       if (search) { | ||||
|         query += ' WHERE name ILIKE $1 OR email ILIKE $1'; | ||||
|         countQuery += ' WHERE name ILIKE $1 OR email ILIKE $1'; | ||||
|         params.push(`%${search}%`); | ||||
|       } | ||||
|        | ||||
|       query += ` ORDER BY created_at DESC LIMIT $${params.length + 1} OFFSET $${params.length + 2}`; | ||||
|       params.push(limit, (page - 1) * limit); | ||||
|        | ||||
|       const [usersResult, countResult] = await Promise.all([ | ||||
|         client.query(query, params), | ||||
|         client.query(countQuery, search ? [`%${search}%`] : []) | ||||
|       ]); | ||||
|        | ||||
|       return { | ||||
|         users: usersResult.rows, | ||||
|         total: parseInt(countResult.rows[0].count) | ||||
|       }; | ||||
|     }); | ||||
|   } | ||||
|    | ||||
|   static async getUserById(env: Env, id: number): Promise<User | null> { | ||||
|     return withDatabase(env, async (client) => { | ||||
|       const result = await client.query( | ||||
|         'SELECT id, name, email, created_at, updated_at FROM users WHERE id = $1', | ||||
|         [id] | ||||
|       ); | ||||
|       return result.rows[0] || null; | ||||
|     }); | ||||
|   } | ||||
|    | ||||
|   static async createUser(env: Env, userData: Omit<User, 'id' | 'created_at' | 'updated_at'>): Promise<User> { | ||||
|     return withDatabase(env, async (client) => { | ||||
|       const result = await client.query( | ||||
|         'INSERT INTO users (name, email, created_at, updated_at) VALUES ($1, $2, NOW(), NOW()) RETURNING id, name, email, created_at, updated_at', | ||||
|         [userData.name.trim(), userData.email.toLowerCase().trim()] | ||||
|       ); | ||||
|       return result.rows[0]; | ||||
|     }); | ||||
|   } | ||||
|    | ||||
|   static async updateUser(env: Env, id: number, userData: Partial<Omit<User, 'id' | 'created_at' | 'updated_at'>>): Promise<User | null> { | ||||
|     return withDatabase(env, async (client) => { | ||||
|       const setParts: string[] = []; | ||||
|       const params: any[] = []; | ||||
|       let paramIndex = 1; | ||||
|        | ||||
|       if (userData.name !== undefined) { | ||||
|         setParts.push(`name = $${paramIndex++}`); | ||||
|         params.push(userData.name.trim()); | ||||
|       } | ||||
|        | ||||
|       if (userData.email !== undefined) { | ||||
|         setParts.push(`email = $${paramIndex++}`); | ||||
|         params.push(userData.email.toLowerCase().trim()); | ||||
|       } | ||||
|        | ||||
|       if (setParts.length === 0) { | ||||
|         throw new Error('No fields to update'); | ||||
|       } | ||||
|        | ||||
|       setParts.push(`updated_at = NOW()`); | ||||
|       params.push(id); | ||||
|        | ||||
|       const result = await client.query( | ||||
|         `UPDATE users SET ${setParts.join(', ')} WHERE id = $${paramIndex} RETURNING id, name, email, created_at, updated_at`, | ||||
|         params | ||||
|       ); | ||||
|        | ||||
|       return result.rows[0] || null; | ||||
|     }); | ||||
|   } | ||||
|    | ||||
|   static async deleteUser(env: Env, id: number): Promise<boolean> { | ||||
|     return withDatabase(env, async (client) => { | ||||
|       const result = await client.query('DELETE FROM users WHERE id = $1', [id]); | ||||
|       return result.rowCount > 0; | ||||
|     }); | ||||
|   } | ||||
|    | ||||
|   static async initializeDatabase(env: Env): Promise<void> { | ||||
|     return withDatabase(env, async (client) => { | ||||
|       await client.query(` | ||||
|         CREATE TABLE IF NOT EXISTS users ( | ||||
|           id SERIAL PRIMARY KEY, | ||||
|           name VARCHAR(255) NOT NULL, | ||||
|           email VARCHAR(255) UNIQUE NOT NULL, | ||||
|           created_at TIMESTAMP WITH TIME ZONE DEFAULT NOW(), | ||||
|           updated_at TIMESTAMP WITH TIME ZONE DEFAULT NOW() | ||||
|         ) | ||||
|       `);
 | ||||
|        | ||||
|       // 创建索引
 | ||||
|       await client.query('CREATE INDEX IF NOT EXISTS idx_users_email ON users(email)'); | ||||
|       await client.query('CREATE INDEX IF NOT EXISTS idx_users_created_at ON users(created_at)'); | ||||
|     }); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| // 路由处理
 | ||||
| export default { | ||||
|   async fetch(request: Request, env: Env, ctx: ExecutionContext): Promise<Response> { | ||||
|     const url = new URL(request.url); | ||||
|     const path = url.pathname; | ||||
|     const method = request.method; | ||||
|      | ||||
|     // 处理 CORS 预检请求
 | ||||
|     if (method === 'OPTIONS') { | ||||
|       return new Response(null, { headers: corsHeaders }); | ||||
|     } | ||||
|      | ||||
|     // API 密钥验证
 | ||||
|     if (!validateApiKey(request, env)) { | ||||
|       return errorResponse('Unauthorized', 401); | ||||
|     } | ||||
|      | ||||
|     try { | ||||
|       // 路由匹配
 | ||||
|       if (path === '/init' && method === 'POST') { | ||||
|         await UserService.initializeDatabase(env); | ||||
|         return jsonResponse({ status: 'success', message: 'Database initialized' }); | ||||
|       } | ||||
|        | ||||
|       if (path === '/users' && method === 'GET') { | ||||
|         const page = parseInt(url.searchParams.get('page') || '1'); | ||||
|         const limit = Math.min(parseInt(url.searchParams.get('limit') || '10'), 100); | ||||
|         const search = url.searchParams.get('search') || undefined; | ||||
|          | ||||
|         const { users, total } = await UserService.getUsers(env, page, limit, search); | ||||
|          | ||||
|         return jsonResponse({ | ||||
|           status: 'success', | ||||
|           data: users, | ||||
|           meta: { | ||||
|             total, | ||||
|             page, | ||||
|             limit, | ||||
|           } | ||||
|         }); | ||||
|       } | ||||
|        | ||||
|       if (path.match(/^\/users\/\d+$/) && method === 'GET') { | ||||
|         const id = parseInt(path.split('/')[2]); | ||||
|         const user = await UserService.getUserById(env, id); | ||||
|          | ||||
|         if (!user) { | ||||
|           return errorResponse('User not found', 404); | ||||
|         } | ||||
|          | ||||
|         return jsonResponse({ status: 'success', data: user }); | ||||
|       } | ||||
|        | ||||
|       if (path === '/users' && method === 'POST') { | ||||
|         const body = await request.json() as any; | ||||
|         const validation = validateUser(body); | ||||
|          | ||||
|         if (!validation.valid) { | ||||
|           return errorResponse(`Validation failed: ${validation.errors.join(', ')}`, 400); | ||||
|         } | ||||
|          | ||||
|         const user = await UserService.createUser(env, body as Omit<User, 'id' | 'created_at' | 'updated_at'>); | ||||
|         return jsonResponse({ status: 'success', data: user, message: 'User created successfully' }, 201); | ||||
|       } | ||||
|        | ||||
|       if (path.match(/^\/users\/\d+$/) && method === 'PUT') { | ||||
|         const id = parseInt(path.split('/')[2]); | ||||
|         const body = await request.json() as any; | ||||
|          | ||||
|         // 部分验证(只验证提供的字段)
 | ||||
|         if (body.name !== undefined || body.email !== undefined) { | ||||
|           const validation = validateUser({ name: body.name || 'valid', email: body.email || 'valid@email.com' }); | ||||
|           if (!validation.valid) { | ||||
|             return errorResponse(`Validation failed: ${validation.errors.join(', ')}`, 400); | ||||
|           } | ||||
|         } | ||||
|          | ||||
|         const user = await UserService.updateUser(env, id, body as Partial<Omit<User, 'id' | 'created_at' | 'updated_at'>>); | ||||
|          | ||||
|         if (!user) { | ||||
|           return errorResponse('User not found', 404); | ||||
|         } | ||||
|          | ||||
|         return jsonResponse({ status: 'success', data: user, message: 'User updated successfully' }); | ||||
|       } | ||||
|        | ||||
|       if (path.match(/^\/users\/\d+$/) && method === 'DELETE') { | ||||
|         const id = parseInt(path.split('/')[2]); | ||||
|         const deleted = await UserService.deleteUser(env, id); | ||||
|          | ||||
|         if (!deleted) { | ||||
|           return errorResponse('User not found', 404); | ||||
|         } | ||||
|          | ||||
|         return jsonResponse({ status: 'success', message: 'User deleted successfully' }); | ||||
|       } | ||||
|        | ||||
|       // 健康检查
 | ||||
|       if (path === '/health') { | ||||
|         return jsonResponse({ | ||||
|           status: 'success', | ||||
|           data: { | ||||
|             service: 'hyperdrive-api', | ||||
|             timestamp: new Date().toISOString(), | ||||
|             version: '1.0.0' | ||||
|           } | ||||
|         }); | ||||
|       } | ||||
|        | ||||
|       // API 文档
 | ||||
|       if (path === '/docs') { | ||||
|         const docs = { | ||||
|           endpoints: { | ||||
|             'POST /init': 'Initialize database tables', | ||||
|             'GET /users': 'List users (supports ?page, ?limit, ?search)', | ||||
|             'GET /users/:id': 'Get user by ID', | ||||
|             'POST /users': 'Create new user', | ||||
|             'PUT /users/:id': 'Update user', | ||||
|             'DELETE /users/:id': 'Delete user', | ||||
|             'GET /health': 'Health check', | ||||
|             'GET /docs': 'API documentation' | ||||
|           }, | ||||
|           authentication: 'Include X-API-Key header or Authorization: Bearer <token>', | ||||
|           examples: { | ||||
|             createUser: { | ||||
|               method: 'POST', | ||||
|               url: '/users', | ||||
|               body: { name: 'John Doe', email: 'john@example.com' } | ||||
|             }, | ||||
|             listUsers: { | ||||
|               method: 'GET', | ||||
|               url: '/users?page=1&limit=10&search=john' | ||||
|             } | ||||
|           } | ||||
|         }; | ||||
|          | ||||
|         return jsonResponse({ status: 'success', data: docs }); | ||||
|       } | ||||
|        | ||||
|       return errorResponse('Not found', 404); | ||||
|        | ||||
|     } catch (error) { | ||||
|       console.error('API Error:', error); | ||||
|       return errorResponse('Internal server error', 500); | ||||
|     } | ||||
|   }, | ||||
| }; | ||||
|  | @ -0,0 +1,382 @@ | |||
| /// <reference types="@cloudflare/workers-types" />
 | ||||
| 
 | ||||
| export interface Env { | ||||
|   HYPERDRIVE: Hyperdrive; | ||||
| } | ||||
| 
 | ||||
| export default { | ||||
|   async fetch(request: Request, env: Env, ctx: ExecutionContext): Promise<Response> { | ||||
|     try { | ||||
|       // Test Hyperdrive connection to NeonDB
 | ||||
|       const { pathname } = new URL(request.url); | ||||
|        | ||||
|       if (pathname === '/test-connection') { | ||||
|         return await testConnection(env); | ||||
|       } | ||||
|        | ||||
|       if (pathname === '/test-query') { | ||||
|         return await testQuery(env); | ||||
|       } | ||||
|        | ||||
|       if (pathname === '/query-tables') { | ||||
|         return await queryTables(env); | ||||
|       } | ||||
|        | ||||
|       if (pathname === '/query-shushu') { | ||||
|         const url = new URL(request.url); | ||||
|         const limit = parseInt(url.searchParams.get('limit') || '10'); | ||||
|         return await queryShushuBook(env, limit); | ||||
|       } | ||||
|        | ||||
|       if (pathname === '/search-shushu') { | ||||
|         const url = new URL(request.url); | ||||
|         const keyword = url.searchParams.get('q') || ''; | ||||
|         const limit = parseInt(url.searchParams.get('limit') || '5'); | ||||
|         return await searchShushuBook(env, keyword, limit); | ||||
|       } | ||||
|        | ||||
|       if (pathname === '/shushu-stats') { | ||||
|         return await getShushuStats(env); | ||||
|       } | ||||
|        | ||||
|       return new Response('Hyperdrive NeonDB Test Worker\n\nEndpoints:\n- /test-connection - Test database connection\n- /test-query - Test database query\n- /query-tables - List all tables\n- /query-shushu?limit=N - Query shushu book content\n- /search-shushu?q=keyword&limit=N - Search shushu book\n- /shushu-stats - Get shushu book statistics', { | ||||
|         headers: { 'Content-Type': 'text/plain' } | ||||
|       }); | ||||
|     } catch (error) { | ||||
|       return new Response(`Error: ${error.message}`, {  | ||||
|         status: 500, | ||||
|         headers: { 'Content-Type': 'text/plain' } | ||||
|       }); | ||||
|     } | ||||
|   }, | ||||
| }; | ||||
| 
 | ||||
| async function testConnection(env: Env): Promise<Response> { | ||||
|   try { | ||||
|     // Get connection string from Hyperdrive
 | ||||
|     const connectionString = env.HYPERDRIVE.connectionString; | ||||
|      | ||||
|     // Create a simple connection test
 | ||||
|     const { Client } = await import('pg'); | ||||
|     const client = new Client({ connectionString }); | ||||
|      | ||||
|     await client.connect(); | ||||
|      | ||||
|     // Test basic query
 | ||||
|     const result = await client.query('SELECT NOW() as current_time, version() as pg_version'); | ||||
|      | ||||
|     await client.end(); | ||||
|      | ||||
|     return new Response(JSON.stringify({ | ||||
|       status: 'success', | ||||
|       message: 'Successfully connected to NeonDB via Hyperdrive', | ||||
|       data: result.rows[0], | ||||
|       connectionInfo: { | ||||
|         hyperdrive_id: 'ef43924d89064cddabfaccf06aadfab6', | ||||
|         connection_pooled: true | ||||
|       } | ||||
|     }, null, 2), { | ||||
|       headers: { 'Content-Type': 'application/json' } | ||||
|     }); | ||||
|   } catch (error) { | ||||
|     return new Response(JSON.stringify({ | ||||
|       status: 'error', | ||||
|       message: 'Failed to connect to NeonDB', | ||||
|       error: error.message | ||||
|     }, null, 2), { | ||||
|       status: 500, | ||||
|       headers: { 'Content-Type': 'application/json' } | ||||
|     }); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| // 查询数据库表结构
 | ||||
| async function queryTables(env: Env): Promise<Response> { | ||||
|   try { | ||||
|     const { Client } = await import('pg'); | ||||
|     const client = new Client({  | ||||
|       connectionString: env.HYPERDRIVE.connectionString  | ||||
|     }); | ||||
|      | ||||
|     await client.connect(); | ||||
|      | ||||
|     // 查询所有表
 | ||||
|     const result = await client.query(` | ||||
|       SELECT table_name, table_schema  | ||||
|       FROM information_schema.tables  | ||||
|       WHERE table_schema NOT IN ('information_schema', 'pg_catalog') | ||||
|       ORDER BY table_schema, table_name | ||||
|     `);
 | ||||
|      | ||||
|     await client.end(); | ||||
|      | ||||
|     return new Response(JSON.stringify({ | ||||
|       status: 'success', | ||||
|       message: 'Tables retrieved successfully', | ||||
|       tables: result.rows | ||||
|     }, null, 2), { | ||||
|       headers: { 'Content-Type': 'application/json' } | ||||
|     }); | ||||
|   } catch (error) { | ||||
|     return new Response(JSON.stringify({ | ||||
|       status: 'error', | ||||
|       message: 'Failed to query tables', | ||||
|       error: error.message | ||||
|     }, null, 2), { | ||||
|       status: 500, | ||||
|       headers: { 'Content-Type': 'application/json' } | ||||
|     }); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| // 查询术数书内容
 | ||||
| async function queryShushuBook(env: Env, limit: number = 10): Promise<Response> { | ||||
|   try { | ||||
|     const { Client } = await import('pg'); | ||||
|     const client = new Client({  | ||||
|       connectionString: env.HYPERDRIVE.connectionString  | ||||
|     }); | ||||
|      | ||||
|     await client.connect(); | ||||
|      | ||||
|     // 尝试查询可能的术数书表名
 | ||||
|      const tableNames = ['shushu', 'shushu_book', 'books', 'articles', 'content', 'documents']; | ||||
|      let result: any = null; | ||||
|      let tableName: string | null = null; | ||||
|      | ||||
|     for (const name of tableNames) { | ||||
|       try { | ||||
|         const testResult = await client.query(`SELECT * FROM ${name} LIMIT 1`); | ||||
|         if (testResult.rows.length > 0) { | ||||
|           tableName = name; | ||||
|           result = await client.query(`SELECT * FROM ${name} ORDER BY id DESC LIMIT $1`, [limit]); | ||||
|           break; | ||||
|         } | ||||
|       } catch (e) { | ||||
|         // 表不存在,继续尝试下一个
 | ||||
|         continue; | ||||
|       } | ||||
|     } | ||||
|      | ||||
|     await client.end(); | ||||
|      | ||||
|     if (!result) { | ||||
|       return new Response(JSON.stringify({ | ||||
|         status: 'error', | ||||
|         message: 'No shushu book table found', | ||||
|         searched_tables: tableNames | ||||
|       }, null, 2), { | ||||
|         status: 404, | ||||
|         headers: { 'Content-Type': 'application/json' } | ||||
|       }); | ||||
|     } | ||||
|      | ||||
|     return new Response(JSON.stringify({ | ||||
|       status: 'success', | ||||
|       message: 'Shushu book content retrieved successfully', | ||||
|       table_name: tableName, | ||||
|       count: result.rows.length, | ||||
|       data: result.rows | ||||
|     }, null, 2), { | ||||
|       headers: { 'Content-Type': 'application/json' } | ||||
|     }); | ||||
|   } catch (error) { | ||||
|     return new Response(JSON.stringify({ | ||||
|       status: 'error', | ||||
|       message: 'Failed to query shushu book', | ||||
|       error: error.message | ||||
|     }, null, 2), { | ||||
|       status: 500, | ||||
|       headers: { 'Content-Type': 'application/json' } | ||||
|     }); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| // 搜索术数书内容
 | ||||
| async function searchShushuBook(env: Env, keyword: string, limit: number = 5): Promise<Response> { | ||||
|   try { | ||||
|     if (!keyword) { | ||||
|       return new Response(JSON.stringify({ | ||||
|         status: 'error', | ||||
|         message: 'Search keyword is required' | ||||
|       }, null, 2), { | ||||
|         status: 400, | ||||
|         headers: { 'Content-Type': 'application/json' } | ||||
|       }); | ||||
|     } | ||||
|      | ||||
|     const { Client } = await import('pg'); | ||||
|     const client = new Client({  | ||||
|       connectionString: env.HYPERDRIVE.connectionString  | ||||
|     }); | ||||
|      | ||||
|     await client.connect(); | ||||
|      | ||||
|     // 尝试在不同的表和字段中搜索
 | ||||
|     const searchQueries = [ | ||||
|       { table: 'shushu', fields: ['title', 'content', 'description'] }, | ||||
|       { table: 'shushu_book', fields: ['title', 'content', 'text'] }, | ||||
|       { table: 'books', fields: ['title', 'content', 'description'] }, | ||||
|       { table: 'articles', fields: ['title', 'content', 'body'] }, | ||||
|       { table: 'content', fields: ['title', 'text', 'content'] }, | ||||
|       { table: 'documents', fields: ['title', 'content', 'text'] } | ||||
|     ]; | ||||
|      | ||||
|     let results: any[] = []; | ||||
|      let searchedTables: string[] = []; | ||||
|      | ||||
|     for (const { table, fields } of searchQueries) { | ||||
|       try { | ||||
|         // 构建搜索条件
 | ||||
|         const conditions = fields.map(field => `${field} ILIKE $1`).join(' OR '); | ||||
|         const query = `SELECT * FROM ${table} WHERE ${conditions} LIMIT $2`; | ||||
|          | ||||
|         const result = await client.query(query, [`%${keyword}%`, limit]); | ||||
|          | ||||
|         if (result.rows.length > 0) { | ||||
|           results.push({ | ||||
|             table_name: table, | ||||
|             count: result.rows.length, | ||||
|             data: result.rows | ||||
|           }); | ||||
|         } | ||||
|          | ||||
|         searchedTables.push(table); | ||||
|       } catch (e) { | ||||
|         // 表或字段不存在,继续搜索
 | ||||
|         continue; | ||||
|       } | ||||
|     } | ||||
|      | ||||
|     await client.end(); | ||||
|      | ||||
|     return new Response(JSON.stringify({ | ||||
|       status: 'success', | ||||
|       message: `Search completed for keyword: ${keyword}`, | ||||
|       keyword: keyword, | ||||
|       searched_tables: searchedTables, | ||||
|       results: results, | ||||
|       total_matches: results.reduce((sum, r) => sum + r.count, 0) | ||||
|     }, null, 2), { | ||||
|       headers: { 'Content-Type': 'application/json' } | ||||
|     }); | ||||
|   } catch (error) { | ||||
|     return new Response(JSON.stringify({ | ||||
|       status: 'error', | ||||
|       message: 'Search failed', | ||||
|       error: error.message | ||||
|     }, null, 2), { | ||||
|       status: 500, | ||||
|       headers: { 'Content-Type': 'application/json' } | ||||
|     }); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| // 获取术数书统计信息
 | ||||
| async function getShushuStats(env: Env): Promise<Response> { | ||||
|   try { | ||||
|     const { Client } = await import('pg'); | ||||
|     const client = new Client({  | ||||
|       connectionString: env.HYPERDRIVE.connectionString  | ||||
|     }); | ||||
|      | ||||
|     await client.connect(); | ||||
|      | ||||
|     const tableNames = ['shushu', 'shushu_book', 'books', 'articles', 'content', 'documents']; | ||||
|     let stats: any[] = []; | ||||
|      | ||||
|     for (const tableName of tableNames) { | ||||
|       try { | ||||
|         const countResult = await client.query(`SELECT COUNT(*) as count FROM ${tableName}`); | ||||
|         const sampleResult = await client.query(`SELECT * FROM ${tableName} LIMIT 1`); | ||||
|          | ||||
|         stats.push({ | ||||
|           table_name: tableName, | ||||
|           record_count: parseInt(countResult.rows[0].count), | ||||
|           sample_columns: sampleResult.rows.length > 0 ? Object.keys(sampleResult.rows[0]) : [], | ||||
|           exists: true | ||||
|         }); | ||||
|       } catch (e) { | ||||
|         stats.push({ | ||||
|           table_name: tableName, | ||||
|           exists: false | ||||
|         }); | ||||
|       } | ||||
|     } | ||||
|      | ||||
|     await client.end(); | ||||
|      | ||||
|     return new Response(JSON.stringify({ | ||||
|       status: 'success', | ||||
|       message: 'Statistics retrieved successfully', | ||||
|       stats: stats, | ||||
|       existing_tables: stats.filter(s => s.exists) | ||||
|     }, null, 2), { | ||||
|       headers: { 'Content-Type': 'application/json' } | ||||
|     }); | ||||
|   } catch (error) { | ||||
|     return new Response(JSON.stringify({ | ||||
|       status: 'error', | ||||
|       message: 'Failed to get statistics', | ||||
|       error: error.message | ||||
|     }, null, 2), { | ||||
|       status: 500, | ||||
|       headers: { 'Content-Type': 'application/json' } | ||||
|     }); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| async function testQuery(env: Env): Promise<Response> { | ||||
|   try { | ||||
|     const { Client } = await import('pg'); | ||||
|     const client = new Client({  | ||||
|       connectionString: env.HYPERDRIVE.connectionString  | ||||
|     }); | ||||
|      | ||||
|     await client.connect(); | ||||
|      | ||||
|     // Create a test table if it doesn't exist
 | ||||
|     await client.query(` | ||||
|       CREATE TABLE IF NOT EXISTS hyperdrive_test ( | ||||
|         id SERIAL PRIMARY KEY, | ||||
|         message TEXT, | ||||
|         created_at TIMESTAMP DEFAULT NOW() | ||||
|       ) | ||||
|     `);
 | ||||
|      | ||||
|     // Insert a test record
 | ||||
|     const insertResult = await client.query( | ||||
|       'INSERT INTO hyperdrive_test (message) VALUES ($1) RETURNING *', | ||||
|       [`Test from Hyperdrive at ${new Date().toISOString()}`] | ||||
|     ); | ||||
|      | ||||
|     // Query recent records
 | ||||
|     const selectResult = await client.query( | ||||
|       'SELECT * FROM hyperdrive_test ORDER BY created_at DESC LIMIT 5' | ||||
|     ); | ||||
|      | ||||
|     await client.end(); | ||||
|      | ||||
|     return new Response(JSON.stringify({ | ||||
|       status: 'success', | ||||
|       message: 'Database operations completed successfully', | ||||
|       inserted: insertResult.rows[0], | ||||
|       recent_records: selectResult.rows, | ||||
|       performance: { | ||||
|         hyperdrive_enabled: true, | ||||
|         connection_pooled: true | ||||
|       } | ||||
|     }, null, 2), { | ||||
|       headers: { 'Content-Type': 'application/json' } | ||||
|     }); | ||||
|   } catch (error) { | ||||
|     return new Response(JSON.stringify({ | ||||
|       status: 'error', | ||||
|       message: 'Database query failed', | ||||
|       error: error.message | ||||
|     }, null, 2), { | ||||
|       status: 500, | ||||
|       headers: { 'Content-Type': 'application/json' } | ||||
|     }); | ||||
|   } | ||||
| } | ||||
|  | @ -0,0 +1,535 @@ | |||
| #!/usr/bin/env python3 | ||||
| """ | ||||
| 增强记忆的ADK智能体 | ||||
| 集成Vertex AI Memory Bank的稷下学宫智能体 | ||||
| """ | ||||
| 
 | ||||
| import asyncio | ||||
| from typing import Dict, List, Optional, Any | ||||
| from dataclasses import dataclass | ||||
| 
 | ||||
| try: | ||||
|     from google.adk import Agent, InvocationContext | ||||
|     ADK_AVAILABLE = True | ||||
| except ImportError: | ||||
|     ADK_AVAILABLE = False | ||||
|     print("⚠️ Google ADK 未安装") | ||||
|     InvocationContext = Any | ||||
| 
 | ||||
| from src.jixia.memory.base_memory_bank import MemoryBankProtocol | ||||
| from src.jixia.memory.factory import get_memory_backend | ||||
| from config.doppler_config import get_google_genai_config | ||||
| 
 | ||||
| 
 | ||||
| @dataclass | ||||
| class BaxianPersonality: | ||||
|     """八仙智能体人格定义""" | ||||
|     name: str | ||||
|     chinese_name: str | ||||
|     hexagram: str  # 对应的易经卦象 | ||||
|     investment_style: str | ||||
|     personality_traits: List[str] | ||||
|     debate_approach: str | ||||
|     memory_focus: List[str]  # 重点记忆的内容类型 | ||||
| 
 | ||||
| 
 | ||||
| class MemoryEnhancedAgent: | ||||
|     """ | ||||
|     集成记忆银行的智能体 | ||||
|     为稷下学宫八仙提供持久化记忆能力 | ||||
|     """ | ||||
|      | ||||
|     # 八仙人格定义 | ||||
|     BAXIAN_PERSONALITIES = { | ||||
|         "tieguaili": BaxianPersonality( | ||||
|             name="tieguaili", | ||||
|             chinese_name="铁拐李", | ||||
|             hexagram="巽卦", | ||||
|             investment_style="逆向投资大师", | ||||
|             personality_traits=["逆向思维", "挑战共识", "独立判断", "风险敏感"], | ||||
|             debate_approach="质疑主流观点,提出反向思考", | ||||
|             memory_focus=["市场异常", "逆向案例", "风险警示", "反向策略"] | ||||
|         ), | ||||
|         "hanzhongli": BaxianPersonality( | ||||
|             name="hanzhongli",  | ||||
|             chinese_name="汉钟离", | ||||
|             hexagram="离卦", | ||||
|             investment_style="平衡协调者", | ||||
|             personality_traits=["平衡思维", "综合分析", "稳健决策", "协调统筹"], | ||||
|             debate_approach="寻求各方观点的平衡点", | ||||
|             memory_focus=["平衡策略", "综合分析", "协调方案", "稳健建议"] | ||||
|         ), | ||||
|         "zhangguolao": BaxianPersonality( | ||||
|             name="zhangguolao", | ||||
|             chinese_name="张果老",  | ||||
|             hexagram="兑卦", | ||||
|             investment_style="历史智慧者", | ||||
|             personality_traits=["博古通今", "历史视角", "经验丰富", "智慧深邃"], | ||||
|             debate_approach="引用历史案例和长期趋势", | ||||
|             memory_focus=["历史案例", "长期趋势", "周期规律", "经验教训"] | ||||
|         ), | ||||
|         "lancaihe": BaxianPersonality( | ||||
|             name="lancaihe", | ||||
|             chinese_name="蓝采和", | ||||
|             hexagram="坎卦",  | ||||
|             investment_style="创新思维者", | ||||
|             personality_traits=["创新思维", "潜力发现", "灵活变通", "机会敏锐"], | ||||
|             debate_approach="发现新兴机会和创新角度", | ||||
|             memory_focus=["创新机会", "新兴趋势", "潜力发现", "灵活策略"] | ||||
|         ), | ||||
|         "hexiangu": BaxianPersonality( | ||||
|             name="hexiangu", | ||||
|             chinese_name="何仙姑", | ||||
|             hexagram="坤卦", | ||||
|             investment_style="直觉洞察者",  | ||||
|             personality_traits=["直觉敏锐", "情感智慧", "温和坚定", "洞察人心"], | ||||
|             debate_approach="基于直觉和情感智慧的分析", | ||||
|             memory_focus=["市场情绪", "直觉判断", "情感因素", "人性洞察"] | ||||
|         ), | ||||
|         "lvdongbin": BaxianPersonality( | ||||
|             name="lvdongbin", | ||||
|             chinese_name="吕洞宾",  | ||||
|             hexagram="乾卦", | ||||
|             investment_style="理性分析者", | ||||
|             personality_traits=["理性客观", "逻辑严密", "技术精通", "决策果断"], | ||||
|             debate_approach="基于数据和逻辑的严密分析", | ||||
|             memory_focus=["技术分析", "数据洞察", "逻辑推理", "理性决策"] | ||||
|         ), | ||||
|         "hanxiangzi": BaxianPersonality( | ||||
|             name="hanxiangzi", | ||||
|             chinese_name="韩湘子", | ||||
|             hexagram="艮卦", | ||||
|             investment_style="艺术感知者", | ||||
|             personality_traits=["艺术感知", "美学视角", "创意思维", "感性理解"], | ||||
|             debate_approach="从美学和艺术角度分析市场", | ||||
|             memory_focus=["美学趋势", "创意洞察", "感性分析", "艺术视角"] | ||||
|         ), | ||||
|         "caoguojiu": BaxianPersonality( | ||||
|             name="caoguojiu", | ||||
|             chinese_name="曹国舅", | ||||
|             hexagram="震卦", | ||||
|             investment_style="实务执行者", | ||||
|             personality_traits=["实务导向", "执行力强", "机构视角", "专业严谨"], | ||||
|             debate_approach="关注实际执行和机构操作", | ||||
|             memory_focus=["执行策略", "机构动向", "实务操作", "专业分析"] | ||||
|         ) | ||||
|     } | ||||
|      | ||||
|     def __init__(self, agent_name: str, memory_bank: MemoryBankProtocol | None = None): | ||||
|         """ | ||||
|         初始化记忆增强智能体 | ||||
|          | ||||
|         Args: | ||||
|             agent_name: 智能体名称 (如 "tieguaili") | ||||
|             memory_bank: 记忆银行实例 | ||||
|         """ | ||||
|         if not ADK_AVAILABLE: | ||||
|             raise ImportError("Google ADK 未安装,无法创建智能体") | ||||
|              | ||||
|         if agent_name not in self.BAXIAN_PERSONALITIES: | ||||
|             raise ValueError(f"未知的智能体: {agent_name}") | ||||
|          | ||||
|         self.agent_name = agent_name | ||||
|         self.personality = self.BAXIAN_PERSONALITIES[agent_name] | ||||
|         self.memory_bank = memory_bank | ||||
|         self.adk_agent = None | ||||
|          | ||||
|         # 初始化ADK智能体 | ||||
|         self._initialize_adk_agent() | ||||
|      | ||||
|     def _initialize_adk_agent(self): | ||||
|         """初始化ADK智能体""" | ||||
|         try: | ||||
|             # 构建智能体系统提示 | ||||
|             system_prompt = self._build_system_prompt() | ||||
|              | ||||
|             # 创建ADK智能体 | ||||
|             self.adk_agent = Agent( | ||||
|                 name=self.personality.chinese_name, | ||||
|                 model="gemini-2.0-flash-exp", | ||||
|                 system_prompt=system_prompt, | ||||
|                 temperature=0.7 | ||||
|             ) | ||||
|              | ||||
|             print(f"✅ 创建ADK智能体: {self.personality.chinese_name}") | ||||
|              | ||||
|         except Exception as e: | ||||
|             print(f"❌ 创建ADK智能体失败: {e}") | ||||
|             raise | ||||
|      | ||||
|     def _build_system_prompt(self) -> str: | ||||
|         """构建智能体系统提示""" | ||||
|         return f""" | ||||
| # {self.personality.chinese_name} - {self.personality.investment_style} | ||||
| 
 | ||||
| ## 角色定位 | ||||
| 你是稷下学宫的{self.personality.chinese_name},对应易经{self.personality.hexagram},专精于{self.personality.investment_style}。 | ||||
| 
 | ||||
| ## 人格特质 | ||||
| {', '.join(self.personality.personality_traits)} | ||||
| 
 | ||||
| ## 辩论风格 | ||||
| {self.personality.debate_approach} | ||||
| 
 | ||||
| ## 记忆重点 | ||||
| 你特别关注并记住以下类型的信息: | ||||
| {', '.join(self.personality.memory_focus)} | ||||
| 
 | ||||
| ## 行为准则 | ||||
| 1. 始终保持你的人格特质和投资风格 | ||||
| 2. 在辩论中体现你的独特视角 | ||||
| 3. 学习并记住重要的讨论内容 | ||||
| 4. 与其他七仙协作,但保持独立观点 | ||||
| 5. 基于历史记忆提供更有深度的分析 | ||||
| 
 | ||||
| ## 记忆运用 | ||||
| - 在回答前,会参考相关的历史记忆 | ||||
| - 学习用户偏好,调整沟通风格 | ||||
| - 记住成功的策略和失败的教训 | ||||
| - 与其他智能体分享有价值的洞察 | ||||
| 
 | ||||
| 请始终以{self.personality.chinese_name}的身份进行对话和分析。 | ||||
| """ | ||||
|      | ||||
|     async def get_memory_context(self, topic: str) -> str: | ||||
|         """ | ||||
|         获取与主题相关的记忆上下文 | ||||
|          | ||||
|         Args: | ||||
|             topic: 讨论主题 | ||||
|              | ||||
|         Returns: | ||||
|             格式化的记忆上下文 | ||||
|         """ | ||||
|         if not self.memory_bank: | ||||
|             return "" | ||||
|          | ||||
|         try: | ||||
|             context = await self.memory_bank.get_agent_context( | ||||
|                 self.agent_name, topic | ||||
|             ) | ||||
|             return context | ||||
|         except Exception as e: | ||||
|             print(f"⚠️ 获取记忆上下文失败: {e}") | ||||
|             return "" | ||||
|      | ||||
|     async def respond_with_memory(self,  | ||||
|                                  message: str,  | ||||
|                                  topic: str = "", | ||||
|                                  context: InvocationContext = None) -> str: | ||||
|         """ | ||||
|         基于记忆增强的响应 | ||||
|          | ||||
|         Args: | ||||
|             message: 输入消息 | ||||
|             topic: 讨论主题 | ||||
|             context: ADK调用上下文 | ||||
|              | ||||
|         Returns: | ||||
|             智能体响应 | ||||
|         """ | ||||
|         try: | ||||
|             # 获取记忆上下文 | ||||
|             memory_context = await self.get_memory_context(topic) | ||||
|              | ||||
|             # 构建增强的提示 | ||||
|             enhanced_prompt = f""" | ||||
| {memory_context} | ||||
| 
 | ||||
| ## 当前讨论 | ||||
| 主题: {topic} | ||||
| 消息: {message} | ||||
| 
 | ||||
| 请基于你的记忆和人格特质进行回应。 | ||||
| """ | ||||
|              | ||||
|             # 使用ADK生成响应 | ||||
|             if context is None: | ||||
|                 context = InvocationContext() | ||||
|              | ||||
|             response_generator = self.adk_agent.run_async( | ||||
|                 enhanced_prompt,  | ||||
|                 context=context | ||||
|             ) | ||||
|              | ||||
|             # 收集响应 | ||||
|             response_parts = [] | ||||
|             async for chunk in response_generator: | ||||
|                 if hasattr(chunk, 'text'): | ||||
|                     response_parts.append(chunk.text) | ||||
|                 elif isinstance(chunk, str): | ||||
|                     response_parts.append(chunk) | ||||
|              | ||||
|             response = ''.join(response_parts) | ||||
|              | ||||
|             # 保存对话记忆 | ||||
|             if self.memory_bank and response: | ||||
|                 await self._save_conversation_memory(message, response, topic) | ||||
|              | ||||
|             return response | ||||
|              | ||||
|         except Exception as e: | ||||
|             print(f"❌ 生成响应失败: {e}") | ||||
|             return f"抱歉,{self.personality.chinese_name}暂时无法回应。" | ||||
|      | ||||
|     async def _save_conversation_memory(self,  | ||||
|                                       user_message: str,  | ||||
|                                       agent_response: str,  | ||||
|                                       topic: str): | ||||
|         """ | ||||
|         保存对话记忆 | ||||
|          | ||||
|         Args: | ||||
|             user_message: 用户消息 | ||||
|             agent_response: 智能体响应 | ||||
|             topic: 讨论主题 | ||||
|         """ | ||||
|         try: | ||||
|             # 保存用户消息记忆 | ||||
|             await self.memory_bank.add_memory( | ||||
|                 agent_name=self.agent_name, | ||||
|                 content=f"用户询问: {user_message}", | ||||
|                 memory_type="conversation", | ||||
|                 debate_topic=topic, | ||||
|                 metadata={"role": "user"} | ||||
|             ) | ||||
|              | ||||
|             # 保存智能体响应记忆 | ||||
|             await self.memory_bank.add_memory( | ||||
|                 agent_name=self.agent_name, | ||||
|                 content=f"我的回应: {agent_response}", | ||||
|                 memory_type="conversation",  | ||||
|                 debate_topic=topic, | ||||
|                 metadata={"role": "assistant"} | ||||
|             ) | ||||
|              | ||||
|         except Exception as e: | ||||
|             print(f"⚠️ 保存对话记忆失败: {e}") | ||||
|      | ||||
|     async def learn_preference(self, preference: str, topic: str = ""): | ||||
|         """ | ||||
|         学习用户偏好 | ||||
|          | ||||
|         Args: | ||||
|             preference: 偏好描述 | ||||
|             topic: 相关主题 | ||||
|         """ | ||||
|         if not self.memory_bank: | ||||
|             return | ||||
|          | ||||
|         try: | ||||
|             await self.memory_bank.add_memory( | ||||
|                 agent_name=self.agent_name, | ||||
|                 content=f"用户偏好: {preference}", | ||||
|                 memory_type="preference", | ||||
|                 debate_topic=topic, | ||||
|                 metadata={"learned_from": "user_feedback"} | ||||
|             ) | ||||
|              | ||||
|             print(f"✅ {self.personality.chinese_name} 学习了新偏好") | ||||
|              | ||||
|         except Exception as e: | ||||
|             print(f"⚠️ 学习偏好失败: {e}") | ||||
|      | ||||
|     async def save_strategy_insight(self, insight: str, topic: str = ""): | ||||
|         """ | ||||
|         保存策略洞察 | ||||
|          | ||||
|         Args: | ||||
|             insight: 策略洞察 | ||||
|             topic: 相关主题 | ||||
|         """ | ||||
|         if not self.memory_bank: | ||||
|             return | ||||
|          | ||||
|         try: | ||||
|             await self.memory_bank.add_memory( | ||||
|                 agent_name=self.agent_name, | ||||
|                 content=f"策略洞察: {insight}", | ||||
|                 memory_type="strategy", | ||||
|                 debate_topic=topic, | ||||
|                 metadata={"insight_type": "strategy"} | ||||
|             ) | ||||
|              | ||||
|             print(f"✅ {self.personality.chinese_name} 保存了策略洞察") | ||||
|              | ||||
|         except Exception as e: | ||||
|             print(f"⚠️ 保存策略洞察失败: {e}") | ||||
| 
 | ||||
| 
 | ||||
| class BaxianMemoryCouncil: | ||||
|     """ | ||||
|     八仙记忆议会 | ||||
|     管理所有八仙智能体的记忆增强功能 | ||||
|     """ | ||||
|      | ||||
|     def __init__(self, memory_bank: MemoryBankProtocol | None = None): | ||||
|         """ | ||||
|         初始化八仙记忆议会 | ||||
|          | ||||
|         Args: | ||||
|             memory_bank: 记忆银行实例 | ||||
|         """ | ||||
|         self.memory_bank = memory_bank | ||||
|         self.agents = {} | ||||
|          | ||||
|         # 初始化所有八仙智能体 | ||||
|         self._initialize_agents() | ||||
|      | ||||
|     def _initialize_agents(self): | ||||
|         """初始化所有八仙智能体""" | ||||
|         for agent_name in MemoryEnhancedAgent.BAXIAN_PERSONALITIES.keys(): | ||||
|             try: | ||||
|                 agent = MemoryEnhancedAgent(agent_name, self.memory_bank) | ||||
|                 self.agents[agent_name] = agent | ||||
|                 print(f"✅ 初始化 {agent.personality.chinese_name}") | ||||
|             except Exception as e: | ||||
|                 print(f"❌ 初始化 {agent_name} 失败: {e}") | ||||
|      | ||||
|     async def conduct_memory_debate(self,  | ||||
|                                    topic: str,  | ||||
|                                    participants: List[str] = None, | ||||
|                                    rounds: int = 3) -> Dict[str, Any]: | ||||
|         """ | ||||
|         进行记忆增强的辩论 | ||||
|          | ||||
|         Args: | ||||
|             topic: 辩论主题 | ||||
|             participants: 参与者列表,None表示所有八仙 | ||||
|             rounds: 辩论轮数 | ||||
|              | ||||
|         Returns: | ||||
|             辩论结果 | ||||
|         """ | ||||
|         if participants is None: | ||||
|             participants = list(self.agents.keys()) | ||||
|          | ||||
|         conversation_history = [] | ||||
|         context = InvocationContext() | ||||
|          | ||||
|         print(f"🏛️ 稷下学宫八仙论道开始: {topic}") | ||||
|          | ||||
|         for round_num in range(rounds): | ||||
|             print(f"\n--- 第 {round_num + 1} 轮 ---") | ||||
|              | ||||
|             for agent_name in participants: | ||||
|                 if agent_name not in self.agents: | ||||
|                     continue | ||||
|                  | ||||
|                 agent = self.agents[agent_name] | ||||
|                  | ||||
|                 # 构建当前轮次的提示 | ||||
|                 round_prompt = f""" | ||||
| 轮次: {round_num + 1}/{rounds} | ||||
| 主题: {topic} | ||||
| 
 | ||||
| 请基于你的记忆和人格特质,对此主题发表观点。 | ||||
| 如果这不是第一轮,请考虑其他仙友的观点并做出回应。 | ||||
| """ | ||||
|                  | ||||
|                 # 获取响应 | ||||
|                 response = await agent.respond_with_memory( | ||||
|                     round_prompt, topic, context | ||||
|                 ) | ||||
|                  | ||||
|                 # 记录对话历史 | ||||
|                 conversation_history.append({ | ||||
|                     "round": round_num + 1, | ||||
|                     "agent": agent_name, | ||||
|                     "chinese_name": agent.personality.chinese_name, | ||||
|                     "content": response | ||||
|                 }) | ||||
|                  | ||||
|                 print(f"{agent.personality.chinese_name}: {response[:100]}...") | ||||
|          | ||||
|         # 保存辩论会话到记忆银行 | ||||
|         if self.memory_bank: | ||||
|             await self.memory_bank.save_debate_session( | ||||
|                 debate_topic=topic, | ||||
|                 participants=participants, | ||||
|                 conversation_history=conversation_history | ||||
|             ) | ||||
|          | ||||
|         return { | ||||
|             "topic": topic, | ||||
|             "participants": participants, | ||||
|             "rounds": rounds, | ||||
|             "conversation_history": conversation_history, | ||||
|             "total_exchanges": len(conversation_history) | ||||
|         } | ||||
|      | ||||
|     async def get_collective_memory_summary(self, topic: str) -> str: | ||||
|         """ | ||||
|         获取集体记忆摘要 | ||||
|          | ||||
|         Args: | ||||
|             topic: 主题 | ||||
|              | ||||
|         Returns: | ||||
|             集体记忆摘要 | ||||
|         """ | ||||
|         if not self.memory_bank: | ||||
|             return "记忆银行未启用" | ||||
|          | ||||
|         summaries = [] | ||||
|          | ||||
|         for agent_name, agent in self.agents.items(): | ||||
|             context = await agent.get_memory_context(topic) | ||||
|             if context and context.strip(): | ||||
|                 summaries.append(context) | ||||
|          | ||||
|         if summaries: | ||||
|             return f"# 稷下学宫集体记忆摘要\n\n" + "\n\n".join(summaries) | ||||
|         else: | ||||
|             return "暂无相关集体记忆" | ||||
| 
 | ||||
| 
 | ||||
| # 便捷函数 | ||||
| async def create_memory_enhanced_council() -> BaxianMemoryCouncil: | ||||
|     """ | ||||
|     创建记忆增强的八仙议会 | ||||
|      | ||||
|     Returns: | ||||
|         配置好的BaxianMemoryCouncil实例 | ||||
|     """ | ||||
|     try: | ||||
|         # 初始化记忆银行 | ||||
|         memory_bank = get_memory_backend() | ||||
|          | ||||
|         # 创建八仙议会 | ||||
|         council = BaxianMemoryCouncil(memory_bank) | ||||
|          | ||||
|         print("🏛️ 稷下学宫记忆增强议会创建完成") | ||||
|         return council | ||||
|          | ||||
|     except Exception as e: | ||||
|         print(f"❌ 创建记忆增强议会失败: {e}") | ||||
|         # 创建无记忆版本 | ||||
|         return BaxianMemoryCouncil(None) | ||||
| 
 | ||||
| 
 | ||||
| if __name__ == "__main__": | ||||
|     async def test_memory_enhanced_agent(): | ||||
|         """测试记忆增强智能体""" | ||||
|         try: | ||||
|             # 创建记忆增强议会 | ||||
|             council = await create_memory_enhanced_council() | ||||
|              | ||||
|             # 进行记忆增强辩论 | ||||
|             result = await council.conduct_memory_debate( | ||||
|                 topic="NVIDIA股票投资分析", | ||||
|                 participants=["tieguaili", "lvdongbin", "hexiangu"], | ||||
|                 rounds=2 | ||||
|             ) | ||||
|              | ||||
|             print(f"\n🏛️ 辩论完成,共 {result['total_exchanges']} 次发言") | ||||
|              | ||||
|             # 获取集体记忆摘要 | ||||
|             summary = await council.get_collective_memory_summary("NVIDIA股票投资分析") | ||||
|             print(f"\n📚 集体记忆摘要:\n{summary}") | ||||
|              | ||||
|         except Exception as e: | ||||
|             print(f"❌ 测试失败: {e}") | ||||
|      | ||||
|     # 运行测试 | ||||
|     asyncio.run(test_memory_enhanced_agent()) | ||||
|  | @ -0,0 +1,216 @@ | |||
| { | ||||
|   "immortals": { | ||||
|     "吕洞宾": { | ||||
|       "title": "主力剑仙", | ||||
|       "specialty": "综合分析与决策", | ||||
|       "description": "作为八仙之首,负责整体投资策略制定,需要最快最准确的数据", | ||||
|       "preferred_apis": { | ||||
|         "stock_quote": "alpha_vantage", | ||||
|         "company_overview": "alpha_vantage", | ||||
|         "market_movers": "yahoo_finance_15", | ||||
|         "market_news": "yahoo_finance_15" | ||||
|       }, | ||||
|       "data_priority": ["实时价格", "公司基本面", "市场动态"], | ||||
|       "api_weight": 0.15 | ||||
|     }, | ||||
|     "何仙姑": { | ||||
|       "title": "风控专家", | ||||
|       "specialty": "风险管理与合规", | ||||
|       "description": "专注风险评估和投资组合管理,需要稳定可靠的数据源", | ||||
|       "preferred_apis": { | ||||
|         "stock_quote": "yahoo_finance_15", | ||||
|         "company_overview": "seeking_alpha", | ||||
|         "market_movers": "webull", | ||||
|         "market_news": "seeking_alpha" | ||||
|       }, | ||||
|       "data_priority": ["波动率", "风险指标", "合规信息"], | ||||
|       "api_weight": 0.12 | ||||
|     }, | ||||
|     "张果老": { | ||||
|       "title": "技术分析师", | ||||
|       "specialty": "技术指标与图表分析", | ||||
|       "description": "专精技术分析,需要详细的价格和成交量数据", | ||||
|       "preferred_apis": { | ||||
|         "stock_quote": "webull", | ||||
|         "company_overview": "alpha_vantage", | ||||
|         "market_movers": "yahoo_finance_15", | ||||
|         "market_news": "yahoo_finance_15" | ||||
|       }, | ||||
|       "data_priority": ["技术指标", "成交量", "价格走势"], | ||||
|       "api_weight": 0.13 | ||||
|     }, | ||||
|     "韩湘子": { | ||||
|       "title": "基本面研究员", | ||||
|       "specialty": "财务分析与估值", | ||||
|       "description": "深度研究公司财务状况和内在价值", | ||||
|       "preferred_apis": { | ||||
|         "stock_quote": "alpha_vantage", | ||||
|         "company_overview": "seeking_alpha", | ||||
|         "market_movers": "webull", | ||||
|         "market_news": "seeking_alpha" | ||||
|       }, | ||||
|       "data_priority": ["财务报表", "估值指标", "盈利预测"], | ||||
|       "api_weight": 0.14 | ||||
|     }, | ||||
|     "汉钟离": { | ||||
|       "title": "量化专家", | ||||
|       "specialty": "数据挖掘与算法交易", | ||||
|       "description": "运用数学模型和算法进行量化分析", | ||||
|       "preferred_apis": { | ||||
|         "stock_quote": "yahoo_finance_15", | ||||
|         "company_overview": "alpha_vantage", | ||||
|         "market_movers": "yahoo_finance_15", | ||||
|         "market_news": "yahoo_finance_15" | ||||
|       }, | ||||
|       "data_priority": ["历史数据", "统计指标", "相关性分析"], | ||||
|       "api_weight": 0.13 | ||||
|     }, | ||||
|     "蓝采和": { | ||||
|       "title": "情绪分析师", | ||||
|       "specialty": "市场情绪与舆情监控", | ||||
|       "description": "分析市场情绪和投资者行为模式", | ||||
|       "preferred_apis": { | ||||
|         "stock_quote": "webull", | ||||
|         "company_overview": "seeking_alpha", | ||||
|         "market_movers": "webull", | ||||
|         "market_news": "seeking_alpha" | ||||
|       }, | ||||
|       "data_priority": ["新闻情绪", "社交媒体", "投资者情绪"], | ||||
|       "api_weight": 0.11 | ||||
|     }, | ||||
|     "曹国舅": { | ||||
|       "title": "宏观分析师", | ||||
|       "specialty": "宏观经济与政策分析", | ||||
|       "description": "关注宏观经济环境和政策影响", | ||||
|       "preferred_apis": { | ||||
|         "stock_quote": "seeking_alpha", | ||||
|         "company_overview": "seeking_alpha", | ||||
|         "market_movers": "yahoo_finance_15", | ||||
|         "market_news": "seeking_alpha" | ||||
|       }, | ||||
|       "data_priority": ["宏观数据", "政策解读", "行业趋势"], | ||||
|       "api_weight": 0.12 | ||||
|     }, | ||||
|     "铁拐李": { | ||||
|       "title": "逆向投资专家", | ||||
|       "specialty": "价值发现与逆向思维", | ||||
|       "description": "寻找被低估的投资机会,逆向思考市场", | ||||
|       "preferred_apis": { | ||||
|         "stock_quote": "alpha_vantage", | ||||
|         "company_overview": "alpha_vantage", | ||||
|         "market_movers": "webull", | ||||
|         "market_news": "yahoo_finance_15" | ||||
|       }, | ||||
|       "data_priority": ["估值偏差", "市场异常", "价值机会"], | ||||
|       "api_weight": 0.10 | ||||
|     } | ||||
|   }, | ||||
|   "api_configurations": { | ||||
|     "alpha_vantage": { | ||||
|       "name": "Alpha Vantage", | ||||
|       "tier": "premium", | ||||
|       "strengths": ["实时数据", "财务数据", "技术指标"], | ||||
|       "rate_limits": { | ||||
|         "per_minute": 500, | ||||
|         "per_month": 500000 | ||||
|       }, | ||||
|       "reliability_score": 0.95, | ||||
|       "response_time_avg": 0.8, | ||||
|       "data_quality": "high", | ||||
|       "cost_per_call": 0.001 | ||||
|     }, | ||||
|     "yahoo_finance_15": { | ||||
|       "name": "Yahoo Finance 15", | ||||
|       "tier": "standard", | ||||
|       "strengths": ["市场数据", "新闻资讯", "实时报价"], | ||||
|       "rate_limits": { | ||||
|         "per_minute": 500, | ||||
|         "per_month": 500000 | ||||
|       }, | ||||
|       "reliability_score": 0.90, | ||||
|       "response_time_avg": 1.2, | ||||
|       "data_quality": "medium", | ||||
|       "cost_per_call": 0.0005 | ||||
|     }, | ||||
|     "webull": { | ||||
|       "name": "Webull", | ||||
|       "tier": "premium", | ||||
|       "strengths": ["搜索功能", "活跃数据", "技术分析"], | ||||
|       "rate_limits": { | ||||
|         "per_minute": 500, | ||||
|         "per_month": 500000 | ||||
|       }, | ||||
|       "reliability_score": 0.88, | ||||
|       "response_time_avg": 1.0, | ||||
|       "data_quality": "high", | ||||
|       "cost_per_call": 0.0008 | ||||
|     }, | ||||
|     "seeking_alpha": { | ||||
|       "name": "Seeking Alpha", | ||||
|       "tier": "standard", | ||||
|       "strengths": ["分析报告", "新闻资讯", "专业观点"], | ||||
|       "rate_limits": { | ||||
|         "per_minute": 500, | ||||
|         "per_month": 500000 | ||||
|       }, | ||||
|       "reliability_score": 0.85, | ||||
|       "response_time_avg": 1.5, | ||||
|       "data_quality": "medium", | ||||
|       "cost_per_call": 0.0006 | ||||
|     } | ||||
|   }, | ||||
|   "load_balancing_strategies": { | ||||
|     "round_robin": { | ||||
|       "description": "轮询分配,确保负载均匀分布", | ||||
|       "enabled": true, | ||||
|       "weight_based": true | ||||
|     }, | ||||
|     "health_aware": { | ||||
|       "description": "基于API健康状态的智能分配", | ||||
|       "enabled": true, | ||||
|       "health_check_interval": 300 | ||||
|     }, | ||||
|     "performance_based": { | ||||
|       "description": "基于响应时间的动态分配", | ||||
|       "enabled": true, | ||||
|       "response_time_threshold": 2.0 | ||||
|     }, | ||||
|     "cost_optimization": { | ||||
|       "description": "成本优化策略,优先使用低成本API", | ||||
|       "enabled": false, | ||||
|       "cost_threshold": 0.001 | ||||
|     } | ||||
|   }, | ||||
|   "failover_matrix": { | ||||
|     "alpha_vantage": ["webull", "yahoo_finance_15", "seeking_alpha"], | ||||
|     "yahoo_finance_15": ["webull", "alpha_vantage", "seeking_alpha"], | ||||
|     "webull": ["alpha_vantage", "yahoo_finance_15", "seeking_alpha"], | ||||
|     "seeking_alpha": ["yahoo_finance_15", "alpha_vantage", "webull"] | ||||
|   }, | ||||
|   "cache_settings": { | ||||
|     "enabled": true, | ||||
|     "ttl_seconds": 300, | ||||
|     "max_entries": 1000, | ||||
|     "cache_strategies": { | ||||
|       "stock_quote": 60, | ||||
|       "company_overview": 3600, | ||||
|       "market_movers": 300, | ||||
|       "market_news": 1800 | ||||
|     } | ||||
|   }, | ||||
|   "monitoring": { | ||||
|     "enabled": true, | ||||
|     "metrics": [ | ||||
|       "api_call_count", | ||||
|       "response_time", | ||||
|       "error_rate", | ||||
|       "cache_hit_rate", | ||||
|       "load_distribution" | ||||
|     ], | ||||
|     "alerts": { | ||||
|       "high_error_rate": 0.1, | ||||
|       "slow_response_time": 3.0, | ||||
|       "api_unavailable": true | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | @ -0,0 +1,130 @@ | |||
| #!/usr/bin/env python3 | ||||
| # -*- coding: utf-8 -*- | ||||
| """ | ||||
| 稷下学宫 Google ADK 论道系统测试 | ||||
| 基于 Google ADK 的八仙论道原型 | ||||
| """ | ||||
| 
 | ||||
| import os | ||||
| import asyncio | ||||
| from google.adk import Agent | ||||
| from google.adk.tools import FunctionTool | ||||
| 
 | ||||
| # 八仙智能体定义 | ||||
| def create_baxian_agents(): | ||||
|     """创建八仙智能体""" | ||||
|      | ||||
|     # 铁拐李 - 逆向思维专家 | ||||
|     tie_guai_li = Agent( | ||||
|         name="铁拐李", | ||||
|         model="gemini-2.0-flash-exp" | ||||
|     ) | ||||
|      | ||||
|     # 汉钟离 - 平衡协调者 | ||||
|     han_zhong_li = Agent( | ||||
|         name="汉钟离", | ||||
|         model="gemini-2.0-flash-exp" | ||||
|     ) | ||||
|      | ||||
|     # 张果老 - 历史智慧者 | ||||
|     zhang_guo_lao = Agent( | ||||
|         name="张果老", | ||||
|         model="gemini-2.0-flash-exp" | ||||
|     ) | ||||
|      | ||||
|     # 蓝采和 - 创新思维者 | ||||
|     lan_cai_he = Agent( | ||||
|         name="蓝采和", | ||||
|         model="gemini-2.0-flash-exp" | ||||
|     ) | ||||
|      | ||||
|     # 何仙姑 - 直觉洞察者 | ||||
|     he_xian_gu = Agent( | ||||
|         name="何仙姑", | ||||
|         model="gemini-2.0-flash-exp" | ||||
|     ) | ||||
|      | ||||
|     # 吕洞宾 - 理性分析者 | ||||
|     lu_dong_bin = Agent( | ||||
|         name="吕洞宾", | ||||
|         model="gemini-2.0-flash-exp" | ||||
|     ) | ||||
|      | ||||
|     # 韩湘子 - 艺术感知者 | ||||
|     han_xiang_zi = Agent( | ||||
|         name="韩湘子", | ||||
|         model="gemini-2.0-flash-exp" | ||||
|     ) | ||||
|      | ||||
|     # 曹国舅 - 实务执行者 | ||||
|     cao_guo_jiu = Agent( | ||||
|         name="曹国舅", | ||||
|         model="gemini-2.0-flash-exp" | ||||
|     ) | ||||
|      | ||||
|     return { | ||||
|         "铁拐李": tie_guai_li, | ||||
|         "汉钟离": han_zhong_li, | ||||
|         "张果老": zhang_guo_lao, | ||||
|         "蓝采和": lan_cai_he, | ||||
|         "何仙姑": he_xian_gu, | ||||
|         "吕洞宾": lu_dong_bin, | ||||
|         "韩湘子": han_xiang_zi, | ||||
|         "曹国舅": cao_guo_jiu | ||||
|     } | ||||
| 
 | ||||
| def test_single_agent(): | ||||
|     """测试单个智能体""" | ||||
|     print("🧪 测试单个智能体...") | ||||
|      | ||||
|     # 创建铁拐李智能体 | ||||
|     tie_guai_li = Agent( | ||||
|         name="铁拐李", | ||||
|         model="gemini-2.0-flash-exp" | ||||
|     ) | ||||
|      | ||||
|     print(f"✅ 智能体 '{tie_guai_li.name}' 创建成功") | ||||
|     print(f"📱 使用模型: {tie_guai_li.model}") | ||||
|      | ||||
|     return tie_guai_li | ||||
| 
 | ||||
| def test_baxian_creation(): | ||||
|     """测试八仙智能体创建""" | ||||
|     print("\n🎭 创建八仙智能体...") | ||||
|      | ||||
|     baxian = create_baxian_agents() | ||||
|      | ||||
|     print(f"✅ 成功创建 {len(baxian)} 个智能体:") | ||||
|     for name, agent in baxian.items(): | ||||
|         print(f"   - {name}: {agent.model}") | ||||
|      | ||||
|     return baxian | ||||
| 
 | ||||
| def main(): | ||||
|     """主测试函数""" | ||||
|     print("🚀 开始稷下学宫 ADK 论道系统测试...") | ||||
|      | ||||
|     # 检查API密钥 | ||||
|     api_key = os.getenv('GOOGLE_API_KEY') | ||||
|     if not api_key: | ||||
|         print("❌ 未找到 GOOGLE_API_KEY 环境变量") | ||||
|         print("请使用: doppler run -- python src/jixia/debates/adk_debate_test.py") | ||||
|         return | ||||
|      | ||||
|     print(f"✅ API密钥已配置 (长度: {len(api_key)} 字符)") | ||||
|      | ||||
|     # 测试单个智能体 | ||||
|     single_agent = test_single_agent() | ||||
|      | ||||
|     # 测试八仙智能体创建 | ||||
|     baxian = test_baxian_creation() | ||||
|      | ||||
|     print("\n🎉 ADK 论道系统基础测试完成!") | ||||
|     print("\n📝 下一步:") | ||||
|     print("   1. 实现智能体间的对话逻辑") | ||||
|     print("   2. 集成 RapidAPI 数据源") | ||||
|     print("   3. 创建论道主题和流程") | ||||
|     print("   4. 连接 Streamlit 界面") | ||||
| 
 | ||||
| if __name__ == "__main__": | ||||
|     main() | ||||
|  | @ -0,0 +1,282 @@ | |||
| #!/usr/bin/env python3 | ||||
| # -*- coding: utf-8 -*- | ||||
| """ | ||||
| 稷下学宫 ADK Memory Bank 论道系统 | ||||
| 实现带有记忆银行的八仙智能体辩论 | ||||
| """ | ||||
| 
 | ||||
| import os | ||||
| import asyncio | ||||
| from google.adk import Agent, Runner | ||||
| from google.adk.sessions import InMemorySessionService | ||||
| from google.adk.memory import MemoryBank, MemoryItem | ||||
| from google.genai import types | ||||
| import json | ||||
| from datetime import datetime | ||||
| from typing import Dict, List, Optional | ||||
| 
 | ||||
| class BaxianMemoryManager: | ||||
|     """八仙记忆管理器""" | ||||
|      | ||||
|     def __init__(self): | ||||
|         self.memory_banks: Dict[str, MemoryBank] = {} | ||||
|         self.agents: Dict[str, Agent] = {} | ||||
|          | ||||
|     async def initialize_baxian_agents(self): | ||||
|         """初始化八仙智能体及其记忆银行""" | ||||
|          | ||||
|         # 八仙角色配置 | ||||
|         baxian_config = { | ||||
|             "铁拐李": { | ||||
|                 "instruction": "你是铁拐李,八仙中的逆向思维专家。你善于从批判和质疑的角度看问题,总是能发现事物的另一面。你会从你的记忆中回忆相关的逆向投资案例和失败教训。", | ||||
|                 "memory_context": "逆向投资案例、市场泡沫警告、风险识别经验" | ||||
|             }, | ||||
|             "吕洞宾": { | ||||
|                 "instruction": "你是吕洞宾,八仙中的理性分析者。你善于平衡各方观点,用理性和逻辑来分析问题。你会从记忆中调用技术分析的成功案例和理论知识。", | ||||
|                 "memory_context": "技术分析理论、成功预测案例、市场趋势分析" | ||||
|             }, | ||||
|             "何仙姑": { | ||||
|                 "instruction": "你是何仙姑,八仙中的风险控制专家。你总是从风险管理的角度思考问题,善于发现潜在危险。你会回忆历史上的重大风险事件。", | ||||
|                 "memory_context": "风险管理案例、黑天鹅事件、危机预警经验" | ||||
|             }, | ||||
|             "张果老": { | ||||
|                 "instruction": "你是张果老,八仙中的历史智慧者。你善于从历史数据中寻找规律和智慧,总是能提供长期视角。你会从记忆中调用历史数据和长期趋势。", | ||||
|                 "memory_context": "历史市场数据、长期投资趋势、周期性规律" | ||||
|             } | ||||
|         } | ||||
|          | ||||
|         # 为每个仙人创建智能体和记忆银行 | ||||
|         for name, config in baxian_config.items(): | ||||
|             # 创建记忆银行 | ||||
|             memory_bank = MemoryBank( | ||||
|                 name=f"{name}_memory_bank", | ||||
|                 description=f"{name}的个人记忆银行,存储{config['memory_context']}" | ||||
|             ) | ||||
|              | ||||
|             # 初始化记忆内容 | ||||
|             await self._initialize_agent_memory(memory_bank, name, config['memory_context']) | ||||
|              | ||||
|             # 创建智能体 | ||||
|             agent = Agent( | ||||
|                 name=name, | ||||
|                 model="gemini-2.0-flash-exp", | ||||
|                 instruction=f"{config['instruction']} 在回答时,请先从你的记忆银行中检索相关信息,然后结合当前话题给出回应。", | ||||
|                 memory_bank=memory_bank | ||||
|             ) | ||||
|              | ||||
|             self.memory_banks[name] = memory_bank | ||||
|             self.agents[name] = agent | ||||
|              | ||||
|         print(f"✅ 已初始化 {len(self.agents)} 个八仙智能体及其记忆银行") | ||||
|          | ||||
|     async def _initialize_agent_memory(self, memory_bank: MemoryBank, agent_name: str, context: str): | ||||
|         """为智能体初始化记忆内容""" | ||||
|          | ||||
|         # 根据角色添加初始记忆 | ||||
|         initial_memories = { | ||||
|             "铁拐李": [ | ||||
|                 "2000年互联网泡沫破裂,许多高估值科技股暴跌90%以上", | ||||
|                 "2008年金融危机前,房地产市场过度繁荣,逆向思维者提前撤离", | ||||
|                 "比特币从2万美元跌到3千美元,提醒我们任何资产都可能大幅回调", | ||||
|                 "巴菲特说过:别人贪婪时我恐惧,别人恐惧时我贪婪" | ||||
|             ], | ||||
|             "吕洞宾": [ | ||||
|                 "移动平均线交叉是经典的技术分析信号", | ||||
|                 "RSI指标超过70通常表示超买,低于30表示超卖", | ||||
|                 "支撑位和阻力位是技术分析的核心概念", | ||||
|                 "成功的技术分析需要结合多个指标综合判断" | ||||
|             ], | ||||
|             "何仙姑": [ | ||||
|                 "2008年雷曼兄弟倒闭引发全球金融危机", | ||||
|                 "长期资本管理公司(LTCM)的失败说明了风险管理的重要性", | ||||
|                 "分散投资是降低风险的基本原则", | ||||
|                 "黑天鹅事件虽然罕见但影响巨大,需要提前准备" | ||||
|             ], | ||||
|             "张果老": [ | ||||
|                 "股市存在7-10年的长期周期", | ||||
|                 "康德拉季耶夫长波理论描述了50-60年的经济周期", | ||||
|                 "历史上每次重大技术革命都带来新的投资机会", | ||||
|                 "长期来看,优质资产总是向上的" | ||||
|             ] | ||||
|         } | ||||
|          | ||||
|         memories = initial_memories.get(agent_name, []) | ||||
|         for memory_text in memories: | ||||
|             memory_item = MemoryItem( | ||||
|                 content=memory_text, | ||||
|                 metadata={ | ||||
|                     "agent": agent_name, | ||||
|                     "type": "historical_knowledge", | ||||
|                     "timestamp": datetime.now().isoformat() | ||||
|                 } | ||||
|             ) | ||||
|             await memory_bank.add_memory(memory_item) | ||||
|              | ||||
|     async def add_debate_memory(self, agent_name: str, content: str, topic: str): | ||||
|         """为智能体添加辩论记忆""" | ||||
|         if agent_name in self.memory_banks: | ||||
|             memory_item = MemoryItem( | ||||
|                 content=content, | ||||
|                 metadata={ | ||||
|                     "agent": agent_name, | ||||
|                     "type": "debate_history", | ||||
|                     "topic": topic, | ||||
|                     "timestamp": datetime.now().isoformat() | ||||
|                 } | ||||
|             ) | ||||
|             await self.memory_banks[agent_name].add_memory(memory_item) | ||||
|              | ||||
|     async def retrieve_relevant_memories(self, agent_name: str, query: str, limit: int = 3) -> List[str]: | ||||
|         """检索智能体的相关记忆""" | ||||
|         if agent_name not in self.memory_banks: | ||||
|             return [] | ||||
|              | ||||
|         try: | ||||
|             memories = await self.memory_banks[agent_name].search(query, limit=limit) | ||||
|             return [memory.content for memory in memories] | ||||
|         except Exception as e: | ||||
|             print(f"⚠️ 记忆检索失败 ({agent_name}): {e}") | ||||
|             return [] | ||||
| 
 | ||||
| class MemoryEnhancedDebate: | ||||
|     """带记忆增强的辩论系统""" | ||||
|      | ||||
|     def __init__(self): | ||||
|         self.memory_manager = BaxianMemoryManager() | ||||
|         self.session_service = InMemorySessionService() | ||||
|         self.runners: Dict[str, Runner] = {} | ||||
|          | ||||
|     async def initialize(self): | ||||
|         """初始化辩论系统""" | ||||
|         await self.memory_manager.initialize_baxian_agents() | ||||
|          | ||||
|         # 创建会话 | ||||
|         self.session = await self.session_service.create_session( | ||||
|             state={}, | ||||
|             app_name="稷下学宫记忆增强论道系统", | ||||
|             user_id="memory_debate_user" | ||||
|         ) | ||||
|          | ||||
|         # 为每个智能体创建Runner | ||||
|         for name, agent in self.memory_manager.agents.items(): | ||||
|             runner = Runner( | ||||
|                 app_name="稷下学宫记忆增强论道系统", | ||||
|                 agent=agent, | ||||
|                 session_service=self.session_service | ||||
|             ) | ||||
|             self.runners[name] = runner | ||||
|              | ||||
|     async def conduct_memory_debate(self, topic: str, participants: List[str] = None): | ||||
|         """进行带记忆的辩论""" | ||||
|         if participants is None: | ||||
|             participants = ["铁拐李", "吕洞宾", "何仙姑", "张果老"] | ||||
|              | ||||
|         print(f"\n🎭 稷下学宫记忆增强论道开始...") | ||||
|         print(f"📋 论道主题: {topic}") | ||||
|         print(f"🎯 参与仙人: {', '.join(participants)}") | ||||
|          | ||||
|         debate_history = [] | ||||
|          | ||||
|         for round_num in range(2):  # 进行2轮辩论 | ||||
|             print(f"\n🔄 第 {round_num + 1} 轮论道:") | ||||
|              | ||||
|             for participant in participants: | ||||
|                 if participant not in self.runners: | ||||
|                     continue | ||||
|                      | ||||
|                 print(f"\n🗣️  {participant} 发言:") | ||||
|                  | ||||
|                 # 检索相关记忆 | ||||
|                 relevant_memories = await self.memory_manager.retrieve_relevant_memories( | ||||
|                     participant, topic, limit=2 | ||||
|                 ) | ||||
|                  | ||||
|                 # 构建包含记忆的提示 | ||||
|                 memory_context = "" | ||||
|                 if relevant_memories: | ||||
|                     memory_context = f"\n从你的记忆中回忆到:\n" + "\n".join([f"- {memory}" for memory in relevant_memories]) | ||||
|                  | ||||
|                 # 构建辩论历史上下文 | ||||
|                 history_context = "" | ||||
|                 if debate_history: | ||||
|                     recent_history = debate_history[-3:]  # 最近3条发言 | ||||
|                     history_context = f"\n最近的论道内容:\n" + "\n".join([f"- {h}" for h in recent_history]) | ||||
|                  | ||||
|                 prompt = f"关于'{topic}'这个话题{memory_context}{history_context}\n\n请结合你的记忆和当前讨论,从你的角色特点出发发表观点。请控制在150字以内。" | ||||
|                  | ||||
|                 # 发送消息并获取回复 | ||||
|                 content = types.Content(role='user', parts=[types.Part(text=prompt)]) | ||||
|                 response = self.runners[participant].run_async( | ||||
|                     user_id=self.session.user_id, | ||||
|                     session_id=self.session.id, | ||||
|                     new_message=content | ||||
|                 ) | ||||
|                  | ||||
|                 # 收集回复 | ||||
|                 reply = "" | ||||
|                 async for event in response: | ||||
|                     if hasattr(event, 'content') and event.content: | ||||
|                         if hasattr(event.content, 'parts') and event.content.parts: | ||||
|                             for part in event.content.parts: | ||||
|                                 if hasattr(part, 'text') and part.text: | ||||
|                                     reply += str(part.text) | ||||
|                  | ||||
|                 if reply.strip(): | ||||
|                     clean_reply = reply.strip() | ||||
|                     print(f"   {clean_reply}") | ||||
|                      | ||||
|                     # 记录到辩论历史 | ||||
|                     debate_entry = f"{participant}: {clean_reply}" | ||||
|                     debate_history.append(debate_entry) | ||||
|                      | ||||
|                     # 添加到记忆银行 | ||||
|                     await self.memory_manager.add_debate_memory( | ||||
|                         participant, clean_reply, topic | ||||
|                     ) | ||||
|                      | ||||
|                 await asyncio.sleep(1)  # 避免API调用过快 | ||||
|                  | ||||
|         print(f"\n🎉 记忆增强论道完成!") | ||||
|         print(f"📝 本次论道共产生 {len(debate_history)} 条发言,已存储到各仙人的记忆银行中。") | ||||
|          | ||||
|         return debate_history | ||||
|          | ||||
|     async def close(self): | ||||
|         """关闭资源""" | ||||
|         for runner in self.runners.values(): | ||||
|             await runner.close() | ||||
| 
 | ||||
| async def main(): | ||||
|     """主函数""" | ||||
|     print("🚀 稷下学宫 ADK Memory Bank 论道系统") | ||||
|      | ||||
|     # 检查API密钥 | ||||
|     api_key = os.getenv('GOOGLE_API_KEY') | ||||
|     if not api_key: | ||||
|         print("❌ 未找到 GOOGLE_API_KEY 环境变量") | ||||
|         print("请使用: doppler run -- python src/jixia/debates/adk_memory_debate.py") | ||||
|         return | ||||
|      | ||||
|     print(f"✅ API密钥已配置") | ||||
|      | ||||
|     # 创建并初始化辩论系统 | ||||
|     debate_system = MemoryEnhancedDebate() | ||||
|      | ||||
|     try: | ||||
|         await debate_system.initialize() | ||||
|          | ||||
|         # 进行辩论 | ||||
|         await debate_system.conduct_memory_debate( | ||||
|             topic="人工智能对投资市场的影响", | ||||
|             participants=["铁拐李", "吕洞宾", "何仙姑", "张果老"] | ||||
|         ) | ||||
|          | ||||
|     except Exception as e: | ||||
|         print(f"❌ 运行失败: {e}") | ||||
|         import traceback | ||||
|         traceback.print_exc() | ||||
|     finally: | ||||
|         await debate_system.close() | ||||
| 
 | ||||
| if __name__ == "__main__": | ||||
|     asyncio.run(main()) | ||||
|  | @ -0,0 +1,252 @@ | |||
| #!/usr/bin/env python3 | ||||
| # -*- coding: utf-8 -*- | ||||
| """ | ||||
| 稷下学宫 ADK 真实论道系统 | ||||
| 实现铁拐李和吕洞宾的实际对话辩论 | ||||
| """ | ||||
| 
 | ||||
| import os | ||||
| import asyncio | ||||
| from google.adk import Agent, Runner | ||||
| from google.adk.sessions import InMemorySessionService | ||||
| from google.genai import types | ||||
| import re | ||||
| import sys | ||||
| from contextlib import contextmanager | ||||
| 
 | ||||
| def create_debate_agents(): | ||||
|     """创建论道智能体""" | ||||
|      | ||||
|     # 铁拐李 - 逆向思维专家 | ||||
|     tie_guai_li = Agent( | ||||
|         name="铁拐李", | ||||
|         model="gemini-2.0-flash-exp", | ||||
|         instruction="你是铁拐李,八仙中的逆向思维专家。你善于从批判和质疑的角度看问题,总是能发现事物的另一面。你的发言风格直接、犀利,但富有智慧。每次发言控制在100字以内。" | ||||
|     ) | ||||
|      | ||||
|     # 吕洞宾 - 理性分析者 | ||||
|     lu_dong_bin = Agent( | ||||
|         name="吕洞宾", | ||||
|         model="gemini-2.0-flash-exp", | ||||
|         instruction="你是吕洞宾,八仙中的理性分析者。你善于平衡各方观点,用理性和逻辑来分析问题。你的发言风格温和而深刻,总是能找到问题的核心。每次发言控制在100字以内。" | ||||
|     ) | ||||
|      | ||||
|     return tie_guai_li, lu_dong_bin | ||||
| 
 | ||||
| async def conduct_debate(): | ||||
|     """进行实际辩论""" | ||||
|     print("🎭 稷下学宫论道开始...") | ||||
|      | ||||
|     # 创建智能体 | ||||
|     tie_guai_li, lu_dong_bin = create_debate_agents() | ||||
|      | ||||
|     print("\n📋 论道主题: 人工智能对未来社会的影响") | ||||
|     print("\n🎯 八仙论道,智慧交锋...") | ||||
|      | ||||
|     try: | ||||
|         print("\n🚀 使用真实ADK调用进行论道...") | ||||
|         await real_adk_debate(tie_guai_li, lu_dong_bin) | ||||
|     except Exception as e: | ||||
|         print(f"\n❌ ADK调用失败: {e}") | ||||
|         print("🔧 回退到模拟对话模式...") | ||||
|         await simple_mock_debate(tie_guai_li, lu_dong_bin) | ||||
| 
 | ||||
| @contextmanager | ||||
| def suppress_stdout(): | ||||
|     """临时抑制stdout输出""" | ||||
|     with open(os.devnull, 'w') as devnull: | ||||
|         old_stdout = sys.stdout | ||||
|         sys.stdout = devnull | ||||
|         try: | ||||
|             yield | ||||
|         finally: | ||||
|             sys.stdout = old_stdout | ||||
| 
 | ||||
| def clean_debug_output(text): | ||||
|     """清理ADK输出中的调试信息""" | ||||
|     if not text: | ||||
|         return "" | ||||
|      | ||||
|     # 移除API密钥相关信息 | ||||
|     text = re.sub(r'Both GOOGLE_API_KEY and GEMINI_API_KEY are set\. Using GOOGLE_API_KEY\.', '', text) | ||||
|      | ||||
|     # 移除Event from unknown agent信息 | ||||
|     text = re.sub(r'Event from an unknown agent: [^\n]*\n?', '', text) | ||||
|      | ||||
|     # 移除多余的空白字符 | ||||
|     text = re.sub(r'\n\s*\n', '\n', text) | ||||
|     text = text.strip() | ||||
|      | ||||
|     return text | ||||
| 
 | ||||
| async def real_adk_debate(tie_guai_li, lu_dong_bin): | ||||
|     """使用真实ADK进行辩论""" | ||||
|     print("\n🔥 真实ADK论道模式") | ||||
|      | ||||
|     # 设置环境变量来抑制ADK调试输出 | ||||
|     os.environ['GOOGLE_CLOUD_DISABLE_GRPC_LOGS'] = 'true' | ||||
|     os.environ['GRPC_VERBOSITY'] = 'NONE' | ||||
|     os.environ['GRPC_TRACE'] = '' | ||||
|      | ||||
|     # 临时抑制警告和调试信息 | ||||
|     import warnings | ||||
|     warnings.filterwarnings('ignore') | ||||
|      | ||||
|     # 设置日志级别 | ||||
|     import logging | ||||
|     logging.getLogger().setLevel(logging.ERROR) | ||||
|      | ||||
|     # 创建会话服务 | ||||
|     session_service = InMemorySessionService() | ||||
|      | ||||
|     # 创建会话 | ||||
|     session = await session_service.create_session( | ||||
|         state={}, | ||||
|         app_name="稷下学宫论道系统", | ||||
|         user_id="debate_user" | ||||
|     ) | ||||
|      | ||||
|     # 创建Runner实例 | ||||
|     tie_runner = Runner( | ||||
|         app_name="稷下学宫论道系统", | ||||
|         agent=tie_guai_li, | ||||
|         session_service=session_service | ||||
|     ) | ||||
|      | ||||
|     lu_runner = Runner( | ||||
|         app_name="稷下学宫论道系统", | ||||
|         agent=lu_dong_bin, | ||||
|         session_service=session_service | ||||
|     ) | ||||
|      | ||||
|     try: | ||||
|         # 第一轮:铁拐李开场 | ||||
|         print("\n🗣️  铁拐李发言:") | ||||
|         tie_prompt = "作为逆向思维专家,请从批判角度分析人工智能对未来社会可能带来的负面影响。请控制在100字以内。" | ||||
|          | ||||
|         tie_content = types.Content(role='user', parts=[types.Part(text=tie_prompt)]) | ||||
|         with suppress_stdout(): | ||||
|             tie_response = tie_runner.run_async( | ||||
|                 user_id=session.user_id, | ||||
|                 session_id=session.id, | ||||
|                 new_message=tie_content | ||||
|             ) | ||||
|          | ||||
|         tie_reply = "" | ||||
|         async for event in tie_response: | ||||
|             # 只处理包含实际文本内容的事件,过滤调试信息 | ||||
|             if hasattr(event, 'content') and event.content: | ||||
|                 if hasattr(event.content, 'parts') and event.content.parts: | ||||
|                     for part in event.content.parts: | ||||
|                         if hasattr(part, 'text') and part.text and part.text.strip(): | ||||
|                             text_content = str(part.text).strip() | ||||
|                             # 过滤掉调试信息和系统消息 | ||||
|                             if not text_content.startswith('Event from') and not 'API_KEY' in text_content: | ||||
|                                 tie_reply += text_content | ||||
|             elif hasattr(event, 'text') and event.text: | ||||
|                 text_content = str(event.text).strip() | ||||
|                 if not text_content.startswith('Event from') and not 'API_KEY' in text_content: | ||||
|                     tie_reply += text_content | ||||
|          | ||||
|         # 清理并输出铁拐李的回复 | ||||
|         clean_tie_reply = clean_debug_output(tie_reply) | ||||
|         if clean_tie_reply: | ||||
|             print(f"   {clean_tie_reply}") | ||||
|          | ||||
|         # 第二轮:吕洞宾回应 | ||||
|         print("\n🗣️  吕洞宾回应:") | ||||
|         lu_prompt = f"铁拐李提到了AI的负面影响:'{tie_reply[:50]}...'。作为理性分析者,请从平衡角度回应,既承认风险又指出机遇。请控制在100字以内。" | ||||
|          | ||||
|         lu_content = types.Content(role='user', parts=[types.Part(text=lu_prompt)]) | ||||
|         with suppress_stdout(): | ||||
|             lu_response = lu_runner.run_async( | ||||
|                 user_id=session.user_id, | ||||
|                 session_id=session.id, | ||||
|                 new_message=lu_content | ||||
|             ) | ||||
|          | ||||
|         lu_reply = "" | ||||
|         async for event in lu_response: | ||||
|             # 只处理包含实际文本内容的事件,过滤调试信息 | ||||
|             if hasattr(event, 'content') and event.content: | ||||
|                 if hasattr(event.content, 'parts') and event.content.parts: | ||||
|                     for part in event.content.parts: | ||||
|                         if hasattr(part, 'text') and part.text and part.text.strip(): | ||||
|                             text_content = str(part.text).strip() | ||||
|                             # 过滤掉调试信息和系统消息 | ||||
|                             if not text_content.startswith('Event from') and not 'API_KEY' in text_content: | ||||
|                                 lu_reply += text_content | ||||
|             elif hasattr(event, 'text') and event.text: | ||||
|                 text_content = str(event.text).strip() | ||||
|                 if not text_content.startswith('Event from') and not 'API_KEY' in text_content: | ||||
|                     lu_reply += text_content | ||||
|          | ||||
|         # 清理并输出吕洞宾的回复 | ||||
|         clean_lu_reply = clean_debug_output(lu_reply) | ||||
|         if clean_lu_reply: | ||||
|             print(f"   {clean_lu_reply}") | ||||
|          | ||||
|         # 第三轮:铁拐李再次发言 | ||||
|         print("\n🗣️  铁拐李再次发言:") | ||||
|         tie_prompt2 = f"吕洞宾提到了AI的机遇:'{lu_reply[:50]}...'。请从逆向思维角度,对这些所谓的机遇进行质疑和反思。请控制在100字以内。" | ||||
|          | ||||
|         tie_content2 = types.Content(role='user', parts=[types.Part(text=tie_prompt2)]) | ||||
|         with suppress_stdout(): | ||||
|             tie_response2 = tie_runner.run_async( | ||||
|                 user_id=session.user_id, | ||||
|                 session_id=session.id, | ||||
|                 new_message=tie_content2 | ||||
|             ) | ||||
|          | ||||
|         tie_reply2 = "" | ||||
|         async for event in tie_response2: | ||||
|             # 只处理包含实际文本内容的事件,过滤调试信息 | ||||
|             if hasattr(event, 'content') and event.content: | ||||
|                 if hasattr(event.content, 'parts') and event.content.parts: | ||||
|                     for part in event.content.parts: | ||||
|                         if hasattr(part, 'text') and part.text and part.text.strip(): | ||||
|                             text_content = str(part.text).strip() | ||||
|                             # 过滤掉调试信息和系统消息 | ||||
|                             if not text_content.startswith('Event from') and not 'API_KEY' in text_content: | ||||
|                                 tie_reply2 += text_content | ||||
|             elif hasattr(event, 'text') and event.text: | ||||
|                 text_content = str(event.text).strip() | ||||
|                 if not text_content.startswith('Event from') and not 'API_KEY' in text_content: | ||||
|                     tie_reply2 += text_content | ||||
|          | ||||
|         # 清理并输出铁拐李的第二次回复 | ||||
|         clean_tie_reply2 = clean_debug_output(tie_reply2) | ||||
|         if clean_tie_reply2: | ||||
|             print(f"   {clean_tie_reply2}") | ||||
|          | ||||
|         print("\n🎉 真实ADK论道完成!") | ||||
|         print("\n📝 智慧交锋,各抒己见,这就是稷下学宫的魅力所在。") | ||||
|          | ||||
|     finally: | ||||
|         # 清理资源 | ||||
|         await tie_runner.close() | ||||
|         await lu_runner.close() | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| def main(): | ||||
|     """主函数""" | ||||
|     print("🚀 稷下学宫 ADK 真实论道系统") | ||||
|      | ||||
|     # 检查API密钥 | ||||
|     api_key = os.getenv('GOOGLE_API_KEY') | ||||
|     if not api_key: | ||||
|         print("❌ 未找到 GOOGLE_API_KEY 环境变量") | ||||
|         print("请使用: doppler run -- python src/jixia/debates/adk_real_debate.py") | ||||
|         return | ||||
|      | ||||
|     print(f"✅ API密钥已配置") | ||||
|      | ||||
|     # 运行异步辩论 | ||||
|     try: | ||||
|         asyncio.run(conduct_debate()) | ||||
|     except Exception as e: | ||||
|         print(f"❌ 运行失败: {e}") | ||||
| 
 | ||||
| if __name__ == "__main__": | ||||
|     main() | ||||
|  | @ -0,0 +1,82 @@ | |||
| #!/usr/bin/env python3 | ||||
| # -*- coding: utf-8 -*- | ||||
| """ | ||||
| 稷下学宫 ADK 简单论道测试 | ||||
| 实现智能体间的基本对话功能 | ||||
| """ | ||||
| 
 | ||||
| import os | ||||
| from google.adk import Agent | ||||
| 
 | ||||
| def create_debate_agents(): | ||||
|     """创建论道智能体""" | ||||
|      | ||||
|     # 铁拐李 - 逆向思维专家 | ||||
|     tie_guai_li = Agent( | ||||
|         name="铁拐李", | ||||
|         model="gemini-2.0-flash-exp" | ||||
|     ) | ||||
|      | ||||
|     # 吕洞宾 - 理性分析者 | ||||
|     lu_dong_bin = Agent( | ||||
|         name="吕洞宾", | ||||
|         model="gemini-2.0-flash-exp" | ||||
|     ) | ||||
|      | ||||
|     return tie_guai_li, lu_dong_bin | ||||
| 
 | ||||
| def simple_debate_test(): | ||||
|     """简单论道测试""" | ||||
|     print("🎭 开始简单论道测试...") | ||||
|      | ||||
|     # 创建智能体 | ||||
|     tie_guai_li, lu_dong_bin = create_debate_agents() | ||||
|      | ||||
|     print("\n📋 论道主题: 人工智能对未来社会的影响") | ||||
|     print("\n🎯 开始论道...") | ||||
|      | ||||
|     try: | ||||
|         # 测试智能体创建 | ||||
|         print("\n✅ 智能体创建成功:") | ||||
|         print(f"   - {tie_guai_li.name}: {tie_guai_li.model}") | ||||
|         print(f"   - {lu_dong_bin.name}: {lu_dong_bin.model}") | ||||
|          | ||||
|         print("\n🎉 简单论道测试完成!") | ||||
|         print("\n📝 智能体基础功能验证成功") | ||||
|          | ||||
|     except Exception as e: | ||||
|         print(f"❌ 论道测试失败: {e}") | ||||
|         return False | ||||
|      | ||||
|     return True | ||||
| 
 | ||||
| def main(): | ||||
|     """主函数""" | ||||
|     print("🚀 稷下学宫 ADK 简单论道系统") | ||||
|      | ||||
|     # 检查API密钥 | ||||
|     api_key = os.getenv('GOOGLE_API_KEY') | ||||
|     if not api_key: | ||||
|         print("❌ 未找到 GOOGLE_API_KEY 环境变量") | ||||
|         print("请使用: doppler run -- python src/jixia/debates/adk_simple_debate.py") | ||||
|         return | ||||
|      | ||||
|     print(f"✅ API密钥已配置") | ||||
|      | ||||
|     # 运行测试 | ||||
|     try: | ||||
|         result = simple_debate_test() | ||||
|         if result: | ||||
|             print("\n📝 测试结果: 成功") | ||||
|             print("\n🎯 下一步开发计划:") | ||||
|             print("   1. 学习ADK的正确调用方式") | ||||
|             print("   2. 实现智能体对话功能") | ||||
|             print("   3. 扩展到八仙全员论道") | ||||
|             print("   4. 集成实时数据源") | ||||
|         else: | ||||
|             print("\n❌ 测试失败") | ||||
|     except Exception as e: | ||||
|         print(f"❌ 运行失败: {e}") | ||||
| 
 | ||||
| if __name__ == "__main__": | ||||
|     main() | ||||
|  | @ -0,0 +1,165 @@ | |||
| #!/usr/bin/env python3 | ||||
| # -*- coding: utf-8 -*- | ||||
| """ | ||||
| 太公心易 - 起承转合辩论系统 | ||||
| """ | ||||
| 
 | ||||
| import json | ||||
| from datetime import datetime | ||||
| from typing import Dict, List, Any | ||||
| from enum import Enum | ||||
| 
 | ||||
| class DebateStage(Enum): | ||||
|     QI = "起"      # 八仙按先天八卦顺序 | ||||
|     CHENG = "承"   # 雁阵式承接 | ||||
|     ZHUAN = "转"   # 自由辩论(36次handoff) | ||||
|     HE = "合"      # 交替总结 | ||||
| 
 | ||||
| class QiChengZhuanHeDebate: | ||||
|     """起承转合辩论系统""" | ||||
|      | ||||
|     def __init__(self): | ||||
|         # 八仙配置(先天八卦顺序) | ||||
|         self.baxian_sequence = ["吕洞宾", "何仙姑", "铁拐李", "汉钟离", "蓝采和", "张果老", "韩湘子", "曹国舅"] | ||||
|          | ||||
|         # 雁阵配置 | ||||
|         self.goose_formation = { | ||||
|             "positive": ["正1", "正2", "正3", "正4"], | ||||
|             "negative": ["反1", "反2", "反3", "反4"] | ||||
|         } | ||||
|          | ||||
|         # 交替总结顺序 | ||||
|         self.alternating_sequence = ["反1", "正1", "反2", "正2", "反3", "正3", "反4", "正4"] | ||||
|          | ||||
|         # 辩论状态 | ||||
|         self.current_stage = DebateStage.QI | ||||
|         self.stage_progress = 0 | ||||
|         self.total_handoffs = 0 | ||||
|         self.debate_history = [] | ||||
|          | ||||
|         # 阶段配置 | ||||
|         self.stage_configs = { | ||||
|             DebateStage.QI: {"max_progress": 8, "description": "八仙按先天八卦顺序"}, | ||||
|             DebateStage.CHENG: {"max_progress": 8, "description": "雁阵式承接"}, | ||||
|             DebateStage.ZHUAN: {"max_progress": 36, "description": "自由辩论"}, | ||||
|             DebateStage.HE: {"max_progress": 8, "description": "交替总结"} | ||||
|         } | ||||
|      | ||||
|     def get_current_speaker(self) -> str: | ||||
|         """获取当前发言者""" | ||||
|         if self.current_stage == DebateStage.QI: | ||||
|             return self.baxian_sequence[self.stage_progress % 8] | ||||
|         elif self.current_stage == DebateStage.CHENG: | ||||
|             if self.stage_progress < 4: | ||||
|                 return self.goose_formation["positive"][self.stage_progress] | ||||
|             else: | ||||
|                 return self.goose_formation["negative"][self.stage_progress - 4] | ||||
|         elif self.current_stage == DebateStage.ZHUAN: | ||||
|             # 简化的优先级算法 | ||||
|             speakers = self.goose_formation["positive"] + self.goose_formation["negative"] | ||||
|             return speakers[self.total_handoffs % 8] | ||||
|         elif self.current_stage == DebateStage.HE: | ||||
|             return self.alternating_sequence[self.stage_progress % 8] | ||||
|          | ||||
|         return "未知发言者" | ||||
|      | ||||
|     def advance_stage(self): | ||||
|         """推进辩论阶段""" | ||||
|         config = self.stage_configs[self.current_stage] | ||||
|          | ||||
|         if self.stage_progress >= config["max_progress"] - 1: | ||||
|             self._transition_to_next_stage() | ||||
|         else: | ||||
|             self.stage_progress += 1 | ||||
|      | ||||
|     def _transition_to_next_stage(self): | ||||
|         """转换到下一阶段""" | ||||
|         transitions = { | ||||
|             DebateStage.QI: DebateStage.CHENG, | ||||
|             DebateStage.CHENG: DebateStage.ZHUAN, | ||||
|             DebateStage.ZHUAN: DebateStage.HE, | ||||
|             DebateStage.HE: None | ||||
|         } | ||||
|          | ||||
|         next_stage = transitions[self.current_stage] | ||||
|         if next_stage: | ||||
|             self.current_stage = next_stage | ||||
|             self.stage_progress = 0 | ||||
|             print(f"🎭 辩论进入{next_stage.value}阶段") | ||||
|         else: | ||||
|             print("🎉 辩论结束!") | ||||
|      | ||||
|     def record_speech(self, speaker: str, message: str): | ||||
|         """记录发言""" | ||||
|         record = { | ||||
|             "timestamp": datetime.now().isoformat(), | ||||
|             "stage": self.current_stage.value, | ||||
|             "progress": self.stage_progress, | ||||
|             "speaker": speaker, | ||||
|             "message": message, | ||||
|             "handoffs": self.total_handoffs | ||||
|         } | ||||
|          | ||||
|         self.debate_history.append(record) | ||||
|          | ||||
|         if self.current_stage == DebateStage.ZHUAN: | ||||
|             self.total_handoffs += 1 | ||||
|      | ||||
|     def get_stage_info(self) -> Dict[str, Any]: | ||||
|         """获取阶段信息""" | ||||
|         config = self.stage_configs[self.current_stage] | ||||
|         return { | ||||
|             "stage": self.current_stage.value, | ||||
|             "progress": self.stage_progress + 1, | ||||
|             "max_progress": config["max_progress"], | ||||
|             "description": config["description"], | ||||
|             "current_speaker": self.get_current_speaker(), | ||||
|             "total_handoffs": self.total_handoffs | ||||
|         } | ||||
|      | ||||
|     def save_state(self, filename: str = "qczh_debate_state.json"): | ||||
|         """保存状态""" | ||||
|         state = { | ||||
|             "current_stage": self.current_stage.value, | ||||
|             "stage_progress": self.stage_progress, | ||||
|             "total_handoffs": self.total_handoffs, | ||||
|             "debate_history": self.debate_history | ||||
|         } | ||||
|          | ||||
|         with open(filename, 'w', encoding='utf-8') as f: | ||||
|             json.dump(state, f, ensure_ascii=False, indent=2) | ||||
|          | ||||
|         print(f"💾 辩论状态已保存到 {filename}") | ||||
| 
 | ||||
| def main(): | ||||
|     """测试函数""" | ||||
|     print("🚀 起承转合辩论系统测试") | ||||
|     print("=" * 50) | ||||
|      | ||||
|     debate = QiChengZhuanHeDebate() | ||||
|      | ||||
|     # 测试各阶段 | ||||
|     test_messages = [ | ||||
|         "起:八仙按先天八卦顺序阐述观点", | ||||
|         "承:雁阵式承接,总体阐述+讥讽", | ||||
|         "转:自由辩论,36次handoff",  | ||||
|         "合:交替总结,最终论证" | ||||
|     ] | ||||
|      | ||||
|     for i, message in enumerate(test_messages): | ||||
|         info = debate.get_stage_info() | ||||
|         speaker = debate.get_current_speaker() | ||||
|          | ||||
|         print(f"\n🎭 阶段: {info['stage']} ({info['progress']}/{info['max_progress']})") | ||||
|         print(f"🗣️ 发言者: {speaker}") | ||||
|         print(f"💬 消息: {message}") | ||||
|          | ||||
|         debate.record_speech(speaker, message) | ||||
|         debate.advance_stage() | ||||
|      | ||||
|     debate.save_state() | ||||
|     print("\n✅ 测试完成!") | ||||
| 
 | ||||
| if __name__ == "__main__": | ||||
|     main() | ||||
| 
 | ||||
|  | @ -0,0 +1,341 @@ | |||
| #!/usr/bin/env python3 | ||||
| # -*- coding: utf-8 -*- | ||||
| """ | ||||
| 太公心易 - 起承转合辩论系统 | ||||
| 基于先天八卦的八仙辩论架构 | ||||
| """ | ||||
| 
 | ||||
| import asyncio | ||||
| import json | ||||
| from datetime import datetime | ||||
| from typing import Dict, List, Any, Optional | ||||
| from dataclasses import dataclass | ||||
| from enum import Enum | ||||
| 
 | ||||
| class DebateStage(Enum): | ||||
|     """辩论阶段枚举""" | ||||
|     QI = "起"      # 八仙按先天八卦顺序 | ||||
|     CHENG = "承"   # 雁阵式承接 | ||||
|     ZHUAN = "转"   # 自由辩论(36次handoff) | ||||
|     HE = "合"      # 交替总结 | ||||
| 
 | ||||
| @dataclass | ||||
| class Speaker: | ||||
|     """发言者数据类""" | ||||
|     name: str | ||||
|     role: str | ||||
|     team: str  # "positive" or "negative" | ||||
|     bagua_position: Optional[int] = None  # 八卦位置(0-7) | ||||
| 
 | ||||
| @dataclass | ||||
| class DebateContext: | ||||
|     """辩论上下文""" | ||||
|     current_stage: DebateStage | ||||
|     stage_progress: int | ||||
|     total_handoffs: int | ||||
|     current_speaker: Optional[str] = None | ||||
|     last_message: Optional[str] = None | ||||
|     debate_history: List[Dict] = None | ||||
| 
 | ||||
| class QiChengZhuanHeDebateSystem: | ||||
|     """起承转合辩论系统""" | ||||
|      | ||||
|     def __init__(self): | ||||
|         # 八仙配置(按先天八卦顺序) | ||||
|         self.baxian_speakers = { | ||||
|             "吕洞宾": Speaker("吕洞宾", "剑仙投资顾问", "neutral", 0),  # 乾 | ||||
|             "何仙姑": Speaker("何仙姑", "慈悲风控专家", "neutral", 1),  # 兑 | ||||
|             "铁拐李": Speaker("铁拐李", "逆向思维专家", "neutral", 2),  # 离 | ||||
|             "汉钟离": Speaker("汉钟离", "平衡协调者", "neutral", 3),    # 震 | ||||
|             "蓝采和": Speaker("蓝采和", "创新思维者", "neutral", 4),    # 巽 | ||||
|             "张果老": Speaker("张果老", "历史智慧者", "neutral", 5),    # 坎 | ||||
|             "韩湘子": Speaker("韩湘子", "艺术感知者", "neutral", 6),    # 艮 | ||||
|             "曹国舅": Speaker("曹国舅", "实务执行者", "neutral", 7)     # 坤 | ||||
|         } | ||||
|          | ||||
|         # 雁阵队伍配置 | ||||
|         self.goose_formation = { | ||||
|             "positive": ["正1", "正2", "正3", "正4"], | ||||
|             "negative": ["反1", "反2", "反3", "反4"] | ||||
|         } | ||||
|          | ||||
|         # 辩论状态 | ||||
|         self.context = DebateContext( | ||||
|             current_stage=DebateStage.QI, | ||||
|             stage_progress=0, | ||||
|             total_handoffs=0, | ||||
|             debate_history=[] | ||||
|         ) | ||||
|          | ||||
|         # 阶段配置 | ||||
|         self.stage_configs = { | ||||
|             DebateStage.QI: { | ||||
|                 "duration": "8-10分钟", | ||||
|                 "max_progress": 8,  # 八仙轮流发言 | ||||
|                 "description": "八仙按先天八卦顺序阐述观点" | ||||
|             }, | ||||
|             DebateStage.CHENG: { | ||||
|                 "duration": "8-10分钟",  | ||||
|                 "max_progress": 8,  # 正反各4人 | ||||
|                 "description": "雁阵式承接,总体阐述+讥讽" | ||||
|             }, | ||||
|             DebateStage.ZHUAN: { | ||||
|                 "duration": "12-15分钟", | ||||
|                 "max_progress": 36,  # 36次handoff | ||||
|                 "description": "自由辩论,优先级算法决定发言" | ||||
|             }, | ||||
|             DebateStage.HE: { | ||||
|                 "duration": "8-10分钟", | ||||
|                 "max_progress": 8,  # 交替总结 | ||||
|                 "description": "交替总结,最终论证" | ||||
|             } | ||||
|         } | ||||
|          | ||||
|         # 优先级算法 | ||||
|         self.priority_algorithm = PriorityAlgorithm() | ||||
|          | ||||
|         # 记忆系统 | ||||
|         self.memory_system = DebateMemorySystem() | ||||
|      | ||||
|     def get_current_speaker(self) -> str: | ||||
|         """获取当前发言者""" | ||||
|         stage = self.context.current_stage | ||||
|         progress = self.context.stage_progress | ||||
|          | ||||
|         if stage == DebateStage.QI: | ||||
|             return self._get_bagua_speaker(progress) | ||||
|         elif stage == DebateStage.CHENG: | ||||
|             return self._get_goose_formation_speaker(progress) | ||||
|         elif stage == DebateStage.ZHUAN: | ||||
|             return self._get_priority_speaker() | ||||
|         elif stage == DebateStage.HE: | ||||
|             return self._get_alternating_speaker(progress) | ||||
|          | ||||
|         return "未知发言者" | ||||
|      | ||||
|     def _get_bagua_speaker(self, progress: int) -> str: | ||||
|         """获取八卦顺序发言者""" | ||||
|         bagua_sequence = ["吕洞宾", "何仙姑", "铁拐李", "汉钟离", "蓝采和", "张果老", "韩湘子", "曹国舅"] | ||||
|         return bagua_sequence[progress % 8] | ||||
|      | ||||
|     def _get_goose_formation_speaker(self, progress: int) -> str: | ||||
|         """获取雁阵发言者""" | ||||
|         if progress < 4: | ||||
|             # 正方雁阵 | ||||
|             return self.goose_formation["positive"][progress] | ||||
|         else: | ||||
|             # 反方雁阵 | ||||
|             return self.goose_formation["negative"][progress - 4] | ||||
|      | ||||
|     def _get_priority_speaker(self) -> str: | ||||
|         """获取优先级发言者""" | ||||
|         return self.priority_algorithm.calculate_next_speaker(self.context) | ||||
|      | ||||
|     def _get_alternating_speaker(self, progress: int) -> str: | ||||
|         """获取交替总结发言者""" | ||||
|         alternating_sequence = ["反1", "正1", "反2", "正2", "反3", "正3", "反4", "正4"] | ||||
|         return alternating_sequence[progress % 8] | ||||
|      | ||||
|     def advance_stage(self): | ||||
|         """推进辩论阶段""" | ||||
|         current_config = self.stage_configs[self.context.current_stage] | ||||
|          | ||||
|         if self.context.stage_progress >= current_config["max_progress"] - 1: | ||||
|             # 当前阶段完成,进入下一阶段 | ||||
|             self._transition_to_next_stage() | ||||
|         else: | ||||
|             # 当前阶段继续 | ||||
|             self.context.stage_progress += 1 | ||||
|      | ||||
|     def _transition_to_next_stage(self): | ||||
|         """转换到下一阶段""" | ||||
|         stage_transitions = { | ||||
|             DebateStage.QI: DebateStage.CHENG, | ||||
|             DebateStage.CHENG: DebateStage.ZHUAN, | ||||
|             DebateStage.ZHUAN: DebateStage.HE, | ||||
|             DebateStage.HE: None  # 辩论结束 | ||||
|         } | ||||
|          | ||||
|         next_stage = stage_transitions[self.context.current_stage] | ||||
|         if next_stage: | ||||
|             self.context.current_stage = next_stage | ||||
|             self.context.stage_progress = 0 | ||||
|             print(f"🎭 辩论进入{next_stage.value}阶段") | ||||
|         else: | ||||
|             print("🎉 辩论结束!") | ||||
|      | ||||
|     def record_speech(self, speaker: str, message: str): | ||||
|         """记录发言""" | ||||
|         speech_record = { | ||||
|             "timestamp": datetime.now().isoformat(), | ||||
|             "stage": self.context.current_stage.value, | ||||
|             "stage_progress": self.context.stage_progress, | ||||
|             "speaker": speaker, | ||||
|             "message": message, | ||||
|             "total_handoffs": self.context.total_handoffs | ||||
|         } | ||||
|          | ||||
|         self.context.debate_history.append(speech_record) | ||||
|         self.context.last_message = message | ||||
|         self.context.current_speaker = speaker | ||||
|          | ||||
|         # 更新记忆系统 | ||||
|         self.memory_system.store_speech(speaker, message, self.context) | ||||
|          | ||||
|         # 如果是转阶段,增加handoff计数 | ||||
|         if self.context.current_stage == DebateStage.ZHUAN: | ||||
|             self.context.total_handoffs += 1 | ||||
|      | ||||
|     def get_stage_info(self) -> Dict[str, Any]: | ||||
|         """获取当前阶段信息""" | ||||
|         stage = self.context.current_stage | ||||
|         config = self.stage_configs[stage] | ||||
|          | ||||
|         return { | ||||
|             "current_stage": stage.value, | ||||
|             "stage_progress": self.context.stage_progress, | ||||
|             "max_progress": config["max_progress"], | ||||
|             "description": config["description"], | ||||
|             "current_speaker": self.get_current_speaker(), | ||||
|             "total_handoffs": self.context.total_handoffs | ||||
|         } | ||||
|      | ||||
|     def save_debate_state(self, filename: str = "debate_state.json"): | ||||
|         """保存辩论状态""" | ||||
|         state_data = { | ||||
|             "context": { | ||||
|                 "current_stage": self.context.current_stage.value, | ||||
|                 "stage_progress": self.context.stage_progress, | ||||
|                 "total_handoffs": self.context.total_handoffs, | ||||
|                 "current_speaker": self.context.current_speaker, | ||||
|                 "last_message": self.context.last_message | ||||
|             }, | ||||
|             "debate_history": self.context.debate_history, | ||||
|             "memory_data": self.memory_system.get_memory_data() | ||||
|         } | ||||
|          | ||||
|         with open(filename, 'w', encoding='utf-8') as f: | ||||
|             json.dump(state_data, f, ensure_ascii=False, indent=2) | ||||
|          | ||||
|         print(f"💾 辩论状态已保存到 {filename}") | ||||
| 
 | ||||
| class PriorityAlgorithm: | ||||
|     """优先级算法""" | ||||
|      | ||||
|     def __init__(self): | ||||
|         self.speaker_weights = { | ||||
|             "rebuttal_urgency": 0.3, | ||||
|             "argument_strength": 0.25, | ||||
|             "time_pressure": 0.2, | ||||
|             "audience_reaction": 0.15, | ||||
|             "strategy_need": 0.1 | ||||
|         } | ||||
|      | ||||
|     def calculate_next_speaker(self, context: DebateContext) -> str: | ||||
|         """计算下一个发言者""" | ||||
|         # 简化的优先级算法 | ||||
|         available_speakers = ["正1", "正2", "正3", "正4", "反1", "反2", "反3", "反4"] | ||||
|          | ||||
|         # 基于当前上下文计算优先级 | ||||
|         priorities = {} | ||||
|         for speaker in available_speakers: | ||||
|             priority_score = self._calculate_speaker_priority(speaker, context) | ||||
|             priorities[speaker] = priority_score | ||||
|          | ||||
|         # 选择最高优先级发言者 | ||||
|         return max(priorities, key=priorities.get) | ||||
|      | ||||
|     def _calculate_speaker_priority(self, speaker: str, context: DebateContext) -> float: | ||||
|         """计算发言者优先级""" | ||||
|         # 简化的优先级计算 | ||||
|         base_score = 0.5 | ||||
|          | ||||
|         # 根据发言者角色调整 | ||||
|         if "正" in speaker: | ||||
|             base_score += 0.1 | ||||
|         if "反" in speaker: | ||||
|             base_score += 0.1 | ||||
|          | ||||
|         # 根据handoff次数调整 | ||||
|         if context.total_handoffs % 2 == 0: | ||||
|             base_score += 0.2 | ||||
|          | ||||
|         return base_score | ||||
| 
 | ||||
| class DebateMemorySystem: | ||||
|     """辩论记忆系统""" | ||||
|      | ||||
|     def __init__(self): | ||||
|         self.speaker_memories = {} | ||||
|         self.debate_memories = [] | ||||
|      | ||||
|     def store_speech(self, speaker: str, message: str, context: DebateContext): | ||||
|         """存储发言记忆""" | ||||
|         if speaker not in self.speaker_memories: | ||||
|             self.speaker_memories[speaker] = [] | ||||
|          | ||||
|         memory_entry = { | ||||
|             "timestamp": datetime.now().isoformat(), | ||||
|             "stage": context.current_stage.value, | ||||
|             "message": message, | ||||
|             "context": { | ||||
|                 "stage_progress": context.stage_progress, | ||||
|                 "total_handoffs": context.total_handoffs | ||||
|             } | ||||
|         } | ||||
|          | ||||
|         self.speaker_memories[speaker].append(memory_entry) | ||||
|         self.debate_memories.append(memory_entry) | ||||
|      | ||||
|     def get_speaker_memory(self, speaker: str, limit: int = 5) -> List[Dict]: | ||||
|         """获取发言者记忆""" | ||||
|         if speaker in self.speaker_memories: | ||||
|             return self.speaker_memories[speaker][-limit:] | ||||
|         return [] | ||||
|      | ||||
|     def get_memory_data(self) -> Dict[str, Any]: | ||||
|         """获取记忆数据""" | ||||
|         return { | ||||
|             "speaker_memories": self.speaker_memories, | ||||
|             "debate_memories": self.debate_memories | ||||
|         } | ||||
| 
 | ||||
| def main(): | ||||
|     """主函数 - 测试起承转合辩论系统""" | ||||
|     print("🚀 太公心易 - 起承转合辩论系统") | ||||
|     print("=" * 60) | ||||
|      | ||||
|     # 创建辩论系统 | ||||
|     debate_system = QiChengZhuanHeDebateSystem() | ||||
|      | ||||
|     # 测试各阶段 | ||||
|     test_messages = [ | ||||
|         "起:八仙按先天八卦顺序阐述观点", | ||||
|         "承:雁阵式承接,总体阐述+讥讽",  | ||||
|         "转:自由辩论,36次handoff", | ||||
|         "合:交替总结,最终论证" | ||||
|     ] | ||||
|      | ||||
|     for i, message in enumerate(test_messages): | ||||
|         stage_info = debate_system.get_stage_info() | ||||
|         current_speaker = debate_system.get_current_speaker() | ||||
|          | ||||
|         print(f"\n🎭 当前阶段: {stage_info['current_stage']}") | ||||
|         print(f"📊 进度: {stage_info['stage_progress'] + 1}/{stage_info['max_progress']}") | ||||
|         print(f"🗣️ 发言者: {current_speaker}") | ||||
|         print(f"💬 消息: {message}") | ||||
|          | ||||
|         # 记录发言 | ||||
|         debate_system.record_speech(current_speaker, message) | ||||
|          | ||||
|         # 推进阶段 | ||||
|         debate_system.advance_stage() | ||||
|      | ||||
|     # 保存状态 | ||||
|     debate_system.save_debate_state() | ||||
|      | ||||
|     print("\n✅ 起承转合辩论系统测试完成!") | ||||
| 
 | ||||
| if __name__ == "__main__": | ||||
|     main() | ||||
| 
 | ||||
|  | @ -0,0 +1,39 @@ | |||
| #!/usr/bin/env python3 | ||||
| """ | ||||
| 通用记忆银行抽象,便于插入不同后端(Vertex、Cloudflare AutoRAG等) | ||||
| """ | ||||
| from __future__ import annotations | ||||
| 
 | ||||
| from typing import Dict, List, Any, Optional, Protocol, runtime_checkable | ||||
| 
 | ||||
| 
 | ||||
| @runtime_checkable | ||||
| class MemoryBankProtocol(Protocol): | ||||
|     async def create_memory_bank(self, agent_name: str, display_name: Optional[str] = None) -> str: ... | ||||
| 
 | ||||
|     async def add_memory( | ||||
|         self, | ||||
|         agent_name: str, | ||||
|         content: str, | ||||
|         memory_type: str = "conversation", | ||||
|         debate_topic: str = "", | ||||
|         metadata: Optional[Dict[str, Any]] = None, | ||||
|     ) -> str: ... | ||||
| 
 | ||||
|     async def search_memories( | ||||
|         self, | ||||
|         agent_name: str, | ||||
|         query: str, | ||||
|         memory_type: Optional[str] = None, | ||||
|         limit: int = 10, | ||||
|     ) -> List[Dict[str, Any]]: ... | ||||
| 
 | ||||
|     async def get_agent_context(self, agent_name: str, debate_topic: str) -> str: ... | ||||
| 
 | ||||
|     async def save_debate_session( | ||||
|         self, | ||||
|         debate_topic: str, | ||||
|         participants: List[str], | ||||
|         conversation_history: List[Dict[str, str]], | ||||
|         outcomes: Optional[Dict[str, Any]] = None, | ||||
|     ) -> None: ... | ||||
|  | @ -0,0 +1,37 @@ | |||
| #!/usr/bin/env python3 | ||||
| """ | ||||
| 记忆银行工厂:根据配置创建 Vertex 实现 | ||||
| """ | ||||
| from __future__ import annotations | ||||
| 
 | ||||
| import os | ||||
| from typing import Optional | ||||
| 
 | ||||
| from .base_memory_bank import MemoryBankProtocol | ||||
| from .vertex_memory_bank import VertexMemoryBank | ||||
| 
 | ||||
| 
 | ||||
| def get_memory_backend(prefer: Optional[str] = None) -> MemoryBankProtocol: | ||||
|     """ | ||||
|     根据环境变量选择记忆后端: | ||||
|     - JIXIA_MEMORY_BACKEND=vertex (默认) | ||||
|     - 如果未设置,默认使用 Vertex | ||||
|     """ | ||||
|     # 从环境变量读取后端选择,默认为 vertex | ||||
|     backend = os.getenv("JIXIA_MEMORY_BACKEND", "vertex").lower() | ||||
|      | ||||
|     if prefer: | ||||
|         backend = prefer.lower() | ||||
| 
 | ||||
|     if backend != "vertex": | ||||
|         raise ValueError(f"不支持的记忆后端: {backend},当前只支持 'vertex'") | ||||
| 
 | ||||
|     # Vertex 作为唯一后端 | ||||
|     try: | ||||
|         mem = VertexMemoryBank.from_config() | ||||
|         return mem | ||||
|     except Exception as e: | ||||
|         # 不可用时抛错 | ||||
|         raise RuntimeError( | ||||
|             "未能创建 Vertex 记忆后端:请配置 Vertex (GOOGLE_*) 环境变量" | ||||
|         ) from e | ||||
|  | @ -0,0 +1,463 @@ | |||
| #!/usr/bin/env python3 | ||||
| """ | ||||
| Vertex AI Memory Bank 集成模块 | ||||
| 为稷下学宫AI辩论系统提供记忆银行功能 | ||||
| """ | ||||
| 
 | ||||
| import os | ||||
| from typing import Dict, List, Optional, Any | ||||
| from dataclasses import dataclass | ||||
| from datetime import datetime | ||||
| import json | ||||
| 
 | ||||
| try: | ||||
|     from google.cloud import aiplatform | ||||
|     # Memory Bank 功能可能还在预览版中,先使用基础功能 | ||||
|     VERTEX_AI_AVAILABLE = True | ||||
| except ImportError: | ||||
|     VERTEX_AI_AVAILABLE = False | ||||
|     print("⚠️ Google Cloud AI Platform 未安装,Memory Bank功能不可用") | ||||
|     print("安装命令: pip install google-cloud-aiplatform") | ||||
| 
 | ||||
| from config.doppler_config import get_google_genai_config | ||||
| 
 | ||||
| 
 | ||||
| @dataclass | ||||
| class MemoryEntry: | ||||
|     """记忆条目数据结构""" | ||||
|     content: str | ||||
|     metadata: Dict[str, Any] | ||||
|     timestamp: datetime | ||||
|     agent_name: str | ||||
|     debate_topic: str | ||||
|     memory_type: str  # "conversation", "preference", "knowledge", "strategy" | ||||
| 
 | ||||
| 
 | ||||
| class VertexMemoryBank: | ||||
|     """ | ||||
|     Vertex AI Memory Bank 管理器 | ||||
|     为八仙辩论系统提供智能记忆功能 | ||||
|     """ | ||||
|      | ||||
|     def __init__(self, project_id: str, location: str = "us-central1"): | ||||
|         """ | ||||
|         初始化Memory Bank | ||||
|          | ||||
|         Args: | ||||
|             project_id: Google Cloud项目ID | ||||
|             location: 部署区域 | ||||
|         """ | ||||
|         if not VERTEX_AI_AVAILABLE: | ||||
|             print("⚠️ Google Cloud AI Platform 未安装,使用本地模拟模式") | ||||
|             # 不抛出异常,允许使用本地模拟模式 | ||||
|              | ||||
|         self.project_id = project_id | ||||
|         self.location = location | ||||
|         self.memory_banks = {}  # 存储不同智能体的记忆银行 | ||||
|         self.local_memories = {}  # 本地记忆存储 (临时方案) | ||||
|          | ||||
|         # 初始化AI Platform | ||||
|         try: | ||||
|             aiplatform.init(project=project_id, location=location) | ||||
|             print(f"✅ Vertex AI 初始化成功: {project_id} @ {location}") | ||||
|         except Exception as e: | ||||
|             print(f"⚠️ Vertex AI 初始化失败,使用本地模拟模式: {e}") | ||||
|          | ||||
|         # 八仙智能体名称映射 | ||||
|         self.baxian_agents = { | ||||
|             "tieguaili": "铁拐李", | ||||
|             "hanzhongli": "汉钟离",  | ||||
|             "zhangguolao": "张果老", | ||||
|             "lancaihe": "蓝采和", | ||||
|             "hexiangu": "何仙姑", | ||||
|             "lvdongbin": "吕洞宾", | ||||
|             "hanxiangzi": "韩湘子", | ||||
|             "caoguojiu": "曹国舅" | ||||
|         } | ||||
|      | ||||
|     @classmethod | ||||
|     def from_config(cls) -> 'VertexMemoryBank': | ||||
|         """ | ||||
|         从配置创建Memory Bank实例 | ||||
|          | ||||
|         Returns: | ||||
|             VertexMemoryBank实例 | ||||
|         """ | ||||
|         config = get_google_genai_config() | ||||
|         project_id = config.get('project_id') | ||||
|         location = config.get('location', 'us-central1') | ||||
|          | ||||
|         if not project_id: | ||||
|             raise ValueError("Google Cloud Project ID 未配置,请设置 GOOGLE_CLOUD_PROJECT_ID") | ||||
|              | ||||
|         return cls(project_id=project_id, location=location) | ||||
|      | ||||
|     async def create_memory_bank(self, agent_name: str, display_name: str = None) -> str: | ||||
|         """ | ||||
|         为指定智能体创建记忆银行 | ||||
|          | ||||
|         Args: | ||||
|             agent_name: 智能体名称 (如 "tieguaili") | ||||
|             display_name: 显示名称 (如 "铁拐李的记忆银行") | ||||
|              | ||||
|         Returns: | ||||
|             记忆银行ID | ||||
|         """ | ||||
|         if not display_name: | ||||
|             chinese_name = self.baxian_agents.get(agent_name, agent_name) | ||||
|             display_name = f"{chinese_name}的记忆银行" | ||||
|          | ||||
|         try: | ||||
|             # 使用本地存储模拟记忆银行 (临时方案) | ||||
|             memory_bank_id = f"memory_bank_{agent_name}_{self.project_id}" | ||||
|              | ||||
|             # 初始化本地记忆存储 | ||||
|             if agent_name not in self.local_memories: | ||||
|                 self.local_memories[agent_name] = [] | ||||
|              | ||||
|             self.memory_banks[agent_name] = memory_bank_id | ||||
|             print(f"✅ 为 {display_name} 创建记忆银行: {memory_bank_id}") | ||||
|              | ||||
|             return memory_bank_id | ||||
|              | ||||
|         except Exception as e: | ||||
|             print(f"❌ 创建记忆银行失败: {e}") | ||||
|             raise | ||||
|      | ||||
|     async def add_memory(self,  | ||||
|                         agent_name: str,  | ||||
|                         content: str,  | ||||
|                         memory_type: str = "conversation", | ||||
|                         debate_topic: str = "", | ||||
|                         metadata: Dict[str, Any] = None) -> str: | ||||
|         """ | ||||
|         添加记忆到指定智能体的记忆银行 | ||||
|          | ||||
|         Args: | ||||
|             agent_name: 智能体名称 | ||||
|             content: 记忆内容 | ||||
|             memory_type: 记忆类型 ("conversation", "preference", "knowledge", "strategy") | ||||
|             debate_topic: 辩论主题 | ||||
|             metadata: 额外元数据 | ||||
|              | ||||
|         Returns: | ||||
|             记忆ID | ||||
|         """ | ||||
|         if agent_name not in self.memory_banks: | ||||
|             await self.create_memory_bank(agent_name) | ||||
|          | ||||
|         if metadata is None: | ||||
|             metadata = {} | ||||
|          | ||||
|         # 构建记忆条目 | ||||
|         memory_entry = MemoryEntry( | ||||
|             content=content, | ||||
|             metadata={ | ||||
|                 **metadata, | ||||
|                 "agent_name": agent_name, | ||||
|                 "chinese_name": self.baxian_agents.get(agent_name, agent_name), | ||||
|                 "memory_type": memory_type, | ||||
|                 "debate_topic": debate_topic, | ||||
|                 "system": "jixia_academy" | ||||
|             }, | ||||
|             timestamp=datetime.now(), | ||||
|             agent_name=agent_name, | ||||
|             debate_topic=debate_topic, | ||||
|             memory_type=memory_type | ||||
|         ) | ||||
|          | ||||
|         try: | ||||
|             # 使用本地存储添加记忆 (临时方案) | ||||
|             memory_id = f"memory_{agent_name}_{len(self.local_memories[agent_name])}" | ||||
|              | ||||
|             # 添加到本地存储 | ||||
|             memory_data = { | ||||
|                 "id": memory_id, | ||||
|                 "content": content, | ||||
|                 "metadata": memory_entry.metadata, | ||||
|                 "timestamp": memory_entry.timestamp.isoformat(), | ||||
|                 "memory_type": memory_type, | ||||
|                 "debate_topic": debate_topic | ||||
|             } | ||||
|              | ||||
|             self.local_memories[agent_name].append(memory_data) | ||||
|              | ||||
|             print(f"✅ 为 {self.baxian_agents.get(agent_name)} 添加记忆: {memory_type}") | ||||
|             return memory_id | ||||
|              | ||||
|         except Exception as e: | ||||
|             print(f"❌ 添加记忆失败: {e}") | ||||
|             raise | ||||
|      | ||||
|     async def search_memories(self,  | ||||
|                              agent_name: str,  | ||||
|                              query: str,  | ||||
|                              memory_type: str = None, | ||||
|                              limit: int = 10) -> List[Dict[str, Any]]: | ||||
|         """ | ||||
|         搜索智能体的相关记忆 | ||||
|          | ||||
|         Args: | ||||
|             agent_name: 智能体名称 | ||||
|             query: 搜索查询 | ||||
|             memory_type: 记忆类型过滤 | ||||
|             limit: 返回结果数量限制 | ||||
|              | ||||
|         Returns: | ||||
|             相关记忆列表 | ||||
|         """ | ||||
|         if agent_name not in self.memory_banks: | ||||
|             return [] | ||||
|          | ||||
|         try: | ||||
|             # 使用本地存储搜索记忆 (临时方案) | ||||
|             if agent_name not in self.local_memories: | ||||
|                 return [] | ||||
|              | ||||
|             memories = self.local_memories[agent_name] | ||||
|             results = [] | ||||
|              | ||||
|             # 简单的文本匹配搜索 | ||||
|             query_lower = query.lower() | ||||
|              | ||||
|             for memory in memories: | ||||
|                 # 检查记忆类型过滤 | ||||
|                 if memory_type and memory.get("memory_type") != memory_type: | ||||
|                     continue | ||||
|                  | ||||
|                 # 检查内容匹配 | ||||
|                 content_lower = memory["content"].lower() | ||||
|                 debate_topic_lower = memory.get("debate_topic", "").lower() | ||||
|                  | ||||
|                 # 在内容或辩论主题中搜索 | ||||
|                 if query_lower in content_lower or query_lower in debate_topic_lower: | ||||
|                     # 计算简单的相关性分数 | ||||
|                     content_matches = content_lower.count(query_lower) | ||||
|                     topic_matches = debate_topic_lower.count(query_lower) | ||||
|                     total_words = len(content_lower.split()) + len(debate_topic_lower.split()) | ||||
|                      | ||||
|                     relevance_score = (content_matches + topic_matches) / max(total_words, 1) | ||||
|                      | ||||
|                     results.append({ | ||||
|                         "content": memory["content"], | ||||
|                         "metadata": memory["metadata"], | ||||
|                         "relevance_score": relevance_score | ||||
|                     }) | ||||
|              | ||||
|             # 按相关性排序并限制结果数量 | ||||
|             results.sort(key=lambda x: x["relevance_score"], reverse=True) | ||||
|             return results[:limit] | ||||
|              | ||||
|         except Exception as e: | ||||
|             print(f"❌ 搜索记忆失败: {e}") | ||||
|             return [] | ||||
|      | ||||
|     async def get_agent_context(self, agent_name: str, debate_topic: str) -> str: | ||||
|         """ | ||||
|         获取智能体在特定辩论主题下的上下文记忆 | ||||
|          | ||||
|         Args: | ||||
|             agent_name: 智能体名称 | ||||
|             debate_topic: 辩论主题 | ||||
|              | ||||
|         Returns: | ||||
|             格式化的上下文字符串 | ||||
|         """ | ||||
|         # 搜索相关记忆 | ||||
|         conversation_memories = await self.search_memories( | ||||
|             agent_name, debate_topic, "conversation", limit=5 | ||||
|         ) | ||||
|         preference_memories = await self.search_memories( | ||||
|             agent_name, debate_topic, "preference", limit=3 | ||||
|         ) | ||||
|         strategy_memories = await self.search_memories( | ||||
|             agent_name, debate_topic, "strategy", limit=3 | ||||
|         ) | ||||
|          | ||||
|         # 构建上下文 | ||||
|         context_parts = [] | ||||
|          | ||||
|         if conversation_memories: | ||||
|             context_parts.append("## 历史对话记忆") | ||||
|             for mem in conversation_memories: | ||||
|                 context_parts.append(f"- {mem['content']}") | ||||
|          | ||||
|         if preference_memories: | ||||
|             context_parts.append("\n## 偏好记忆") | ||||
|             for mem in preference_memories: | ||||
|                 context_parts.append(f"- {mem['content']}") | ||||
|          | ||||
|         if strategy_memories: | ||||
|             context_parts.append("\n## 策略记忆") | ||||
|             for mem in strategy_memories: | ||||
|                 context_parts.append(f"- {mem['content']}") | ||||
|          | ||||
|         chinese_name = self.baxian_agents.get(agent_name, agent_name) | ||||
|         if context_parts: | ||||
|             return f"# {chinese_name}的记忆上下文\n\n" + "\n".join(context_parts) | ||||
|         else: | ||||
|             return f"# {chinese_name}的记忆上下文\n\n暂无相关记忆。" | ||||
|      | ||||
|     async def save_debate_session(self,  | ||||
|                                  debate_topic: str,  | ||||
|                                  participants: List[str], | ||||
|                                  conversation_history: List[Dict[str, str]], | ||||
|                                  outcomes: Dict[str, Any] = None) -> None: | ||||
|         """ | ||||
|         保存完整的辩论会话到各参与者的记忆银行 | ||||
|          | ||||
|         Args: | ||||
|             debate_topic: 辩论主题 | ||||
|             participants: 参与者列表 | ||||
|             conversation_history: 对话历史 | ||||
|             outcomes: 辩论结果和洞察 | ||||
|         """ | ||||
|         for agent_name in participants: | ||||
|             if agent_name not in self.baxian_agents: | ||||
|                 continue | ||||
|              | ||||
|             # 保存对话历史 | ||||
|             conversation_summary = self._summarize_conversation( | ||||
|                 conversation_history, agent_name | ||||
|             ) | ||||
|             await self.add_memory( | ||||
|                 agent_name=agent_name, | ||||
|                 content=conversation_summary, | ||||
|                 memory_type="conversation", | ||||
|                 debate_topic=debate_topic, | ||||
|                 metadata={ | ||||
|                     "participants": participants, | ||||
|                     "session_length": len(conversation_history) | ||||
|                 } | ||||
|             ) | ||||
|              | ||||
|             # 保存策略洞察 | ||||
|             if outcomes: | ||||
|                 strategy_insight = self._extract_strategy_insight( | ||||
|                     outcomes, agent_name | ||||
|                 ) | ||||
|                 if strategy_insight: | ||||
|                     await self.add_memory( | ||||
|                         agent_name=agent_name, | ||||
|                         content=strategy_insight, | ||||
|                         memory_type="strategy", | ||||
|                         debate_topic=debate_topic, | ||||
|                         metadata={"session_outcome": outcomes} | ||||
|                     ) | ||||
|      | ||||
|     def _summarize_conversation(self,  | ||||
|                                conversation_history: List[Dict[str, str]],  | ||||
|                                agent_name: str) -> str: | ||||
|         """ | ||||
|         为特定智能体总结对话历史 | ||||
|          | ||||
|         Args: | ||||
|             conversation_history: 对话历史 | ||||
|             agent_name: 智能体名称 | ||||
|              | ||||
|         Returns: | ||||
|             对话总结 | ||||
|         """ | ||||
|         agent_messages = [ | ||||
|             msg for msg in conversation_history  | ||||
|             if msg.get("agent") == agent_name | ||||
|         ] | ||||
|          | ||||
|         if not agent_messages: | ||||
|             return "本次辩论中未发言" | ||||
|          | ||||
|         chinese_name = self.baxian_agents.get(agent_name, agent_name) | ||||
|         summary = f"{chinese_name}在本次辩论中的主要观点:\n" | ||||
|          | ||||
|         for i, msg in enumerate(agent_messages[:3], 1):  # 只取前3条主要观点 | ||||
|             summary += f"{i}. {msg.get('content', '')[:100]}...\n" | ||||
|          | ||||
|         return summary | ||||
|      | ||||
|     def _extract_strategy_insight(self,  | ||||
|                                  outcomes: Dict[str, Any],  | ||||
|                                  agent_name: str) -> Optional[str]: | ||||
|         """ | ||||
|         从辩论结果中提取策略洞察 | ||||
|          | ||||
|         Args: | ||||
|             outcomes: 辩论结果 | ||||
|             agent_name: 智能体名称 | ||||
|              | ||||
|         Returns: | ||||
|             策略洞察或None | ||||
|         """ | ||||
|         # 这里可以根据实际的outcomes结构来提取洞察 | ||||
|         # 暂时返回一个简单的示例 | ||||
|         chinese_name = self.baxian_agents.get(agent_name, agent_name) | ||||
|          | ||||
|         if "winner" in outcomes and outcomes["winner"] == agent_name: | ||||
|             return f"{chinese_name}在本次辩论中获胜,其论证策略值得保持。" | ||||
|         elif "insights" in outcomes and agent_name in outcomes["insights"]: | ||||
|             return outcomes["insights"][agent_name] | ||||
|          | ||||
|         return None | ||||
| 
 | ||||
| 
 | ||||
| # 便捷函数 | ||||
| async def initialize_baxian_memory_banks(project_id: str, location: str = "us-central1") -> VertexMemoryBank: | ||||
|     """ | ||||
|     初始化所有八仙智能体的记忆银行 | ||||
|      | ||||
|     Args: | ||||
|         project_id: Google Cloud项目ID | ||||
|         location: 部署区域 | ||||
|          | ||||
|     Returns: | ||||
|         配置好的VertexMemoryBank实例 | ||||
|     """ | ||||
|     memory_bank = VertexMemoryBank(project_id, location) | ||||
|      | ||||
|     print("🏛️ 正在为稷下学宫八仙创建记忆银行...") | ||||
|      | ||||
|     for agent_key, chinese_name in memory_bank.baxian_agents.items(): | ||||
|         try: | ||||
|             await memory_bank.create_memory_bank(agent_key) | ||||
|         except Exception as e: | ||||
|             print(f"⚠️ 创建 {chinese_name} 记忆银行时出错: {e}") | ||||
|      | ||||
|     print("✅ 八仙记忆银行初始化完成") | ||||
|     return memory_bank | ||||
| 
 | ||||
| 
 | ||||
| if __name__ == "__main__": | ||||
|     import asyncio | ||||
|      | ||||
|     async def test_memory_bank(): | ||||
|         """测试Memory Bank功能""" | ||||
|         try: | ||||
|             # 从配置创建Memory Bank | ||||
|             memory_bank = VertexMemoryBank.from_config() | ||||
|              | ||||
|             # 测试创建记忆银行 | ||||
|             await memory_bank.create_memory_bank("tieguaili") | ||||
|              | ||||
|             # 测试添加记忆 | ||||
|             await memory_bank.add_memory( | ||||
|                 agent_name="tieguaili", | ||||
|                 content="在讨论NVIDIA股票时,我倾向于逆向思维,关注潜在风险。", | ||||
|                 memory_type="preference", | ||||
|                 debate_topic="NVIDIA投资分析" | ||||
|             ) | ||||
|              | ||||
|             # 测试搜索记忆 | ||||
|             results = await memory_bank.search_memories( | ||||
|                 agent_name="tieguaili", | ||||
|                 query="NVIDIA", | ||||
|                 limit=5 | ||||
|             ) | ||||
|              | ||||
|             print(f"搜索结果: {len(results)} 条记忆") | ||||
|             for result in results: | ||||
|                 print(f"- {result['content']}") | ||||
|              | ||||
|         except Exception as e: | ||||
|             print(f"❌ 测试失败: {e}") | ||||
|      | ||||
|     # 运行测试 | ||||
|     asyncio.run(test_memory_bank()) | ||||
|  | @ -0,0 +1,299 @@ | |||
| // API 测试示例脚本
 | ||||
| // 演示如何使用 Hyperdrive API 进行 CRUD 操作
 | ||||
| 
 | ||||
| const API_BASE_URL = 'https://hyperdrive-neondb-test.<your-subdomain>.workers.dev'; | ||||
| const API_KEY = 'your-api-key'; // 可选,如果设置了 API_SECRET
 | ||||
| 
 | ||||
| // 通用请求函数
 | ||||
| async function apiRequest(endpoint, options = {}) { | ||||
|   const url = `${API_BASE_URL}${endpoint}`; | ||||
|   const headers = { | ||||
|     'Content-Type': 'application/json', | ||||
|     ...(API_KEY && { 'X-API-Key': API_KEY }), | ||||
|     ...options.headers | ||||
|   }; | ||||
|    | ||||
|   try { | ||||
|     const response = await fetch(url, { | ||||
|       ...options, | ||||
|       headers | ||||
|     }); | ||||
|      | ||||
|     const data = await response.json(); | ||||
|      | ||||
|     if (!response.ok) { | ||||
|       throw new Error(`API Error: ${data.message || response.statusText}`); | ||||
|     } | ||||
|      | ||||
|     return data; | ||||
|   } catch (error) { | ||||
|     console.error(`Request failed for ${endpoint}:`, error.message); | ||||
|     throw error; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| // API 测试函数
 | ||||
| class ApiTester { | ||||
|   static async testHealthCheck() { | ||||
|     console.log('\n🏥 Testing health check...'); | ||||
|     try { | ||||
|       const result = await apiRequest('/health'); | ||||
|       console.log('✅ Health check passed:', result.data); | ||||
|       return true; | ||||
|     } catch (error) { | ||||
|       console.log('❌ Health check failed:', error.message); | ||||
|       return false; | ||||
|     } | ||||
|   } | ||||
|    | ||||
|   static async initializeDatabase() { | ||||
|     console.log('\n🗄️ Initializing database...'); | ||||
|     try { | ||||
|       const result = await apiRequest('/init', { method: 'POST' }); | ||||
|       console.log('✅ Database initialized:', result.message); | ||||
|       return true; | ||||
|     } catch (error) { | ||||
|       console.log('❌ Database initialization failed:', error.message); | ||||
|       return false; | ||||
|     } | ||||
|   } | ||||
|    | ||||
|   static async createUser(name, email) { | ||||
|     console.log(`\n👤 Creating user: ${name} (${email})...`); | ||||
|     try { | ||||
|       const result = await apiRequest('/users', { | ||||
|         method: 'POST', | ||||
|         body: JSON.stringify({ name, email }) | ||||
|       }); | ||||
|       console.log('✅ User created:', result.data); | ||||
|       return result.data; | ||||
|     } catch (error) { | ||||
|       console.log('❌ User creation failed:', error.message); | ||||
|       return null; | ||||
|     } | ||||
|   } | ||||
|    | ||||
|   static async getUsers(page = 1, limit = 10, search = null) { | ||||
|     console.log(`\n📋 Getting users (page ${page}, limit ${limit}${search ? `, search: ${search}` : ''})...`); | ||||
|     try { | ||||
|       let endpoint = `/users?page=${page}&limit=${limit}`; | ||||
|       if (search) endpoint += `&search=${encodeURIComponent(search)}`; | ||||
|        | ||||
|       const result = await apiRequest(endpoint); | ||||
|       console.log('✅ Users retrieved:', { | ||||
|         count: result.data.length, | ||||
|         total: result.meta.total, | ||||
|         users: result.data.map(u => `${u.name} (${u.email})`) | ||||
|       }); | ||||
|       return result; | ||||
|     } catch (error) { | ||||
|       console.log('❌ Failed to get users:', error.message); | ||||
|       return null; | ||||
|     } | ||||
|   } | ||||
|    | ||||
|   static async getUserById(id) { | ||||
|     console.log(`\n🔍 Getting user by ID: ${id}...`); | ||||
|     try { | ||||
|       const result = await apiRequest(`/users/${id}`); | ||||
|       console.log('✅ User found:', result.data); | ||||
|       return result.data; | ||||
|     } catch (error) { | ||||
|       console.log('❌ Failed to get user:', error.message); | ||||
|       return null; | ||||
|     } | ||||
|   } | ||||
|    | ||||
|   static async updateUser(id, updates) { | ||||
|     console.log(`\n✏️ Updating user ${id}:`, updates); | ||||
|     try { | ||||
|       const result = await apiRequest(`/users/${id}`, { | ||||
|         method: 'PUT', | ||||
|         body: JSON.stringify(updates) | ||||
|       }); | ||||
|       console.log('✅ User updated:', result.data); | ||||
|       return result.data; | ||||
|     } catch (error) { | ||||
|       console.log('❌ Failed to update user:', error.message); | ||||
|       return null; | ||||
|     } | ||||
|   } | ||||
|    | ||||
|   static async deleteUser(id) { | ||||
|     console.log(`\n🗑️ Deleting user ${id}...`); | ||||
|     try { | ||||
|       const result = await apiRequest(`/users/${id}`, { method: 'DELETE' }); | ||||
|       console.log('✅ User deleted:', result.message); | ||||
|       return true; | ||||
|     } catch (error) { | ||||
|       console.log('❌ Failed to delete user:', error.message); | ||||
|       return false; | ||||
|     } | ||||
|   } | ||||
|    | ||||
|   static async getApiDocs() { | ||||
|     console.log('\n📚 Getting API documentation...'); | ||||
|     try { | ||||
|       const result = await apiRequest('/docs'); | ||||
|       console.log('✅ API Documentation:'); | ||||
|       console.log('Endpoints:', result.data.endpoints); | ||||
|       console.log('Authentication:', result.data.authentication); | ||||
|       console.log('Examples:', result.data.examples); | ||||
|       return result.data; | ||||
|     } catch (error) { | ||||
|       console.log('❌ Failed to get API docs:', error.message); | ||||
|       return null; | ||||
|     } | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| // 完整的测试流程
 | ||||
| async function runFullTest() { | ||||
|   console.log('🚀 Starting Hyperdrive API Test Suite'); | ||||
|   console.log('====================================='); | ||||
|    | ||||
|   // 1. 健康检查
 | ||||
|   const healthOk = await ApiTester.testHealthCheck(); | ||||
|   if (!healthOk) { | ||||
|     console.log('\n❌ Health check failed. Please check your deployment.'); | ||||
|     return; | ||||
|   } | ||||
|    | ||||
|   // 2. 获取 API 文档
 | ||||
|   await ApiTester.getApiDocs(); | ||||
|    | ||||
|   // 3. 初始化数据库
 | ||||
|   await ApiTester.initializeDatabase(); | ||||
|    | ||||
|   // 4. 创建测试用户
 | ||||
|   const user1 = await ApiTester.createUser('张三', 'zhangsan@example.com'); | ||||
|   const user2 = await ApiTester.createUser('李四', 'lisi@example.com'); | ||||
|   const user3 = await ApiTester.createUser('王五', 'wangwu@example.com'); | ||||
|    | ||||
|   if (!user1 || !user2 || !user3) { | ||||
|     console.log('\n❌ Failed to create test users.'); | ||||
|     return; | ||||
|   } | ||||
|    | ||||
|   // 5. 获取用户列表
 | ||||
|   await ApiTester.getUsers(); | ||||
|    | ||||
|   // 6. 搜索用户
 | ||||
|   await ApiTester.getUsers(1, 10, '张'); | ||||
|    | ||||
|   // 7. 获取单个用户
 | ||||
|   await ApiTester.getUserById(user1.id); | ||||
|    | ||||
|   // 8. 更新用户
 | ||||
|   await ApiTester.updateUser(user1.id, { | ||||
|     name: '张三丰', | ||||
|     email: 'zhangsanfeng@example.com' | ||||
|   }); | ||||
|    | ||||
|   // 9. 验证更新
 | ||||
|   await ApiTester.getUserById(user1.id); | ||||
|    | ||||
|   // 10. 分页测试
 | ||||
|   await ApiTester.getUsers(1, 2); // 第一页,每页2条
 | ||||
|   await ApiTester.getUsers(2, 2); // 第二页,每页2条
 | ||||
|    | ||||
|   // 11. 删除用户
 | ||||
|   await ApiTester.deleteUser(user3.id); | ||||
|    | ||||
|   // 12. 验证删除
 | ||||
|   await ApiTester.getUserById(user3.id); // 应该返回 404
 | ||||
|    | ||||
|   // 13. 最终用户列表
 | ||||
|   await ApiTester.getUsers(); | ||||
|    | ||||
|   console.log('\n🎉 API Test Suite Completed!'); | ||||
|   console.log('============================'); | ||||
| } | ||||
| 
 | ||||
| // 性能测试
 | ||||
| async function performanceTest() { | ||||
|   console.log('\n⚡ Performance Test'); | ||||
|   console.log('=================='); | ||||
|    | ||||
|   const startTime = Date.now(); | ||||
|   const promises = []; | ||||
|    | ||||
|   // 并发创建10个用户
 | ||||
|   for (let i = 0; i < 10; i++) { | ||||
|     promises.push( | ||||
|       ApiTester.createUser(`测试用户${i}`, `test${i}@example.com`) | ||||
|     ); | ||||
|   } | ||||
|    | ||||
|   try { | ||||
|     const results = await Promise.all(promises); | ||||
|     const endTime = Date.now(); | ||||
|     const duration = endTime - startTime; | ||||
|      | ||||
|     console.log(`✅ Created ${results.filter(r => r).length} users in ${duration}ms`); | ||||
|     console.log(`📊 Average: ${(duration / 10).toFixed(2)}ms per user`); | ||||
|      | ||||
|     // 清理测试数据
 | ||||
|     console.log('\n🧹 Cleaning up test data...'); | ||||
|     for (const user of results.filter(r => r)) { | ||||
|       await ApiTester.deleteUser(user.id); | ||||
|     } | ||||
|      | ||||
|   } catch (error) { | ||||
|     console.log('❌ Performance test failed:', error.message); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| // 错误处理测试
 | ||||
| async function errorHandlingTest() { | ||||
|   console.log('\n🚨 Error Handling Test'); | ||||
|   console.log('======================'); | ||||
|    | ||||
|   // 测试无效数据
 | ||||
|   console.log('\n Testing invalid user data...'); | ||||
|   await ApiTester.createUser('', 'invalid-email'); // 应该失败
 | ||||
|    | ||||
|   // 测试不存在的用户
 | ||||
|   console.log('\n Testing non-existent user...'); | ||||
|   await ApiTester.getUserById(99999); // 应该返回 404
 | ||||
|    | ||||
|   // 测试无效的更新
 | ||||
|   console.log('\n Testing invalid update...'); | ||||
|   await ApiTester.updateUser(99999, { name: 'Test' }); // 应该返回 404
 | ||||
| } | ||||
| 
 | ||||
| // 主函数
 | ||||
| async function main() { | ||||
|   console.log('请确保已经部署了 Worker 并更新了 API_BASE_URL'); | ||||
|   console.log('如果设置了 API_SECRET,请更新 API_KEY 变量\n'); | ||||
|    | ||||
|   try { | ||||
|     await runFullTest(); | ||||
|     await performanceTest(); | ||||
|     await errorHandlingTest(); | ||||
|   } catch (error) { | ||||
|     console.error('\n💥 Test suite failed:', error.message); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| // 如果直接运行此脚本
 | ||||
| if (typeof window === 'undefined') { | ||||
|   // Node.js 环境
 | ||||
|   const fetch = require('node-fetch'); | ||||
|   global.fetch = fetch; | ||||
|   main(); | ||||
| } else { | ||||
|   // 浏览器环境
 | ||||
|   console.log('在浏览器控制台中运行: main()'); | ||||
| } | ||||
| 
 | ||||
| // 导出函数供其他模块使用
 | ||||
| if (typeof module !== 'undefined' && module.exports) { | ||||
|   module.exports = { | ||||
|     ApiTester, | ||||
|     runFullTest, | ||||
|     performanceTest, | ||||
|     errorHandlingTest, | ||||
|     main | ||||
|   }; | ||||
| } | ||||
|  | @ -0,0 +1,79 @@ | |||
| // Test script to validate Hyperdrive configuration for remote deployment
 | ||||
| // This script helps test the configuration without local PostgreSQL
 | ||||
| 
 | ||||
| const { execSync } = require('child_process'); | ||||
| const fs = require('fs'); | ||||
| 
 | ||||
| console.log('🚀 Testing Hyperdrive Configuration for Remote Deployment'); | ||||
| console.log('======================================================'); | ||||
| 
 | ||||
| try { | ||||
|   // Check wrangler configuration
 | ||||
|   console.log('\n📋 Validating wrangler.toml...'); | ||||
|    | ||||
|   const wranglerContent = fs.readFileSync('wrangler.toml', 'utf8'); | ||||
|   console.log('✅ wrangler.toml loaded successfully'); | ||||
|    | ||||
|   // Validate configuration syntax
 | ||||
|   try { | ||||
|     execSync('wrangler config validate', { stdio: 'pipe' }); | ||||
|     console.log('✅ wrangler.toml syntax is valid'); | ||||
|   } catch (error) { | ||||
|     console.log('⚠️  Configuration validation warning (this is normal for Hyperdrive)'); | ||||
|   } | ||||
|    | ||||
|   // Check if we can authenticate with Cloudflare
 | ||||
|   console.log('\n🔐 Checking Cloudflare authentication...'); | ||||
|   try { | ||||
|     const whoami = execSync('wrangler whoami', { encoding: 'utf8' }); | ||||
|     console.log('✅ Authenticated with Cloudflare'); | ||||
|     console.log(`   ${whoami.trim()}`); | ||||
|   } catch (error) { | ||||
|     console.log('❌ Not authenticated with Cloudflare'); | ||||
|     console.log('   Run: wrangler login'); | ||||
|     return; | ||||
|   } | ||||
|    | ||||
|   // List Hyperdrive configurations
 | ||||
|   console.log('\n🔗 Checking Hyperdrive configurations...'); | ||||
|   try { | ||||
|     const hyperdrives = execSync('wrangler hyperdrive list', { encoding: 'utf8' }); | ||||
|     console.log('✅ Hyperdrive configurations:'); | ||||
|     console.log(hyperdrives); | ||||
|   } catch (error) { | ||||
|     console.log('⚠️  Could not list Hyperdrive configurations'); | ||||
|     console.log('   Error:', error.message); | ||||
|   } | ||||
|    | ||||
|   // Check specific Hyperdrive
 | ||||
|   console.log('\n🎯 Checking specific Hyperdrive ID...'); | ||||
|   try { | ||||
|     const hyperdriveInfo = execSync('wrangler hyperdrive get ef43924d89064cddabfaccf06aadfab6', { encoding: 'utf8' }); | ||||
|     console.log('✅ Hyperdrive configuration found:'); | ||||
|     console.log(hyperdriveInfo); | ||||
|   } catch (error) { | ||||
|     console.log('❌ Could not find Hyperdrive configuration'); | ||||
|     console.log('   Error:', error.message); | ||||
|     console.log('   Make sure the Hyperdrive ID is correct and exists in your account'); | ||||
|   } | ||||
|    | ||||
|   console.log('\n📝 Configuration Summary:'); | ||||
|   console.log('   - Worker Name: hyperdrive-neondb-test'); | ||||
|   console.log('   - Hyperdrive ID: ef43924d89064cddabfaccf06aadfab6'); | ||||
|   console.log('   - Binding: HYPERDRIVE'); | ||||
|   console.log('   - Database Type: NeonDB (PostgreSQL)'); | ||||
|    | ||||
|   console.log('\n🚀 Deployment Commands:'); | ||||
|   console.log('   1. Deploy to production: wrangler deploy'); | ||||
|   console.log('   2. Test endpoints after deployment:'); | ||||
|   console.log('      - https://hyperdrive-neondb-test.<your-subdomain>.workers.dev/test-connection'); | ||||
|   console.log('      - https://hyperdrive-neondb-test.<your-subdomain>.workers.dev/test-query'); | ||||
|    | ||||
|   console.log('\n💡 Tips:'); | ||||
|   console.log('   - Hyperdrive provides connection pooling and caching for your database'); | ||||
|   console.log('   - It reduces latency and improves performance for database queries'); | ||||
|   console.log('   - The worker will automatically use the Hyperdrive connection in production'); | ||||
|    | ||||
| } catch (error) { | ||||
|   console.error('❌ Error during testing:', error.message); | ||||
| } | ||||
|  | @ -0,0 +1,93 @@ | |||
| // Simple test script for Hyperdrive configuration
 | ||||
| // This script helps verify the wrangler.toml configuration
 | ||||
| 
 | ||||
| console.log('🚀 Hyperdrive NeonDB Test Configuration'); | ||||
| console.log('====================================='); | ||||
| 
 | ||||
| // Check if wrangler.toml exists and has correct configuration
 | ||||
| const fs = require('fs'); | ||||
| const path = require('path'); | ||||
| 
 | ||||
| try { | ||||
|   const wranglerPath = path.join(__dirname, 'wrangler.toml'); | ||||
|    | ||||
|   if (fs.existsSync(wranglerPath)) { | ||||
|     console.log('✅ wrangler.toml found'); | ||||
|      | ||||
|     const content = fs.readFileSync(wranglerPath, 'utf8'); | ||||
|      | ||||
|     // Check for Hyperdrive configuration
 | ||||
|     if (content.includes('hyperdrive')) { | ||||
|       console.log('✅ Hyperdrive configuration found'); | ||||
|     } else { | ||||
|       console.log('❌ Hyperdrive configuration missing'); | ||||
|     } | ||||
|      | ||||
|     // Check for binding
 | ||||
|     if (content.includes('binding = "HYPERDRIVE"')) { | ||||
|       console.log('✅ HYPERDRIVE binding configured'); | ||||
|     } else { | ||||
|       console.log('❌ HYPERDRIVE binding missing'); | ||||
|     } | ||||
|      | ||||
|     // Check for Hyperdrive ID
 | ||||
|     if (content.includes('ef43924d89064cddabfaccf06aadfab6')) { | ||||
|       console.log('✅ Hyperdrive ID configured'); | ||||
|     } else { | ||||
|       console.log('❌ Hyperdrive ID missing'); | ||||
|     } | ||||
|      | ||||
|     // Check for nodejs_compat
 | ||||
|     if (content.includes('nodejs_compat')) { | ||||
|       console.log('✅ nodejs_compat flag enabled'); | ||||
|     } else { | ||||
|       console.log('❌ nodejs_compat flag missing'); | ||||
|     } | ||||
|      | ||||
|   } else { | ||||
|     console.log('❌ wrangler.toml not found'); | ||||
|   } | ||||
|    | ||||
|   // Check if src/index.ts exists
 | ||||
|   const indexPath = path.join(__dirname, 'src', 'index.ts'); | ||||
|   if (fs.existsSync(indexPath)) { | ||||
|     console.log('✅ src/index.ts found'); | ||||
|   } else { | ||||
|     console.log('❌ src/index.ts missing'); | ||||
|   } | ||||
|    | ||||
|   // Check if package.json exists
 | ||||
|   const packagePath = path.join(__dirname, 'package.json'); | ||||
|   if (fs.existsSync(packagePath)) { | ||||
|     console.log('✅ package.json found'); | ||||
|      | ||||
|     const packageContent = JSON.parse(fs.readFileSync(packagePath, 'utf8')); | ||||
|      | ||||
|     // Check for required dependencies
 | ||||
|     if (packageContent.dependencies && packageContent.dependencies.pg) { | ||||
|       console.log('✅ pg dependency configured'); | ||||
|     } else { | ||||
|       console.log('❌ pg dependency missing'); | ||||
|     } | ||||
|      | ||||
|     if (packageContent.devDependencies && packageContent.devDependencies['@cloudflare/workers-types']) { | ||||
|       console.log('✅ Cloudflare Workers types configured'); | ||||
|     } else { | ||||
|       console.log('❌ Cloudflare Workers types missing'); | ||||
|     } | ||||
|   } | ||||
|    | ||||
|   console.log('\n📋 Next Steps:'); | ||||
|   console.log('1. Run: wrangler dev --local (for local testing)'); | ||||
|   console.log('2. Run: wrangler dev (for remote testing with Hyperdrive)'); | ||||
|   console.log('3. Test endpoints:'); | ||||
|   console.log('   - http://localhost:8787/test-connection'); | ||||
|   console.log('   - http://localhost:8787/test-query'); | ||||
|   console.log('\n🔧 Hyperdrive Configuration:'); | ||||
|   console.log('   - Hyperdrive ID: ef43924d89064cddabfaccf06aadfab6'); | ||||
|   console.log('   - Binding: HYPERDRIVE'); | ||||
|   console.log('   - Database: NeonDB (PostgreSQL)'); | ||||
|    | ||||
| } catch (error) { | ||||
|   console.error('❌ Error checking configuration:', error.message); | ||||
| } | ||||
|  | @ -0,0 +1,67 @@ | |||
| #!/bin/bash | ||||
| 
 | ||||
| # 术数书 API 测试脚本 | ||||
| # 使用 curl 命令测试 Hyperdrive + NeonDB 术数书查询系统 | ||||
| 
 | ||||
| API_BASE="https://hyperdrive.seekkey.tech" | ||||
| 
 | ||||
| echo "🚀 术数书 Hyperdrive API 测试" | ||||
| echo "================================" | ||||
| 
 | ||||
| # 测试连接 | ||||
| echo "" | ||||
| echo "🔗 测试数据库连接..." | ||||
| curl -s "$API_BASE/test-connection" | jq -r '.message // .error // "连接测试完成"' | ||||
| 
 | ||||
| # 查询表结构 | ||||
| echo "" | ||||
| echo "📋 查询数据库表结构..." | ||||
| echo "发现的表:" | ||||
| curl -s "$API_BASE/query-tables" | jq -r '.tables[]? | "- \(.table_name) (\(.table_schema))"' | ||||
| 
 | ||||
| # 获取术数书统计 | ||||
| echo "" | ||||
| echo "📊 术数书统计信息..." | ||||
| curl -s "$API_BASE/shushu-stats" | jq -r '.existing_tables[]? | "- \(.table_name): \(.record_count) 条记录"' | ||||
| 
 | ||||
| # 查询术数书内容 | ||||
| echo "" | ||||
| echo "📚 查询术数书内容 (前3条)..." | ||||
| echo "书籍信息:" | ||||
| curl -s "$API_BASE/query-shushu?limit=3" | jq -r '.data[]? | "- ID: \(.id), 标题: \(.title), 作者: \(.author), 类别: \(.category)"' | ||||
| 
 | ||||
| # 搜索功能测试 | ||||
| echo "" | ||||
| echo "🔍 搜索测试..." | ||||
| echo "搜索关键词: 易经" | ||||
| curl -s "$API_BASE/search-shushu?q=易经&limit=2" | jq -r '.total_matches // 0 | "找到 \(.) 条匹配记录"' | ||||
| 
 | ||||
| echo "搜索关键词: 八卦" | ||||
| curl -s "$API_BASE/search-shushu?q=八卦&limit=2" | jq -r '.total_matches // 0 | "找到 \(.) 条匹配记录"' | ||||
| 
 | ||||
| echo "搜索关键词: 面相" | ||||
| curl -s "$API_BASE/search-shushu?q=面相&limit=2" | jq -r '.total_matches // 0 | "找到 \(.) 条匹配记录"' | ||||
| 
 | ||||
| # 性能测试 | ||||
| echo "" | ||||
| echo "⚡ 性能测试..." | ||||
| echo "测试查询响应时间:" | ||||
| time curl -s "$API_BASE/query-shushu?limit=1" > /dev/null | ||||
| 
 | ||||
| echo "" | ||||
| echo "✅ API 测试完成!" | ||||
| echo "" | ||||
| echo "📖 可用端点:" | ||||
| echo "- GET $API_BASE/ - 系统信息" | ||||
| echo "- GET $API_BASE/test-connection - 测试连接" | ||||
| echo "- GET $API_BASE/query-tables - 查询表结构" | ||||
| echo "- GET $API_BASE/query-shushu?limit=N - 查询术数书" | ||||
| echo "- GET $API_BASE/search-shushu?q=keyword&limit=N - 搜索术数书" | ||||
| echo "- GET $API_BASE/shushu-stats - 统计信息" | ||||
| echo "" | ||||
| echo "🎯 与 AutoRAG 对比优势:" | ||||
| echo "- ✅ 毫秒级响应 (Hyperdrive 边缘缓存)" | ||||
| echo "- ✅ 精确查询 (SQL vs 向量相似性)" | ||||
| echo "- ✅ 实时数据 (直连数据库)" | ||||
| echo "- ✅ 成本优化 (连接池 + 缓存)" | ||||
| echo "- ✅ 全球分布 (Cloudflare 边缘网络)" | ||||
Some files were not shown because too many files have changed in this diff Show More
		Loading…
	
		Reference in New Issue