Files
life-echo/api/agents/prompts/conversation_prompts.py

344 lines
15 KiB
Python
Raw Normal View History

2026-01-07 11:56:53 +08:00
"""
对话 Agent 提示词模板和访谈问题库
"""
from enum import Enum
from typing import List, Dict
import random
2026-01-07 11:56:53 +08:00
class ConversationStage(str, Enum):
"""对话阶段枚举"""
CHILDHOOD = "childhood" # 童年
EDUCATION = "education" # 教育
CAREER = "career" # 事业
FAMILY = "family" # 家庭
BELIEFS = "beliefs" # 信念
SUMMARY = "summary" # 人生总结
# 访谈问题库
INTERVIEW_QUESTIONS: Dict[ConversationStage, List[str]] = {
ConversationStage.CHILDHOOD: [
"你是在哪里长大的?小时候周围的环境是什么样的,有哪些让你印象深刻的童年记忆?",
"童年时期的你是个怎样的孩子?有没有做过什么淘气或有趣的事情,现在想起来还会让你发笑?",
"能聊聊你小时候的家庭吗?比如父母是怎样的人,他们对你的成长有什么影响吗?",
"小时候你有过什么梦想?那时候你最想长大后做什么?",
],
ConversationStage.EDUCATION: [
"上学的时候你是个怎么样的学生?你喜欢学校生活吗?",
"在求学过程中,有没有哪位老师或同学对你影响特别大?能说说他们的故事吗?",
"学生时代你参加过什么课外活动或者比赛吗?有没有哪段经历让你特别难忘?",
"那时候你对未来有什么打算吗?比如毕业后想从事什么职业,或者希望过怎样的生活?",
],
ConversationStage.CAREER: [
"第一次走出校园开始工作时,你还记得当时的情景吗?当时你的心情怎么样,有发生什么难忘的事吗?",
"你当初是怎么选择进入现在这个行业的?其中有什么契机或故事吗?",
"在工作中有没有遇到过特别大的挑战或低谷?当时你是怎么挺过来的?",
"职业生涯中有没有哪个成就或时刻让你特别自豪?能跟我分享一下那个故事吗?",
"在事业的发展过程中,有哪些重要的转折点?比如跳槽、升职或者创业,这些经历对你意味着什么?",
"回顾这一路,有哪些人对你的事业帮助最大或者影响最深?有没有特别想感谢的贵人或伙伴?",
],
ConversationStage.FAMILY: [
"可以聊聊你和你伴侣的故事吗?你们是怎么认识的,又是什么让你决定与他/她携手一生?",
"孩子在你的生活中意味着什么?做父母的过程中,有没有让你特别骄傲或者难忘的瞬间?",
"在家庭生活中,有没有什么传统或者特别的习惯,让你感到温馨和快乐?",
"平时你和家人喜欢一起做些什么?周末或假日你们通常会怎么度过?",
"你觉得家庭在你的人生中扮演了一个怎样的角色?",
"工作和家庭要怎么兼顾呢?你是如何平衡事业和家庭的?在两边兼顾的时候有没有遇到困难,后来又是怎么克服的?",
],
ConversationStage.BELIEFS: [
"你人生中有没有一些一直坚守的信念或者座右铭?这些信念给了你怎样的力量或者影响?",
"对你来说,哪些价值观是最重要的?这些价值观是受到哪些人或经历的影响而形成的呢?",
"当你遇到困难和低谷的时候,是什么支撑着你坚持下去?",
"你如何看待'成功''幸福'?对你来说它们分别意味着什么?",
2026-01-07 11:56:53 +08:00
],
ConversationStage.SUMMARY: [
"回顾你走过的路,你觉得这一生中最重要的经验或教训是什么?",
"在你的生活中,你最感激的人和事有哪些?有没有特别觉得自己很幸运的地方?",
"如果能对年轻时候的自己说几句话,你会想告诉他/她什么?",
"展望未来,你还有什么愿望或目标吗?有没有一直想尝试但还没来得及做的事情?",
"最后,你希望家人和后代记住你是一个怎样的人?",
],
}
def get_system_prompt(current_stage: ConversationStage, covered_topics: List[str], user_latest_response: str) -> str:
"""
生成对话 Agent 的系统提示词
Args:
current_stage: 当前对话阶段
covered_topics: 已聊过的话题列表
user_latest_response: 用户最新回答
Returns:
系统提示词字符串
"""
stage_name_map = {
ConversationStage.CHILDHOOD: "童年",
ConversationStage.EDUCATION: "教育",
ConversationStage.CAREER: "事业",
ConversationStage.FAMILY: "家庭",
ConversationStage.BELIEFS: "信念",
ConversationStage.SUMMARY: "人生总结",
}
covered_topics_str = "".join(covered_topics) if covered_topics else "暂无"
prompt = f"""你是「岁月知己」,一位资深的人生故事访谈者,专注于帮助用户回忆和讲述人生经历。
2026-01-21 22:31:03 +01:00
## 角色定位
你如同一位老朋友用真诚温暖的方式倾听用户的故事通过自然的对话引导用户分享更多细节
## 访谈技巧
1. 积极倾听先对用户分享的内容给予简短回应表达理解和共鸣
2. 深度追问围绕用户提到的具体场景人物感受进行细节追问
3. 一次一问每次只提一个问题给用户充分思考和表达的空间
4. 自然过渡当一个话题聊透后自然引入下一个相关话题
## 输出要求
- 直接输出你要对用户说的话
- 禁止输出任何括号注释思考过程策略说明
- 禁止使用"我注意到""我想了解"等采访腔调
- 语气要像朋友聊天自然亲切
当前阶段{stage_name_map.get(current_stage, current_stage.value)}
2026-01-07 11:56:53 +08:00
已聊话题{covered_topics_str}
2026-01-21 22:31:03 +01:00
请直接回应用户不要有任何元描述"""
2026-01-07 11:56:53 +08:00
return prompt
def get_questions_for_stage(stage: ConversationStage) -> List[str]:
"""获取指定阶段的所有问题"""
return INTERVIEW_QUESTIONS.get(stage, [])
2026-01-21 22:31:03 +01:00
SLOT_NAME_MAP = {
# 童年
"place": "成长的地方",
"people": "重要的人",
"daily_life": "日常生活",
"emotion": "童年感受",
"turning_event": "难忘的事",
# 教育
"school": "学校经历",
"city": "求学的城市",
"motivation": "学习动力",
"challenge": "遇到的挑战",
"change": "成长变化",
# 事业
"job": "工作内容",
"environment": "工作环境",
"decision": "重要决定",
"pressure": "压力与困难",
"growth": "职业成长",
# 家庭
"relationship": "家人关系",
"conflict": "矛盾与化解",
"support": "相互支持",
"responsibility": "家庭责任",
# 信念
"value": "核心价值观",
"regret": "遗憾与释怀",
"pride": "骄傲的事",
"lesson": "人生经验",
}
# 阶段关联话题(用于自然过渡)
STAGE_RELATED_TOPICS = {
"childhood": ["family", "education"], # 童年可以自然聊到家庭、教育
"education": ["childhood", "career"], # 教育可以聊到童年、事业
"career": ["education", "family", "belief"], # 事业可以聊到教育、家庭、信念
"family": ["childhood", "career", "belief"], # 家庭可以聊到童年、事业、信念
"belief": ["career", "family"], # 信念可以聊到事业、家庭
}
# 轻松话题(用于调节气氛)
LIGHT_TOPICS = [
"有什么爱好或者特别喜欢的消遣方式吗?",
"最近有什么让你开心的事吗?",
"有没有什么趣事想分享?",
"平时喜欢看什么书或者电影吗?",
]
# 回应风格模板(增加多样性)
RESPONSE_STYLES = [
"empathy", # 共情式回应
"curious", # 好奇追问
"reflection", # 感慨反思
"lighthearted", # 轻松调侃
"connection", # 关联自身(虚构)
]
2026-01-21 22:31:03 +01:00
def get_guided_conversation_prompt(
current_stage: str,
empty_slots: List[str],
filled_slots: Dict[str, str],
user_message: str,
conversation_turn: int = 0,
same_topic_turns: int = 0,
all_stages_coverage: Dict[str, Dict] = None,
detected_user_stage: str = "",
2026-01-21 22:31:03 +01:00
) -> str:
"""
生成状态感知的对话提示词
Args:
current_stage: 系统当前跟踪的阶段
empty_slots: 当前阶段未填充的槽位
filled_slots: 当前阶段已填充的槽位
user_message: 用户消息
conversation_turn: 总对话轮数
same_topic_turns: 同一话题的轮数
all_stages_coverage: 所有阶段的覆盖情况 {stage: {total, filled, empty, ratio}}
detected_user_stage: 检测到用户正在谈论的阶段可能和 current_stage 不同
2026-01-21 22:31:03 +01:00
"""
stage_name_map = {
"childhood": "童年时光",
"education": "求学经历",
"career": "职业生涯",
"family": "家庭生活",
"belief": "人生信念",
}
current_stage_name = stage_name_map.get(current_stage, current_stage)
user_stage_name = stage_name_map.get(detected_user_stage, "") if detected_user_stage else ""
# 判断用户是否在聊一个不同于系统当前阶段的话题
user_jumped = detected_user_stage and detected_user_stage != current_stage
# --- 构建当前聊天上下文 ---
2026-01-21 22:31:03 +01:00
# 转换 slot 名称为中文
empty_slots_readable = [SLOT_NAME_MAP.get(s, s) for s in empty_slots]
empty_slots_str = "".join(empty_slots_readable) if empty_slots_readable else "已聊得很充分"
2026-01-21 22:31:03 +01:00
filled_info = []
for key, value in filled_slots.items():
readable_key = SLOT_NAME_MAP.get(key, key)
filled_info.append(f"{readable_key}: {value[:50]}..." if len(value) > 50 else f"{readable_key}: {value}")
filled_slots_str = "\n".join(filled_info) if filled_info else "刚开始聊"
# --- 构建全局进度概览 ---
progress_lines = []
uncovered_stages = []
if all_stages_coverage:
for stage in ["childhood", "education", "career", "family", "belief"]:
cov = all_stages_coverage.get(stage, {})
filled_n = cov.get("filled", 0)
total_n = cov.get("total", 0)
sname = stage_name_map.get(stage, stage)
if filled_n == 0:
progress_lines.append(f" {sname}:还没聊到")
uncovered_stages.append(sname)
elif filled_n < total_n:
progress_lines.append(f" {sname}:聊了一些({filled_n}/{total_n}")
else:
progress_lines.append(f" {sname}:已聊得很充分 ✓")
progress_str = "\n".join(progress_lines) if progress_lines else ""
# --- 动态策略 ---
filled_count = len(filled_slots)
should_switch_topic = same_topic_turns >= 3 or (filled_count >= 2 and same_topic_turns >= 2)
should_lighten_mood = conversation_turn > 0 and conversation_turn % 5 == 0
should_try_new_stage = filled_count >= 3 and len(empty_slots) <= 2
# 获取相关阶段
related_stages = STAGE_RELATED_TOPICS.get(current_stage, [])
related_stages_str = "".join([stage_name_map.get(s, s) for s in related_stages])
# 选择回应风格
style = random.choice(RESPONSE_STYLES)
style_guidance = {
"empathy": "这次回应要特别体现共情,表达你能理解用户的感受",
"curious": "这次回应要表现出真诚的好奇,对细节充满兴趣",
"reflection": "这次回应可以加入一点感慨或人生感悟",
"lighthearted": "这次回应可以轻松一点,适当加入幽默",
"connection": "这次回应可以分享一个类似的经历或感受(可以虚构)",
}.get(style, "")
2026-01-21 22:31:03 +01:00
# --- 构建动态指导 ---
dynamic_guidance = ""
if user_jumped:
dynamic_guidance += f"""
- **用户正在聊{user_stage_name}的话题跟着他/她的节奏走不要试图拉回{current_stage_name}**
- 顺着用户的思路帮他/她把这个话题聊深聊透
- 这是很自然的事情人回忆往事经常会跳跃你要做的是陪伴和倾听"""
else:
if should_lighten_mood:
dynamic_guidance += "\n- 聊了一会儿了,可以适当轻松一下,聊点有趣的"
if should_switch_topic and empty_slots_readable:
dynamic_guidance += f"\n- 这个话题聊得差不多了,可以自然转到:{empty_slots_str}"
if should_try_new_stage and related_stages:
dynamic_guidance += f"\n- 如果自然的话,可以尝试聊聊相关的话题,比如{related_stages_str}"
# --- 缺失章节补充提示(仅在用户没有跳转、且当前话题聊得差不多时) ---
uncovered_hint = ""
if not user_jumped and uncovered_stages and should_try_new_stage:
uncovered_hint = f"\n- 还没聊到的人生阶段有:{''.join(uncovered_stages)},如果聊天中有自然的契机,可以轻轻带一句,但不要刻意"
# --- 组合 prompt ---
# 根据是否跳转,调整主题描述
if user_jumped:
topic_desc = f"你们原本在聊「{current_stage_name}」,但用户自然地聊到了「{user_stage_name}」的内容"
else:
topic_desc = f"你们聊到了「{current_stage_name}」这个话题"
prompt = f"""你是「岁月知己」,用户的老朋友,正在和他/她聊人生故事。{topic_desc}
## 已经聊到的内容({current_stage_name}
2026-01-21 22:31:03 +01:00
{filled_slots_str}
## 还可以聊的方向({current_stage_name}
2026-01-21 22:31:03 +01:00
{empty_slots_str}
## 整体进度
{progress_str}
2026-01-21 22:31:03 +01:00
## 用户刚才说
"{user_message}"
## 回应风格
{style_guidance}
2026-01-21 22:31:03 +01:00
## 你的任务
1. **回应用户**先对用户说的内容做出真诚回应不是总结而是有温度的反馈
2. **跟随用户**如果用户聊到了其他人生阶段的内容比如从童年跳到工作完全没问题顺着他/她的思路继续聊回忆本来就是跳跃的不要强行拉回某个固定话题
3. **保持自然**不要每次都追问有时候可以分享感受表达好奇或者轻松聊两句
4. **适时引导**跟着用户的节奏聊了几轮后如果有自然的时机可以温和地引向还没聊到的人生阶段但绝不要生硬
5. **追问要具体**如果要追问问具体的细节比如"那时候是什么季节""身边有谁陪着你""当时心里什么感觉"
{dynamic_guidance}{uncovered_hint}
2026-01-21 22:31:03 +01:00
## 回复格式
- 如果内容较多可以分成 2-3 条消息 [SPLIT] 分隔
- 每条消息保持自然像微信聊天一样
- 第一条消息是回应第二条可以是追问或者换话题
- 如果内容简单一条消息即可
2026-01-21 22:31:03 +01:00
## 严格禁止
- 禁止输出括号注释思考过程
- 禁止说"我注意到""我想问""让我们聊聊"
- 禁止生硬地问"还有什么想分享的吗"
- 禁止反复追问同一件事
- 禁止每次都以问题结尾
- **禁止在用户聊别的话题时强行拉回之前的话题**
## 好的回应示例
- "哈哈,你这说的让我想起..."轻松
- "这段经历听起来真不容易啊"共情
- "那个年代的xxx确实是这样"理解
- "所以后来怎么样了?"好奇
- "对了你刚才提到xxx那个时候..."换话题
2026-01-21 22:31:03 +01:00
直接输出你要说的话多条消息用 [SPLIT] 分隔"""
return prompt
# 保留向后兼容的函数名
def get_conversation_prompt(current_stage: ConversationStage, covered_topics: List[str], user_latest_response: str) -> str:
"""向后兼容的函数"""
return get_system_prompt(current_stage, covered_topics, user_latest_response)