|
@@ -0,0 +1,725 @@
|
|
|
|
|
+{
|
|
|
|
|
+ "cells": [
|
|
|
|
|
+ {
|
|
|
|
|
+ "cell_type": "markdown",
|
|
|
|
|
+ "id": "38b007c8",
|
|
|
|
|
+ "metadata": {},
|
|
|
|
|
+ "source": [
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ "## GiftGeniusAgent——你的送礼智能Agent\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ "### 项目简介\n",
|
|
|
|
|
+ "本项目演示一个基于HelloAgents框架的智能送礼Agent\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ "### 作者信息\n",
|
|
|
|
|
+ "- 姓名:张善祺\n",
|
|
|
|
|
+ "- GitHub:@jack6249\n",
|
|
|
|
|
+ "- 日期:2025-11-21\n"
|
|
|
|
|
+ ]
|
|
|
|
|
+ },
|
|
|
|
|
+ {
|
|
|
|
|
+ "cell_type": "markdown",
|
|
|
|
|
+ "id": "62a3b6a3",
|
|
|
|
|
+ "metadata": {},
|
|
|
|
|
+ "source": [
|
|
|
|
|
+ "### 第1部分:环境配置"
|
|
|
|
|
+ ]
|
|
|
|
|
+ },
|
|
|
|
|
+ {
|
|
|
|
|
+ "cell_type": "code",
|
|
|
|
|
+ "execution_count": null,
|
|
|
|
|
+ "id": "dd9b8a20",
|
|
|
|
|
+ "metadata": {},
|
|
|
|
|
+ "outputs": [],
|
|
|
|
|
+ "source": [
|
|
|
|
|
+ "#导入库和参数配置\n",
|
|
|
|
|
+ "from hello_agents import SimpleAgent, HelloAgentsLLM, ReflectionAgent, ToolRegistry\n",
|
|
|
|
|
+ "from hello_agents.tools import Tool, ToolParameter\n",
|
|
|
|
|
+ "from typing import Dict, Any, List\n",
|
|
|
|
|
+ "from tavily import TavilyClient\n",
|
|
|
|
|
+ "import os\n",
|
|
|
|
|
+ "import json\n",
|
|
|
|
|
+ "import re\n",
|
|
|
|
|
+ "import numpy as np \n",
|
|
|
|
|
+ "from dotenv import load_dotenv\n",
|
|
|
|
|
+ "import asyncio\n",
|
|
|
|
|
+ "import nest_asyncio\n",
|
|
|
|
|
+ "from mcp.client.sse import sse_client\n",
|
|
|
|
|
+ "from mcp.client.session import ClientSession\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ "load_dotenv()\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ "#LLM参数\n",
|
|
|
|
|
+ "LLM_MODEL_ID = os.getenv(\"LLM_MODEL_ID\")\n",
|
|
|
|
|
+ "LLM_API_KEY = os.getenv(\"LLM_API_KEY\")\n",
|
|
|
|
|
+ "LLM_BASE_URL = os.getenv(\"LLM_BASE_URL\")\n",
|
|
|
|
|
+ "#Tavily参数\n",
|
|
|
|
|
+ "TAVILY_API_KEY = os.getenv(\"TAVILY_API_KEY\",\"\")\n",
|
|
|
|
|
+ "#百度MCP参数\n",
|
|
|
|
|
+ "BAIDU_TOKEN = os.getenv(\"BAIDU_MCP_TOKEN\",\"\")\n",
|
|
|
|
|
+ "#输入json路径配置\n",
|
|
|
|
|
+ "INPUT_FILENAME = \"data/test_cases.json\"\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ "# 搜索源配置\n",
|
|
|
|
|
+ "# 可选值: \"tavily\" (通用/海外) 或 \"baidu\" (电商/国内)\n",
|
|
|
|
|
+ "os.environ[\"SEARCH_PROVIDER\"] = \"baidu\" \n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ "print(\"✅ 环境配置完成\")\n"
|
|
|
|
|
+ ]
|
|
|
|
|
+ },
|
|
|
|
|
+ {
|
|
|
|
|
+ "cell_type": "markdown",
|
|
|
|
|
+ "id": "9404c299",
|
|
|
|
|
+ "metadata": {},
|
|
|
|
|
+ "source": [
|
|
|
|
|
+ "### 第2部分:定义工具"
|
|
|
|
|
+ ]
|
|
|
|
|
+ },
|
|
|
|
|
+ {
|
|
|
|
|
+ "cell_type": "code",
|
|
|
|
|
+ "execution_count": null,
|
|
|
|
|
+ "id": "37b994bc",
|
|
|
|
|
+ "metadata": {},
|
|
|
|
|
+ "outputs": [],
|
|
|
|
|
+ "source": [
|
|
|
|
|
+ "# [Cell 2 终极版] 定义统一搜索工具 (兼容 Tavily 和 Baidu)\n",
|
|
|
|
|
+ "# 允许 Jupyter 运行异步\n",
|
|
|
|
|
+ "nest_asyncio.apply()\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ "class BatchSearchTool(Tool):\n",
|
|
|
|
|
+ " def __init__(self):\n",
|
|
|
|
|
+ " super().__init__(\n",
|
|
|
|
|
+ " name=\"batch_search\",\n",
|
|
|
|
|
+ " description=\"统一搜索工具,支持 Tavily 和 Baidu 切换。\"\n",
|
|
|
|
|
+ " )\n",
|
|
|
|
|
+ " self.provider = os.environ.get(\"SEARCH_PROVIDER\", \"tavily\").lower()\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ " def run(self, parameters: Any) -> str:\n",
|
|
|
|
|
+ " return \"请使用 Python 代码直接调用 search_raw 方法获取数据。\"\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ " def search_raw(self, query: str) -> List[Dict]:\n",
|
|
|
|
|
+ " if self.provider == \"baidu\":\n",
|
|
|
|
|
+ " return self._search_baidu(query)\n",
|
|
|
|
|
+ " else:\n",
|
|
|
|
|
+ " return self._search_tavily(query)\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ " # --- 引擎 A: Tavily ---\n",
|
|
|
|
|
+ " def _search_tavily(self, query: str) -> List[Dict]:\n",
|
|
|
|
|
+ " api_key = os.environ.get(\"TAVILY_API_KEY\")\n",
|
|
|
|
|
+ " if not api_key: return []\n",
|
|
|
|
|
+ " print(f\" 🚀 [Tavily] 正在搜索: {query} ...\")\n",
|
|
|
|
|
+ " try:\n",
|
|
|
|
|
+ " tavily = TavilyClient(api_key=api_key)\n",
|
|
|
|
|
+ " response = tavily.search(query, max_results=5, include_images=True)\n",
|
|
|
|
|
+ " results = []\n",
|
|
|
|
|
+ " if 'results' in response:\n",
|
|
|
|
|
+ " for r in response['results']:\n",
|
|
|
|
|
+ " results.append({\n",
|
|
|
|
|
+ " \"title\": r['title'], \"url\": r['url'], \"content\": r['content'], \n",
|
|
|
|
|
+ " \"type\": \"text\", \"img\": \"\" # Tavily 文本通常不带图\n",
|
|
|
|
|
+ " })\n",
|
|
|
|
|
+ " if 'images' in response and response['images']:\n",
|
|
|
|
|
+ " results.append({\"images\": response['images'][:3], \"type\": \"image\"})\n",
|
|
|
|
|
+ " return results\n",
|
|
|
|
|
+ " except Exception as e:\n",
|
|
|
|
|
+ " print(f\" ⚠️ Tavily 异常: {e}\")\n",
|
|
|
|
|
+ " return []\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ " # --- 引擎 B: Baidu MCP ---\n",
|
|
|
|
|
+ " def _search_baidu(self, query: str) -> List[Dict]:\n",
|
|
|
|
|
+ " token = os.environ.get(\"BAIDU_MCP_TOKEN\")\n",
|
|
|
|
|
+ " if not token: return []\n",
|
|
|
|
|
+ " print(f\" 🐼 [百度优选] 正在搜索: {query} ...\")\n",
|
|
|
|
|
+ " try:\n",
|
|
|
|
|
+ " raw_json_str = asyncio.run(self._async_baidu_call(query, token))\n",
|
|
|
|
|
+ " print(f\" 🔍 原始 JSON 响应: {raw_json_str}\")\n",
|
|
|
|
|
+ " return self._parse_baidu_response(raw_json_str)\n",
|
|
|
|
|
+ " except Exception as e:\n",
|
|
|
|
|
+ " print(f\" ⚠️ 百度 MCP 异常: {e}\")\n",
|
|
|
|
|
+ " return []\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ " async def _async_baidu_call(self, query: str, token: str) -> str:\n",
|
|
|
|
|
+ " sse_url = f\"https://mcp-youxuan.baidu.com/mcp/sse?key={token}\"\n",
|
|
|
|
|
+ " async with sse_client(sse_url) as (read, write):\n",
|
|
|
|
|
+ " async with ClientSession(read, write) as session:\n",
|
|
|
|
|
+ " await session.initialize()\n",
|
|
|
|
|
+ " result = await session.call_tool(\"goods_search\", arguments={\"query\": query})\n",
|
|
|
|
|
+ " return result.content[0].text if result.content else \"\"\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ " def _parse_baidu_response(self, json_str: str) -> List[Dict]:\n",
|
|
|
|
|
+ " results = []; images = []\n",
|
|
|
|
|
+ " try:\n",
|
|
|
|
|
+ " data = json.loads(json_str)\n",
|
|
|
|
|
+ " items = data if isinstance(data, list) else []\n",
|
|
|
|
|
+ " \n",
|
|
|
|
|
+ " for item in items[:5]:\n",
|
|
|
|
|
+ " title = item.get(\"goodsName\") or item.get(\"title\") or \"未知商品\"\n",
|
|
|
|
|
+ " price = item.get(\"price\") or item.get(\"minPrice\") or \"\"\n",
|
|
|
|
|
+ " shop = item.get(\"shopName\") or item.get(\"mall\") or \"\"\n",
|
|
|
|
|
+ " url = item.get(\"detailUrl\") or item.get(\"url\") or item.get(\"ori_url\") or \"#\"\n",
|
|
|
|
|
+ " img = item.get(\"imgUrl\") or item.get(\"picUrl\") or item.get(\"img\")\n",
|
|
|
|
|
+ " \n",
|
|
|
|
|
+ " content = f\"价格: {price}元。店铺: {shop}。商品详情: {title}\"\n",
|
|
|
|
|
+ " \n",
|
|
|
|
|
+ " # 📝【修复点】直接在 text 类型结果里绑定 img\n",
|
|
|
|
|
+ " results.append({\n",
|
|
|
|
|
+ " \"title\": title, \"url\": url, \"content\": content, \n",
|
|
|
|
|
+ " \"type\": \"text\", \"img\": img \n",
|
|
|
|
|
+ " })\n",
|
|
|
|
|
+ " if img: images.append(img)\n",
|
|
|
|
|
+ " \n",
|
|
|
|
|
+ " if images: results.append({\"images\": images[:3], \"type\": \"image\"})\n",
|
|
|
|
|
+ " \n",
|
|
|
|
|
+ " except json.JSONDecodeError:\n",
|
|
|
|
|
+ " print(\" ⚠️ 百度返回非 JSON 数据\")\n",
|
|
|
|
|
+ " return results\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ " def get_parameters(self):\n",
|
|
|
|
|
+ " return [ToolParameter(name=\"query\", type=\"string\", description=\"关键词\")]\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ "tool_registry = ToolRegistry()\n",
|
|
|
|
|
+ "tool_registry.register_tool(BatchSearchTool())\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ "print(\"✅ 统一搜索工具已加载!\")\n",
|
|
|
|
|
+ "print(f\"当前模式: {'百度优选 (电商)' if os.environ.get('SEARCH_PROVIDER') == 'baidu' else 'Tavily (通用)'}\")"
|
|
|
|
|
+ ]
|
|
|
|
|
+ },
|
|
|
|
|
+ {
|
|
|
|
|
+ "cell_type": "markdown",
|
|
|
|
|
+ "id": "e8ffecb0",
|
|
|
|
|
+ "metadata": {},
|
|
|
|
|
+ "source": [
|
|
|
|
|
+ "### 第3部分:创建智能体\n",
|
|
|
|
|
+ "\n"
|
|
|
|
|
+ ]
|
|
|
|
|
+ },
|
|
|
|
|
+ {
|
|
|
|
|
+ "cell_type": "code",
|
|
|
|
|
+ "execution_count": null,
|
|
|
|
|
+ "id": "6c5a2e82",
|
|
|
|
|
+ "metadata": {},
|
|
|
|
|
+ "outputs": [],
|
|
|
|
|
+ "source": [
|
|
|
|
|
+ "# 初始化大模型\n",
|
|
|
|
|
+ "llm = HelloAgentsLLM()\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ "# --- 1. 军师 (Profiler) - 已升级支持多维度画像 ---\n",
|
|
|
|
|
+ "PROFILER_PROMPT = \"\"\"\n",
|
|
|
|
|
+ "你是一个精通 MBTI 人格分析与消费市场趋势的 \"送礼军师\"。\n",
|
|
|
|
|
+ "你的任务是根据用户提供的多维度画像,制定 3 个**极度精准**的搜索关键词。\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ "【⚠️ 时效性死命令 (CRITICAL)】\n",
|
|
|
|
|
+ "当前时间视作 **2025年11月**。\n",
|
|
|
|
|
+ "1. **严禁过时**:绝对不要推荐 2024 年或更早的旧款(除非是经典恒久款如黑胶唱片)。\n",
|
|
|
|
|
+ "2. **价格尺度**:给出的价格单位是人民币元。请严格遵照范围进行联想,禁止超出预算范围。\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ "为了确保推荐质量,请参考以下的【优秀思考范例】:\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ "### 范例 1\n",
|
|
|
|
|
+ "**用户画像**: \n",
|
|
|
|
|
+ "- 女, 26岁, ISFP (探险家), 金牛座\n",
|
|
|
|
|
+ "- 预算: 500-1000元\n",
|
|
|
|
|
+ "- 场景: 情人节\n",
|
|
|
|
|
+ "- 自定义: 喜欢有质感的生活小物\n",
|
|
|
|
|
+ "**军师分析**: \n",
|
|
|
|
|
+ "ISFP 重视审美和感官体验,金牛座喜欢实实在在的质感。情人节需要浪漫。\n",
|
|
|
|
|
+ "**生成策略**:\n",
|
|
|
|
|
+ "1. 观夏 (To Summer) 昆仑煮雪 晶石香薰 (符合质感与审美)\n",
|
|
|
|
|
+ "2. 野兽派 2025 情人节限定 睡衣礼盒 (金牛座喜欢的舒适)\n",
|
|
|
|
|
+ "3. 富士 Instax mini Evo 拍立得 (记录生活瞬间)\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ "### 范例 2\n",
|
|
|
|
|
+ "**用户画像**: \n",
|
|
|
|
|
+ "- 男, 30岁, INTJ (建筑师), 处女座\n",
|
|
|
|
|
+ "- 预算: 1000元以上\n",
|
|
|
|
|
+ "- 场景: 生日\n",
|
|
|
|
|
+ "- 自定义: 程序员,喜欢整洁\n",
|
|
|
|
|
+ "**军师分析**: \n",
|
|
|
|
|
+ "INTJ 追求极致的逻辑和效率,处女座有洁癖,喜欢桌面整洁。\n",
|
|
|
|
|
+ "**生成策略**:\n",
|
|
|
|
|
+ "1. Keychron Q1 Pro 机械键盘 铝坨坨 (符合极客对工具的追求)\n",
|
|
|
|
|
+ "2. 明基 (BenQ) ScreenBar Halo 屏幕挂灯 (极致护眼与桌面美学)\n",
|
|
|
|
|
+ "3. 赫曼米勒 (Herman Miller) 显示器支架 (人体工学)\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ "### 范例 3\n",
|
|
|
|
|
+ "**用户画像**: \n",
|
|
|
|
|
+ "- 女, 20岁, ENFP (竞选者), 狮子座\n",
|
|
|
|
|
+ "- 预算: 300元以内\n",
|
|
|
|
|
+ "- 场景: 圣诞节\n",
|
|
|
|
|
+ "- 自定义: 喜欢二次元,痛包\n",
|
|
|
|
|
+ "**军师分析**: \n",
|
|
|
|
|
+ "ENFP 热情奔放,狮子座喜欢张扬、闪亮的东西。预算有限但要素多。\n",
|
|
|
|
|
+ "**生成策略**:\n",
|
|
|
|
|
+ "1. 泡泡玛特 圣诞系列 盲盒整端 (符合节日气氛和二次元)\n",
|
|
|
|
|
+ "2. WEGO 痛包 镭射款 (符合自定义需求,狮子座喜欢的亮眼)\n",
|
|
|
|
|
+ "3. Chiikawa 吉伊卡哇 圣诞公仔 (当下顶流二次元IP)\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ "---\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ "**现在的任务**:\n",
|
|
|
|
|
+ "请根据以下【当前用户画像】进行分析,模仿上述范例的深度,制定搜索策略。\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ "【当前用户画像】\n",
|
|
|
|
|
+ "{user_profile_text}\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ "【关键词生成要求】\n",
|
|
|
|
|
+ "1. **必须具体**:格式为 `[品牌] + [产品名/系列] + [限定/属性]`。\n",
|
|
|
|
|
+ "2. **拒绝大词**:严禁搜索 \"礼物\"、\"口红\"、\"玩具\" 这种泛词。\n",
|
|
|
|
|
+ "3. **必须包含品牌**:根据预算推断合适的品牌(如:预算低选名创优品/泡泡玛特,预算高选Dior/索尼)。\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ "【输出格式】\n",
|
|
|
|
|
+ "只输出 3 行搜索关键词,每行一个。不要输出分析过程,不要序号。\n",
|
|
|
|
|
+ "\"\"\"\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ "profiler_agent = SimpleAgent(\n",
|
|
|
|
|
+ " llm=llm,\n",
|
|
|
|
|
+ " name=\"Agent_Profiler\",\n",
|
|
|
|
|
+ " system_prompt=PROFILER_PROMPT\n",
|
|
|
|
|
+ ")\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ "# ==============================================================================\n",
|
|
|
|
|
+ "# 2. 种草达人 (Pitcher) - 文案创作 (加入风格指导)\n",
|
|
|
|
|
+ "# ==============================================================================\n",
|
|
|
|
|
+ "PITCHER_PROMPT = \"\"\"\n",
|
|
|
|
|
+ "你是一个 **金牌种草文案**。\n",
|
|
|
|
|
+ "用户会给你一个 **【商品名称】**。\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ "### 🎯 关键要点 (Few Points)\n",
|
|
|
|
|
+ "1. **痛点直击**: 一句话说清楚为什么买它(限定?显白?绝美?)。\n",
|
|
|
|
|
+ "2. **情绪价值**: 使用 \"绝绝子\", \"氛围感\", \"心动\" 等高频热词。\n",
|
|
|
|
|
+ "3. **字数限制**: 严格控制在 **40字以内**,短小精悍。\n",
|
|
|
|
|
+ "4. **Emoji**: 必须包含 1-2 个 emoji。\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ "### 🌟 创作范例 (Few-Shot)\n",
|
|
|
|
|
+ "**输入**: Dior 999 烈艳蓝金\n",
|
|
|
|
|
+ "**输出**: 💄本宫不死终是妃!Dior 999 传奇正红,显白更有气场,送女友绝对没错!\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ "**输入**: 泡泡玛特 Labubu 坐坐派对\n",
|
|
|
|
|
+ "**输出**: ✨太可爱了吧!Labubu 坐坐派对系列,每一个都丑萌到心巴上,摆在桌上超治愈~\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ "**输入**: 罗技 MX Master 3S\n",
|
|
|
|
|
+ "**输出**: 🖱️打工人本命!罗技 Master 3S 静音又顺滑,人体工学设计,手腕再也不累了。\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ "---\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ "**当前任务**:\n",
|
|
|
|
|
+ "请为【{input}】写一句朋友圈风格种草语。\n",
|
|
|
|
|
+ "\"\"\"\n",
|
|
|
|
|
+ "pitcher_agent = SimpleAgent(\n",
|
|
|
|
|
+ " llm=llm,\n",
|
|
|
|
|
+ " name=\"Agent_Pitcher\",\n",
|
|
|
|
|
+ " system_prompt=PITCHER_PROMPT\n",
|
|
|
|
|
+ ")\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ "print(\"✅ 智能体初始化完成!\")"
|
|
|
|
|
+ ]
|
|
|
|
|
+ },
|
|
|
|
|
+ {
|
|
|
|
|
+ "cell_type": "markdown",
|
|
|
|
|
+ "id": "25907c14",
|
|
|
|
|
+ "metadata": {},
|
|
|
|
|
+ "source": [
|
|
|
|
|
+ "### 第4部分:读取数据"
|
|
|
|
|
+ ]
|
|
|
|
|
+ },
|
|
|
|
|
+ {
|
|
|
|
|
+ "cell_type": "code",
|
|
|
|
|
+ "execution_count": null,
|
|
|
|
|
+ "id": "58bd39dd",
|
|
|
|
|
+ "metadata": {},
|
|
|
|
|
+ "outputs": [],
|
|
|
|
|
+ "source": [
|
|
|
|
|
+ "def load_user_profile(filename):\n",
|
|
|
|
|
+ " # 1. 检查文件是否存在\n",
|
|
|
|
|
+ " if not os.path.exists(filename):\n",
|
|
|
|
|
+ " print(f\"⚠️ 未找到配置文件: {filename}\")\n",
|
|
|
|
|
+ " # 如果没有文件,将默认数据写入文件\n",
|
|
|
|
|
+ " default_data = {\n",
|
|
|
|
|
+ " \"性别\": \"女\",\n",
|
|
|
|
|
+ " \"年龄\": \"24岁\",\n",
|
|
|
|
|
+ " \"MBTI\": \"ENFP\",\n",
|
|
|
|
|
+ " \"星座\": \"天秤座\",\n",
|
|
|
|
|
+ " \"预算\": \"500元以内\",\n",
|
|
|
|
|
+ " \"节日\": \"恋爱一周年纪念日\",\n",
|
|
|
|
|
+ " \"自定义\": \"喜欢二次元,平时喜欢喝咖啡,不要送太实用的家电\"\n",
|
|
|
|
|
+ " }\n",
|
|
|
|
|
+ " with open(filename, \"w\", encoding=\"utf-8\") as f:\n",
|
|
|
|
|
+ " json.dump(default_data, f, ensure_ascii=False, indent=4)\n",
|
|
|
|
|
+ " print(f\"✅ 已自动生成默认配置文件,请修改 {filename} 后再次运行。\")\n",
|
|
|
|
|
+ " return default_data\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ " # 2. 读取文件内容\n",
|
|
|
|
|
+ " try:\n",
|
|
|
|
|
+ " with open(filename, \"r\", encoding=\"utf-8\") as f:\n",
|
|
|
|
|
+ " data = json.load(f)\n",
|
|
|
|
|
+ " print(f\"✅ 成功加载用户画像: {filename}\")\n",
|
|
|
|
|
+ " print(f\"📋 内容预览: {json.dumps(data, ensure_ascii=False)}\")\n",
|
|
|
|
|
+ " return data\n",
|
|
|
|
|
+ " except Exception as e:\n",
|
|
|
|
|
+ " print(f\"❌ 读取 JSON 失败: {e}\")\n",
|
|
|
|
|
+ " return {}\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ "# 加载数据\n",
|
|
|
|
|
+ "user_input_data = load_user_profile(INPUT_FILENAME)\n"
|
|
|
|
|
+ ]
|
|
|
|
|
+ },
|
|
|
|
|
+ {
|
|
|
|
|
+ "cell_type": "markdown",
|
|
|
|
|
+ "id": "0fdcab5a",
|
|
|
|
|
+ "metadata": {},
|
|
|
|
|
+ "source": [
|
|
|
|
|
+ "### 第5部分:生成礼物计划"
|
|
|
|
|
+ ]
|
|
|
|
|
+ },
|
|
|
|
|
+ {
|
|
|
|
|
+ "cell_type": "code",
|
|
|
|
|
+ "execution_count": null,
|
|
|
|
|
+ "id": "cdaef6b1",
|
|
|
|
|
+ "metadata": {},
|
|
|
|
|
+ "outputs": [],
|
|
|
|
|
+ "source": [
|
|
|
|
|
+ "def parse_budget_range(budget_str):\n",
|
|
|
|
|
+ " \"\"\"解析用户预算字符串,返回 (min, max)\"\"\"\n",
|
|
|
|
|
+ " nums = [float(x) for x in re.findall(r'\\d+', str(budget_str).replace(',', ''))]\n",
|
|
|
|
|
+ " if not nums: return 0, 999999 \n",
|
|
|
|
|
+ " if \"以内\" in budget_str or \"以下\" in budget_str: return 0, nums[0]\n",
|
|
|
|
|
+ " if \"以上\" in budget_str: return nums[0], 999999\n",
|
|
|
|
|
+ " if len(nums) >= 2: return min(nums), max(nums)\n",
|
|
|
|
|
+ " return 0, nums[0]\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ "def extract_all_prices(raw_results):\n",
|
|
|
|
|
+ " \"\"\"从搜索结果列表中提取所有有效的价格\"\"\"\n",
|
|
|
|
|
+ " prices = []\n",
|
|
|
|
|
+ " for res in raw_results:\n",
|
|
|
|
|
+ " # 只处理文本类型的结果\n",
|
|
|
|
|
+ " if res.get('type') == 'text':\n",
|
|
|
|
|
+ " text = res.get('title', '') + \" \" + res.get('content', '')\n",
|
|
|
|
|
+ " # 匹配 ¥, $, 元 等格式\n",
|
|
|
|
|
+ " matches = re.findall(r'(?:¥|¥|\\$|HK\\$|NT\\$)\\s*(\\d+(?:,\\d{3})*(?:\\.\\d+)?)', text)\n",
|
|
|
|
|
+ " for m in matches:\n",
|
|
|
|
|
+ " val = float(m.replace(',', ''))\n",
|
|
|
|
|
+ " # 过滤掉像年份(2025)或过小/过大的异常值\n",
|
|
|
|
|
+ " if 10 < val < 100000 and val not in [2024, 2025, 2026]:\n",
|
|
|
|
|
+ " prices.append(val)\n",
|
|
|
|
|
+ " # 备用正则:匹配 \"xxx元\"\n",
|
|
|
|
|
+ " matches_yuan = re.findall(r'(\\d+(?:,\\d{3})*(?:\\.\\d+)?)\\s*元', text)\n",
|
|
|
|
|
+ " for m in matches_yuan:\n",
|
|
|
|
|
+ " val = float(m.replace(',', ''))\n",
|
|
|
|
|
+ " if 10 < val < 100000 and val not in [2024, 2025, 2026]:\n",
|
|
|
|
|
+ " prices.append(val)\n",
|
|
|
|
|
+ " return prices\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ "def find_best_product(hunter, profiler_agent, keyword, budget_min, budget_max):\n",
|
|
|
|
|
+ " limit_upper = budget_max * 1.2\n",
|
|
|
|
|
+ " limit_lower = budget_min * 0.8\n",
|
|
|
|
|
+ " \n",
|
|
|
|
|
+ " all_candidates = []\n",
|
|
|
|
|
+ " current_kw = keyword\n",
|
|
|
|
|
+ " \n",
|
|
|
|
|
+ " # --- Round 1: 首次搜索 ---\n",
|
|
|
|
|
+ " print(f\" 🕵️ 第1次搜索: {current_kw} 价格\")\n",
|
|
|
|
|
+ " results_1 = hunter.search_raw(f\"{current_kw} 价格\")\n",
|
|
|
|
|
+ " \n",
|
|
|
|
|
+ " fallback_img = \"\"\n",
|
|
|
|
|
+ " for r in results_1:\n",
|
|
|
|
|
+ " if r.get('images'): \n",
|
|
|
|
|
+ " fallback_img = r['images'][0]\n",
|
|
|
|
|
+ " break\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ " has_valid_info = False\n",
|
|
|
|
|
+ " for res in results_1:\n",
|
|
|
|
|
+ " if res.get('type') == 'text':\n",
|
|
|
|
|
+ " has_valid_info = True\n",
|
|
|
|
|
+ " p_vals = extract_all_prices([res])\n",
|
|
|
|
|
+ " if p_vals:\n",
|
|
|
|
|
+ " res['price_val'] = p_vals[0]\n",
|
|
|
|
|
+ " # 📝【核心修复点1】记录当前结果所属的关键词\n",
|
|
|
|
|
+ " res['source_kw'] = current_kw \n",
|
|
|
|
|
+ " all_candidates.append(res)\n",
|
|
|
|
|
+ " \n",
|
|
|
|
|
+ " if limit_lower <= p_vals[0] <= limit_upper:\n",
|
|
|
|
|
+ " if not res.get('img') and fallback_img: res['img'] = fallback_img\n",
|
|
|
|
|
+ " return res, f\"约 {p_vals[0]}元\", current_kw\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ " # --- 机制 3: 无数据防御 ---\n",
|
|
|
|
|
+ " if not has_valid_info:\n",
|
|
|
|
|
+ " print(f\" ⚠️ [机制3触发] 首次搜索无有效信息。\")\n",
|
|
|
|
|
+ " correction_prompt = f\"原策略 '{current_kw}' 搜索结果为空,请推荐一个同品类但更热门的具体商品型号。只输出关键词。\"\n",
|
|
|
|
|
+ " new_kw = profiler_agent.run(correction_prompt).strip()\n",
|
|
|
|
|
+ " print(f\" 🔄 军师换词: {new_kw}\")\n",
|
|
|
|
|
+ " current_kw = new_kw\n",
|
|
|
|
|
+ " \n",
|
|
|
|
|
+ " results = hunter.search_raw(f\"{current_kw} 价格\")\n",
|
|
|
|
|
+ " \n",
|
|
|
|
|
+ " # 更新 fallback_img\n",
|
|
|
|
|
+ " fallback_img = \"\" \n",
|
|
|
|
|
+ " for r in results:\n",
|
|
|
|
|
+ " if r.get('images'): \n",
|
|
|
|
|
+ " fallback_img = r['images'][0]\n",
|
|
|
|
|
+ " break\n",
|
|
|
|
|
+ " \n",
|
|
|
|
|
+ " for res in results:\n",
|
|
|
|
|
+ " if res.get('type') == 'text':\n",
|
|
|
|
|
+ " p_vals = extract_all_prices([res])\n",
|
|
|
|
|
+ " if p_vals:\n",
|
|
|
|
|
+ " res['price_val'] = p_vals[0]\n",
|
|
|
|
|
+ " # 📝【核心修复点1】记录关键词\n",
|
|
|
|
|
+ " res['source_kw'] = current_kw\n",
|
|
|
|
|
+ " all_candidates.append(res)\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ " # --- 机制 1 & 2: 价格修正 ---\n",
|
|
|
|
|
+ " avg_price = np.mean([c['price_val'] for c in all_candidates]) if all_candidates else 0\n",
|
|
|
|
|
+ " \n",
|
|
|
|
|
+ " if avg_price > 0:\n",
|
|
|
|
|
+ " correction_prompt = \"\"\n",
|
|
|
|
|
+ " if avg_price > limit_upper:\n",
|
|
|
|
|
+ " print(f\" 💸 [机制1触发] 均价 {int(avg_price)} > 上限 {int(limit_upper)},找平替...\")\n",
|
|
|
|
|
+ " correction_prompt = f\"原策略 '{current_kw}' 均价约 {int(avg_price)}元,超预算 ({budget_max}元)。请推荐一个同品类更便宜的具体型号(平替)。只输出关键词。\"\n",
|
|
|
|
|
+ " elif avg_price < limit_lower:\n",
|
|
|
|
|
+ " print(f\" 📉 [机制2触发] 均价 {int(avg_price)} < 下限 {int(limit_lower)},找升级款...\")\n",
|
|
|
|
|
+ " correction_prompt = f\"原策略 '{current_kw}' 均价约 {int(avg_price)}元,低于预算下限 ({budget_min}元)。请推荐一个同品类更高端的型号。只输出关键词。\"\n",
|
|
|
|
|
+ " \n",
|
|
|
|
|
+ " if correction_prompt:\n",
|
|
|
|
|
+ " new_kw = profiler_agent.run(correction_prompt).strip()\n",
|
|
|
|
|
+ " print(f\" 🔄 军师修正: {new_kw}\")\n",
|
|
|
|
|
+ " current_kw = new_kw\n",
|
|
|
|
|
+ " \n",
|
|
|
|
|
+ " results_2 = hunter.search_raw(f\"{new_kw} 价格\")\n",
|
|
|
|
|
+ " \n",
|
|
|
|
|
+ " # 更新 fallback_img\n",
|
|
|
|
|
+ " fallback_img = \"\" \n",
|
|
|
|
|
+ " for r in results_2:\n",
|
|
|
|
|
+ " if r.get('images'): \n",
|
|
|
|
|
+ " fallback_img = r['images'][0]\n",
|
|
|
|
|
+ " break\n",
|
|
|
|
|
+ " \n",
|
|
|
|
|
+ " for res in results_2:\n",
|
|
|
|
|
+ " if res.get('type') == 'text':\n",
|
|
|
|
|
+ " p_vals = extract_all_prices([res])\n",
|
|
|
|
|
+ " if p_vals:\n",
|
|
|
|
|
+ " res['price_val'] = p_vals[0]\n",
|
|
|
|
|
+ " # 📝【核心修复点1】记录关键词\n",
|
|
|
|
|
+ " res['source_kw'] = current_kw\n",
|
|
|
|
|
+ " all_candidates.append(res) \n",
|
|
|
|
|
+ " \n",
|
|
|
|
|
+ " if limit_lower <= p_vals[0] <= limit_upper:\n",
|
|
|
|
|
+ " if not res.get('img') and fallback_img: res['img'] = fallback_img\n",
|
|
|
|
|
+ " tag = \"(平替)\" if avg_price > limit_upper else \"(升级)\"\n",
|
|
|
|
|
+ " return res, f\"约 {p_vals[0]}元 {tag}\", current_kw\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ " # --- 机制 4: 兜底防御 ---\n",
|
|
|
|
|
+ " print(\" ⚠️ [机制4触发] 启用强制兜底模式...\")\n",
|
|
|
|
|
+ " best_fallback = None\n",
|
|
|
|
|
+ " status_msg = \"暂无报价\"\n",
|
|
|
|
|
+ " \n",
|
|
|
|
|
+ " if all_candidates:\n",
|
|
|
|
|
+ " # 选离预算最近的\n",
|
|
|
|
|
+ " target = (budget_min + budget_max) / 2\n",
|
|
|
|
|
+ " best_fallback = sorted(all_candidates, key=lambda x: abs(x['price_val'] - target))[0]\n",
|
|
|
|
|
+ " p = best_fallback['price_val']\n",
|
|
|
|
|
+ " \n",
|
|
|
|
|
+ " if p > limit_upper: status_msg = f\"约 {p}元 (⚠️超预算)\"\n",
|
|
|
|
|
+ " elif p < limit_lower: status_msg = f\"约 {p}元 (📉低于预算)\"\n",
|
|
|
|
|
+ " else: status_msg = f\"约 {p}元\"\n",
|
|
|
|
|
+ " \n",
|
|
|
|
|
+ " elif results_1:\n",
|
|
|
|
|
+ " # 实在没数据,硬取第一条\n",
|
|
|
|
|
+ " for res in results_1:\n",
|
|
|
|
|
+ " if res.get('type') == 'text': \n",
|
|
|
|
|
+ " best_fallback = res\n",
|
|
|
|
|
+ " # 兜底时如果也没价格,就用原始关键词\n",
|
|
|
|
|
+ " best_fallback['source_kw'] = keyword \n",
|
|
|
|
|
+ " break\n",
|
|
|
|
|
+ " \n",
|
|
|
|
|
+ " if best_fallback:\n",
|
|
|
|
|
+ " if not best_fallback.get('img') and fallback_img:\n",
|
|
|
|
|
+ " best_fallback['img'] = fallback_img\n",
|
|
|
|
|
+ " \n",
|
|
|
|
|
+ " # 📝【核心修复点2】返回结果里记录的那个 source_kw,而不是当前的 current_kw\n",
|
|
|
|
|
+ " final_name_to_use = best_fallback.get('source_kw', current_kw)\n",
|
|
|
|
|
+ " \n",
|
|
|
|
|
+ " return best_fallback, status_msg, final_name_to_use\n",
|
|
|
|
|
+ " \n",
|
|
|
|
|
+ " return None, \"搜索失败\", keyword"
|
|
|
|
|
+ ]
|
|
|
|
|
+ },
|
|
|
|
|
+ {
|
|
|
|
|
+ "cell_type": "code",
|
|
|
|
|
+ "execution_count": null,
|
|
|
|
|
+ "id": "4ec82cc7",
|
|
|
|
|
+ "metadata": {},
|
|
|
|
|
+ "outputs": [],
|
|
|
|
|
+ "source": [
|
|
|
|
|
+ "if not user_input_data:\n",
|
|
|
|
|
+ " print(\"❌ 未加载用户数据\")\n",
|
|
|
|
|
+ "else:\n",
|
|
|
|
|
+ " # 0. 解析预算\n",
|
|
|
|
|
+ " b_min, b_max = parse_budget_range(user_input_data.get('预算', ''))\n",
|
|
|
|
|
+ " print(f\"\\n💰 预算范围: {b_min} - {b_max}元\")\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ " # 1. 军师制定策略\n",
|
|
|
|
|
+ " profile_text = \"\\n\".join([f\"- {k}: {v if v else '未知/不限'}\" for k, v in user_input_data.items()])\n",
|
|
|
|
|
+ " print(f\"\\n🚀 任务启动...\\n{'-'*40}\")\n",
|
|
|
|
|
+ " print(\"\\n🧠 [1/3] 军师正在制定初步策略...\")\n",
|
|
|
|
|
+ " search_strategy = profiler_agent.run(f\"请根据以下用户画像制定搜索策略:\\n\\n{profile_text}\")\n",
|
|
|
|
|
+ " print(f\"📝 策略: \\n{search_strategy}\")\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ " # 2. 准备循环\n",
|
|
|
|
|
+ " keywords = [k.strip() for k in search_strategy.replace(\",\", \",\").replace(\"\\n\", \",\").split(',') if k.strip()]\n",
|
|
|
|
|
+ " final_items = []\n",
|
|
|
|
|
+ " hunter = BatchSearchTool()\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ " print(f\"\\n🔄 进入处理流程 (共 {len(keywords)} 个商品)...\")\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ " for index, kw in enumerate(keywords):\n",
|
|
|
|
|
+ " print(f\"\\n 👉 [商品 {index+1}/{len(keywords)}] 正在处理: {kw}\")\n",
|
|
|
|
|
+ " \n",
|
|
|
|
|
+ " # 调用智能搜索函数 (传入 min 和 max)\n",
|
|
|
|
|
+ " valid_result, price_status, final_kw = find_best_product(hunter, profiler_agent, kw, b_min, b_max)\n",
|
|
|
|
|
+ " \n",
|
|
|
|
|
+ " if not valid_result:\n",
|
|
|
|
|
+ " print(\" ❌ 彻底无数据,跳过。\")\n",
|
|
|
|
|
+ " continue\n",
|
|
|
|
|
+ " \n",
|
|
|
|
|
+ " # === 生成文案 ===\n",
|
|
|
|
|
+ " product_name = valid_result.get('title', final_kw)\n",
|
|
|
|
|
+ " \n",
|
|
|
|
|
+ " print(f\" ✍️ 正在撰写文案: {product_name[:30]}...\")\n",
|
|
|
|
|
+ " pitch_prompt = f\"\"\"\n",
|
|
|
|
|
+ " 商品:{product_name}\n",
|
|
|
|
|
+ " 价格:{price_status}\n",
|
|
|
|
|
+ " 卖点片段:{valid_result.get('content', '')[:200]}...\n",
|
|
|
|
|
+ " \n",
|
|
|
|
|
+ " 请写一句30字以内的种草文案。\n",
|
|
|
|
|
+ " \"\"\"\n",
|
|
|
|
|
+ " pitch = pitcher_agent.run(pitch_prompt)\n",
|
|
|
|
|
+ " \n",
|
|
|
|
|
+ " final_items.append({\n",
|
|
|
|
|
+ " \"name\": final_kw, \n",
|
|
|
|
|
+ " \"title_full\": product_name,\n",
|
|
|
|
|
+ " \"price\": price_status,\n",
|
|
|
|
|
+ " \"desc\": pitch.replace(\"\\n\", \" \").strip(),\n",
|
|
|
|
|
+ " \"img\": valid_result.get('img', ''),\n",
|
|
|
|
|
+ " \"link\": valid_result.get('url', '')\n",
|
|
|
|
|
+ " })\n",
|
|
|
|
|
+ " print(f\" ✅ 已收录 (状态: {price_status})\")"
|
|
|
|
|
+ ]
|
|
|
|
|
+ },
|
|
|
|
|
+ {
|
|
|
|
|
+ "cell_type": "markdown",
|
|
|
|
|
+ "id": "0197ef43",
|
|
|
|
|
+ "metadata": {},
|
|
|
|
|
+ "source": [
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ "### 第6部分:输出礼物计划"
|
|
|
|
|
+ ]
|
|
|
|
|
+ },
|
|
|
|
|
+ {
|
|
|
|
|
+ "cell_type": "code",
|
|
|
|
|
+ "execution_count": null,
|
|
|
|
|
+ "id": "8426fb13",
|
|
|
|
|
+ "metadata": {},
|
|
|
|
|
+ "outputs": [],
|
|
|
|
|
+ "source": [
|
|
|
|
|
+ "# --- 4. 渲染与保存 ---\n",
|
|
|
|
|
+ "print(f\"\\n💾 正在生成最终报告...\")\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ "if not final_items:\n",
|
|
|
|
|
+ " final_md = \"很抱歉,网络搜索似乎出现了问题,未能获取到任何商品信息。\"\n",
|
|
|
|
|
+ "else:\n",
|
|
|
|
|
+ " table_header = \"| 🎁 礼物名称 | 💰 价格 | ✨ 种草理由 | 🖼️ 图片/链接 |\\n| :--- | :--- | :--- | :--- |\\n\"\n",
|
|
|
|
|
+ " table_rows = []\n",
|
|
|
|
|
+ " \n",
|
|
|
|
|
+ " for item in final_items:\n",
|
|
|
|
|
+ " # 1. 清洗文本字段 (防止 | 破坏表格)\n",
|
|
|
|
|
+ " name = item.get('name', '未知').replace(\"|\", \"/\")\n",
|
|
|
|
|
+ " price = item.get('price', '暂无').replace(\"|\", \"/\")\n",
|
|
|
|
|
+ " desc = item.get('desc', '').replace(\"|\", \"/\")\n",
|
|
|
|
|
+ " \n",
|
|
|
|
|
+ " # 2. 🚨【核心修复】清洗链接中的竖线\n",
|
|
|
|
|
+ " # 百度/京东链接常包含 '|',必须替换为 '%7C',否则 Markdown 表格会炸\n",
|
|
|
|
|
+ " raw_link = item.get('link', '#')\n",
|
|
|
|
|
+ " safe_link = raw_link.replace(\"|\", \"%7C\")\n",
|
|
|
|
|
+ " \n",
|
|
|
|
|
+ " raw_img = item.get('img', '')\n",
|
|
|
|
|
+ " safe_img = raw_img.replace(\"|\", \"%7C\")\n",
|
|
|
|
|
+ " \n",
|
|
|
|
|
+ " # 3. 构建媒体列\n",
|
|
|
|
|
+ " if safe_img and safe_img.startswith(\"http\"):\n",
|
|
|
|
|
+ " # 图片链接套购买链接\n",
|
|
|
|
|
+ " media = f\"[]({safe_link})\"\n",
|
|
|
|
|
+ " else:\n",
|
|
|
|
|
+ " media = f\"[点击购买]({safe_link})\"\n",
|
|
|
|
|
+ " \n",
|
|
|
|
|
+ " # 4. 组装行 (注意名字上的链接也要用 safe_link)\n",
|
|
|
|
|
+ " # 使用 strip() 去除可能的首尾空格\n",
|
|
|
|
|
+ " row = f\"| [{name}]({safe_link}) | {price} | {desc} | {media} |\"\n",
|
|
|
|
|
+ " table_rows.append(row)\n",
|
|
|
|
|
+ " \n",
|
|
|
|
|
+ " final_md = table_header + \"\\n\".join(table_rows)\n",
|
|
|
|
|
+ "filename = \"outputs/gift_plan_output.md\"\n",
|
|
|
|
|
+ "# 确保输出目录存在\n",
|
|
|
|
|
+ "os.makedirs(os.path.dirname(filename), exist_ok=True)\n",
|
|
|
|
|
+ "\n",
|
|
|
|
|
+ "with open(filename, \"w\", encoding=\"utf-8\") as f:\n",
|
|
|
|
|
+ " f.write(final_md)\n",
|
|
|
|
|
+ "print(f\"🎉 任务完成!文件已保存: {os.path.abspath(filename)}\")"
|
|
|
|
|
+ ]
|
|
|
|
|
+ },
|
|
|
|
|
+ {
|
|
|
|
|
+ "cell_type": "markdown",
|
|
|
|
|
+ "id": "d8b884b5",
|
|
|
|
|
+ "metadata": {},
|
|
|
|
|
+ "source": [
|
|
|
|
|
+ "### 第7部分:总结与展望"
|
|
|
|
|
+ ]
|
|
|
|
|
+ },
|
|
|
|
|
+ {
|
|
|
|
|
+ "cell_type": "markdown",
|
|
|
|
|
+ "id": "08b0b8e8",
|
|
|
|
|
+ "metadata": {},
|
|
|
|
|
+ "source": [
|
|
|
|
|
+ "#### 实现的功能\n",
|
|
|
|
|
+ "- 基于用户输入的个人信息,生成符合预算的礼物建议\n",
|
|
|
|
|
+ "- 支持用户自定义预算范围、节日、个人喜好等\n",
|
|
|
|
|
+ "- 支持百度MCP和Tavily API双数据源,利用搜索引擎获取最新的商品信息和价格\n",
|
|
|
|
|
+ "- 提供可视化的建议结果展示\n",
|
|
|
|
|
+ "#### 遇到的挑战与解决方案\n",
|
|
|
|
|
+ "- 大模型的“幻觉”问题(JSON格式错误/编造数据)\n",
|
|
|
|
|
+ " - 解决方案:放弃让 LLM 直接生成最终数据。改为使用 Python 正则表达式 从搜索结果中暴力提取硬数据(价格、图片),仅让 LLM 负责生成文案。代码逻辑负责准确性,模型负责创造性。\n",
|
|
|
|
|
+ "- 上下文过长导致提取失败\n",
|
|
|
|
|
+ " - 解决方案:结合实际业务场景,分析各个阶段对上下文的要求,在搜索阶段限制返回长度。同时通过拆分 “硬数据流”(找参数)和 “软数据流”(找卖点),大幅降低单次上下文长度,提升响应速度。\n",
|
|
|
|
|
+ "- 大模型推荐的礼品价格超出预算\n",
|
|
|
|
|
+ " - 解决方案:引入检核机制。如果搜到的商品均价超预算,系统会自动呼叫“军师”重新制定“平替”策略,直到找到合适商品为止。\n",
|
|
|
|
|
+ "- Agent传入的参数格式问题\n",
|
|
|
|
|
+ " - 解决方案:在工具层兼容 Agent 传入的各种参数格式(JSON/字符串、逗号/换行符分隔),确保搜索指令不丢失。\n",
|
|
|
|
|
+ "#### 未来改进方向\n",
|
|
|
|
|
+ "- 前端交互:开发前端页面,替代目前的Notebook交互,提供更好的用户交互体验\n",
|
|
|
|
|
+ "- 数据源深度集成:完全接入百度优选MCP 的比价与历史价格接口,获取更精准的实时价格和库存信息,实现“全网比价”\n",
|
|
|
|
|
+ "- 丰富选项:增加更多的个人喜好选项,如喜欢的商品类型、品牌等\n"
|
|
|
|
|
+ ]
|
|
|
|
|
+ }
|
|
|
|
|
+ ],
|
|
|
|
|
+ "metadata": {
|
|
|
|
|
+ "kernelspec": {
|
|
|
|
|
+ "display_name": "ai_3.10",
|
|
|
|
|
+ "language": "python",
|
|
|
|
|
+ "name": "python3"
|
|
|
|
|
+ },
|
|
|
|
|
+ "language_info": {
|
|
|
|
|
+ "codemirror_mode": {
|
|
|
|
|
+ "name": "ipython",
|
|
|
|
|
+ "version": 3
|
|
|
|
|
+ },
|
|
|
|
|
+ "file_extension": ".py",
|
|
|
|
|
+ "mimetype": "text/x-python",
|
|
|
|
|
+ "name": "python",
|
|
|
|
|
+ "nbconvert_exporter": "python",
|
|
|
|
|
+ "pygments_lexer": "ipython3",
|
|
|
|
|
+ "version": "3.10.19"
|
|
|
|
|
+ }
|
|
|
|
|
+ },
|
|
|
|
|
+ "nbformat": 4,
|
|
|
|
|
+ "nbformat_minor": 5
|
|
|
|
|
+}
|