chat_tools.py 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181
  1. from fastapi import APIRouter, HTTPException
  2. from datetime import datetime
  3. from ..core.ark_client import config, client
  4. from ..schemas.chat import ChatMessage, ChatResponse, CircleRequest, CirclePromptConfig, HistoricalFigure, RephraseRequest, FigureUpsert
  5. from ..db.souyue_mongo import get_mblog_by_id
  6. from ..db.mongo import get_circle_prompt, upsert_circle_prompt, get_all_figures, get_figure_by_id, insert_figure, update_figure, delete_figure
  7. router = APIRouter()
  8. def _build_prompt(product_text: str, prompt_config: dict) -> str:
  9. name = prompt_config.get("name", "兴趣圈")
  10. role = prompt_config.get("role", "活跃用户")
  11. style = prompt_config.get("style", "自然亲切,有活人感")
  12. keywords: list = prompt_config.get("keywords") or []
  13. forbidden: list = prompt_config.get("forbidden") or []
  14. extraInstruction = prompt_config.get("extra_instruction")
  15. lines = [
  16. f"你是{role},活跃在{name}兴趣圈。",
  17. "请根据以下帖子信息,生成一条10-30字的评论,要求:",
  18. "1. 内容指向性强,结合帖子具体内容",
  19. f"2. 风格:{style}",
  20. ]
  21. seq = 3
  22. if keywords:
  23. lines.append(f"{seq}. 适当融入关键词(自然使用):{', '.join(keywords)}")
  24. seq += 1
  25. if forbidden:
  26. lines.append(f"{seq}. 禁止使用以下词语:{', '.join(forbidden)}")
  27. seq += 1
  28. lines.append(f"{seq}. 语言自然,不要暴露你是AI")
  29. # 额外要求
  30. if extraInstruction:
  31. lines.append(f"【额外要求】{','.join(extraInstruction)}")
  32. lines.append(f"\n帖子内容:{product_text}")
  33. return "\n".join(lines)
  34. # 存储/更新兴趣圈提示词模版(appName 已存在则覆盖)
  35. @router.post("/prompt")
  36. async def save_circle_prompt(promptcfg: CirclePromptConfig):
  37. try:
  38. upsert_circle_prompt(promptcfg.model_dump())
  39. return {"message": "保存成功", "appName": promptcfg.appName}
  40. except Exception as e:
  41. raise HTTPException(status_code=500, detail=f"保存失败: {str(e)}")
  42. # 评论帖子的马甲机器人,无状态,支持批量对多个帖子智能回复
  43. @router.post("/airesp", response_model=ChatResponse)
  44. async def generate_circle_comment(request: CircleRequest):
  45. doc = get_mblog_by_id(request.id)
  46. if not doc:
  47. raise HTTPException(status_code=404, detail="帖子不存在")
  48. title = doc.get("title", "")
  49. brief = doc.get("brief", "")
  50. nickname = doc.get("nickname", "")
  51. app_name = doc.get("appName", "")
  52. images: list = doc.get("images") or []
  53. product_text = f"主题:{title}\n摘要:{brief}\n发布者:{nickname}"
  54. if images:
  55. product_text += "\n图片:\n" + "\n".join(images)
  56. prompt_config = get_circle_prompt(app_name)
  57. prompt = _build_prompt(product_text, prompt_config)
  58. response = client.responses.create(
  59. model=config.MODEL_NAME,
  60. input=[{"role": "user", "content": prompt}],
  61. stream=False,
  62. store=False,
  63. )
  64. message_content = ""
  65. for item in response.output:
  66. if hasattr(item, 'type') and item.type == 'message' and hasattr(item, 'content'):
  67. if isinstance(item.content, list):
  68. for content_item in item.content:
  69. if hasattr(content_item, 'text'):
  70. message_content += content_item.text
  71. else:
  72. message_content += str(item.content)
  73. if not message_content:
  74. raise HTTPException(status_code=500, detail="AI未能生成评论")
  75. return ChatResponse(
  76. message=ChatMessage(role="assistant", content=message_content, timestamp=datetime.now()),
  77. model=response.model,
  78. usage=response.usage.model_dump() if response.usage else None,
  79. )
  80. # ===================== 历史人物管理 =====================
  81. # 获取历史人物列表
  82. @router.get("/figures", response_model=list[HistoricalFigure])
  83. async def list_figures():
  84. return get_all_figures()
  85. # 获取单个历史人物
  86. @router.get("/figures/{id}", response_model=HistoricalFigure)
  87. async def get_figure(id: str):
  88. doc = get_figure_by_id(id)
  89. if not doc:
  90. raise HTTPException(status_code=404, detail="历史人物不存在")
  91. return doc
  92. # 新增历史人物
  93. @router.post("/figures", response_model=HistoricalFigure)
  94. async def create_figure(figure: FigureUpsert):
  95. inserted_id = insert_figure(figure.model_dump())
  96. if not inserted_id:
  97. raise HTTPException(status_code=500, detail="新增失败")
  98. return {**figure.model_dump(), "_id": inserted_id}
  99. # 修改历史人物
  100. @router.put("/figures/{id}", response_model=HistoricalFigure)
  101. async def modify_figure(id: str, figure: FigureUpsert):
  102. matched = update_figure(id, figure.model_dump())
  103. if not matched:
  104. raise HTTPException(status_code=404, detail="历史人物不存在")
  105. return {**figure.model_dump(), "_id": id}
  106. # 删除历史人物
  107. @router.delete("/figures/{id}")
  108. async def remove_figure(id: str):
  109. deleted = delete_figure(id)
  110. if not deleted:
  111. raise HTTPException(status_code=404, detail="历史人物不存在")
  112. return {"message": "删除成功", "id": id}
  113. # ===================== 润色接口 =====================
  114. # 润色接口
  115. @router.post("/rephrase")
  116. async def rephrase_as_figure(request: RephraseRequest):
  117. figure = get_figure_by_id(request.figureId)
  118. if not figure:
  119. raise HTTPException(status_code=404, detail="历史人物不存在")
  120. prompt = (
  121. f"你是{figure['name']},{figure['description']},生活在{figure['era']}。\n"
  122. f"请将以下话语改写成{figure['name']}的说话风格,保留原意,体现其性格特点({figure['prompt']})。\n"
  123. f"只输出改写后的内容,不要解释、不要加引号。\n"
  124. f"原文:{request.text}"
  125. )
  126. response = client.responses.create(
  127. model=config.MODEL_NAME,
  128. input=[{"role": "user", "content": prompt}],
  129. stream=False,
  130. store=False,
  131. # thinking={"type":"auto"},
  132. )
  133. rephrased = ""
  134. for item in response.output:
  135. if hasattr(item, 'type') and item.type == 'message' and hasattr(item, 'content'):
  136. if isinstance(item.content, list):
  137. for content_item in item.content:
  138. if hasattr(content_item, 'text'):
  139. rephrased += content_item.text
  140. else:
  141. rephrased += str(item.content)
  142. if not rephrased:
  143. raise HTTPException(status_code=500, detail="AI未能生成润色结果")
  144. return {
  145. "original": request.text,
  146. "rephrased": rephrased,
  147. "figure": figure["name"],
  148. }