better:重构personinfo,使用Person类和类属性

This commit is contained in:
SengokuCola
2025-08-12 14:33:13 +08:00
parent 1e7f3a92a6
commit ae254de494
21 changed files with 468 additions and 1202 deletions

View File

@@ -1,6 +1,5 @@
from src.common.logger import get_logger
from .person_info import PersonInfoManager, get_person_info_manager
import time
from .person_info import Person
import random
from src.llm_models.utils_model import LLMRequest
from src.config.config import global_config, model_config
@@ -8,11 +7,7 @@ from src.chat.utils.chat_message_builder import build_readable_messages
import json
from json_repair import repair_json
from datetime import datetime
from difflib import SequenceMatcher
import jieba
from sklearn.feature_extraction.text import TfidfVectorizer
from sklearn.metrics.pairwise import cosine_similarity
from typing import List, Dict, Any, Tuple
from typing import List, Dict, Any
from src.chat.utils.prompt_builder import Prompt, global_prompt_manager
import traceback
@@ -52,8 +47,7 @@ def init_prompt():
}}
]
如果没有,就输出none,或返回空数组:
[]
如果没有,就输出空数组:[]
""",
"relation_points",
)
@@ -83,7 +77,9 @@ def init_prompt():
"attitude": 0,
"confidence": 0.5
}}
现在请你输出json:
如果无法看出对方对你的态度,就只输出空数组:[]
现在,请你输出:
""",
"attitude_to_me_prompt",
)
@@ -115,7 +111,9 @@ def init_prompt():
"neuroticism": 0,
"confidence": 0.5
}}
现在请你输出json:
如果无法看出对方的神经质程度,就只输出空数组:[]
现在,请你输出:
""",
"neuroticism_prompt",
)
@@ -124,46 +122,15 @@ class RelationshipManager:
def __init__(self):
self.relationship_llm = LLMRequest(
model_set=model_config.model_task_config.utils, request_type="relationship.person"
) # 用于动作规划
@staticmethod
async def is_known_some_one(platform, user_id):
"""判断是否认识某人"""
person_info_manager = get_person_info_manager()
return await person_info_manager.is_person_known(platform, user_id)
@staticmethod
async def first_knowing_some_one(platform: str, user_id: str, user_nickname: str, user_cardname: str):
"""判断是否认识某人"""
person_id = PersonInfoManager.get_person_id(platform, user_id)
# 生成唯一的 person_name
person_info_manager = get_person_info_manager()
unique_nickname = await person_info_manager._generate_unique_person_name(user_nickname)
data = {
"platform": platform,
"user_id": user_id,
"nickname": user_nickname,
"konw_time": int(time.time()),
"person_name": unique_nickname, # 使用唯一的 person_name
}
# 先创建用户基本信息,使用安全创建方法避免竞态条件
await person_info_manager._safe_create_person_info(person_id=person_id, data=data)
# 更新昵称
await person_info_manager.update_one_field(
person_id=person_id, field_name="nickname", value=user_nickname, data=data
)
# 尝试生成更好的名字
# await person_info_manager.qv_person_name(
# person_id=person_id, user_nickname=user_nickname, user_cardname=user_cardname, user_avatar=user_avatar
# )
)
async def get_points(self,
person_name: str,
nickname: str,
readable_messages: str,
name_mapping: Dict[str, str],
timestamp: float,
current_points: List[Tuple[str, float, str]]):
person_name: str,
nickname: str,
readable_messages: str,
name_mapping: Dict[str, str],
timestamp: float,
person: Person):
alias_str = ", ".join(global_config.bot.alias_names)
current_time = datetime.fromtimestamp(timestamp).strftime("%Y-%m-%d %H:%M:%S")
@@ -198,9 +165,7 @@ class RelationshipManager:
points_data = json.loads(points)
# 只处理正确的格式,错误格式直接跳过
if points_data == "none" or not points_data:
points_list = []
elif isinstance(points_data, str) and points_data.lower() == "none":
if points_data == "none" or not points_data or (isinstance(points_data, str) and points_data.lower() == "none") or (isinstance(points_data, list) and len(points_data) == 0):
points_list = []
elif isinstance(points_data, list):
points_list = [(item["point"], float(item["weight"]), current_time) for item in points_data]
@@ -238,15 +203,15 @@ class RelationshipManager:
return
current_points.extend(points_list)
person.points.extend(points_list)
# 如果points超过10条按权重随机选择多余的条目移动到forgotten_points
if len(current_points) > 20:
if len(person.points) > 20:
# 计算当前时间
current_time = datetime.fromtimestamp(timestamp).strftime("%Y-%m-%d %H:%M:%S")
# 计算每个点的最终权重(原始权重 * 时间权重)
weighted_points = []
for point in current_points:
for point in person.points:
time_weight = self.calculate_time_weight(point[2], current_time)
final_weight = point[1] * time_weight
weighted_points.append((point, final_weight))
@@ -270,16 +235,15 @@ class RelationshipManager:
idx_to_remove = random.randrange(len(remaining_points))
remaining_points[idx_to_remove] = point
return remaining_points
return current_points
person.points = remaining_points
return person
async def get_attitude_to_me(self, person_name, nickname, readable_messages, timestamp, current_attitude):
async def get_attitude_to_me(self, person_name, nickname, readable_messages, timestamp, person: Person):
alias_str = ", ".join(global_config.bot.alias_names)
current_time = datetime.fromtimestamp(timestamp).strftime("%Y-%m-%d %H:%M:%S")
# 解析当前态度值
attitude_parts = current_attitude.split(',')
current_attitude_score = float(attitude_parts[0]) if len(attitude_parts) > 0 else 0.0
total_confidence = float(attitude_parts[1]) if len(attitude_parts) > 1 else 1.0
current_attitude_score = person.attitude_to_me
total_confidence = person.attitude_to_me_confidence
prompt = await global_prompt_manager.format_prompt(
"attitude_to_me_prompt",
@@ -301,23 +265,31 @@ class RelationshipManager:
attitude = repair_json(attitude)
attitude_data = json.loads(attitude)
if attitude_data == "none" or not attitude_data or (isinstance(attitude_data, str) and attitude_data.lower() == "none") or (isinstance(attitude_data, list) and len(attitude_data) == 0):
return ""
# 确保 attitude_data 是字典格式
if not isinstance(attitude_data, dict):
logger.warning(f"LLM返回了错误的JSON格式跳过解析: {type(attitude_data)}, 内容: {attitude_data}")
return ""
attitude_score = attitude_data["attitude"]
confidence = attitude_data["confidence"]
new_confidence = total_confidence + confidence
new_attitude_score = (current_attitude_score * total_confidence + attitude_score * confidence)/new_confidence
person.attitude_to_me = new_attitude_score
person.attitude_to_me_confidence = new_confidence
return f"{new_attitude_score:.3f},{new_confidence:.3f}"
return person
async def get_neuroticism(self, person_name, nickname, readable_messages, timestamp, current_neuroticism):
async def get_neuroticism(self, person_name, nickname, readable_messages, timestamp, person: Person):
alias_str = ", ".join(global_config.bot.alias_names)
current_time = datetime.fromtimestamp(timestamp).strftime("%Y-%m-%d %H:%M:%S")
# 解析当前态度值
neuroticism_parts = current_neuroticism.split(',')
current_neuroticism_score = float(neuroticism_parts[0]) if len(neuroticism_parts) > 0 else 0.0
total_confidence = float(neuroticism_parts[1]) if len(neuroticism_parts) > 1 else 1.0
current_neuroticism_score = person.neuroticism
total_confidence = person.neuroticism_confidence
prompt = await global_prompt_manager.format_prompt(
"neuroticism_prompt",
@@ -339,6 +311,14 @@ class RelationshipManager:
neuroticism = repair_json(neuroticism)
neuroticism_data = json.loads(neuroticism)
if neuroticism_data == "none" or not neuroticism_data or (isinstance(neuroticism_data, str) and neuroticism_data.lower() == "none") or (isinstance(neuroticism_data, list) and len(neuroticism_data) == 0):
return ""
# 确保 neuroticism_data 是字典格式
if not isinstance(neuroticism_data, dict):
logger.warning(f"LLM返回了错误的JSON格式跳过解析: {type(neuroticism_data)}, 内容: {neuroticism_data}")
return ""
neuroticism_score = neuroticism_data["neuroticism"]
confidence = neuroticism_data["confidence"]
@@ -346,8 +326,10 @@ class RelationshipManager:
new_neuroticism_score = (current_neuroticism_score * total_confidence + neuroticism_score * confidence)/new_confidence
person.neuroticism = new_neuroticism_score
person.neuroticism_confidence = new_confidence
return f"{new_neuroticism_score:.3f},{new_confidence:.3f}"
return person
async def update_person_impression(self, person_id, timestamp, bot_engaged_messages: List[Dict[str, Any]]):
@@ -360,21 +342,13 @@ class RelationshipManager:
timestamp: 时间戳 (用于记录交互时间)
bot_engaged_messages: bot参与的消息列表
"""
person_info_manager = get_person_info_manager()
person_name = await person_info_manager.get_value(person_id, "person_name")
nickname = await person_info_manager.get_value(person_id, "nickname")
know_times: float = await person_info_manager.get_value(person_id, "know_times") or 0 # type: ignore
current_points = await person_info_manager.get_value(person_id, "points") or []
attitude_to_me = await person_info_manager.get_value(person_id, "attitude_to_me") or "0,1"
neuroticism = await person_info_manager.get_value(person_id, "neuroticism") or "5,1"
# personality_block =get_individuality().get_personality_prompt(x_person=2, level=2)
# identity_block =get_individuality().get_identity_prompt(x_person=2, level=2)
person = Person(person_id=person_id)
person_name = person.person_name
nickname = person.nickname
know_times: float = person.know_times
user_messages = bot_engaged_messages
current_time = datetime.fromtimestamp(timestamp).strftime("%Y-%m-%d %H:%M:%S")
# 匿名化消息
# 创建用户名称映射
name_mapping = {}
@@ -383,33 +357,23 @@ class RelationshipManager:
# 遍历消息,构建映射
for msg in user_messages:
await person_info_manager.get_or_create_person(
platform=msg.get("chat_info_platform"), # type: ignore
user_id=msg.get("user_id"), # type: ignore
nickname=msg.get("user_nickname"), # type: ignore
user_cardname=msg.get("user_cardname"), # type: ignore
)
replace_user_id: str = msg.get("user_id") # type: ignore
replace_platform: str = msg.get("chat_info_platform") # type: ignore
replace_person_id = PersonInfoManager.get_person_id(replace_platform, replace_user_id)
replace_person_name = await person_info_manager.get_value(replace_person_id, "person_name")
msg_person = Person(user_id=msg.get("user_id"), platform=msg.get("chat_info_platform"))
# 跳过机器人自己
if replace_user_id == global_config.bot.qq_account:
if msg_person.user_id == global_config.bot.qq_account:
name_mapping[f"{global_config.bot.nickname}"] = f"{global_config.bot.nickname}"
continue
# 跳过目标用户
if replace_person_name == person_name:
name_mapping[replace_person_name] = f"{person_name}"
if msg_person.person_name == person_name:
name_mapping[msg_person.person_name] = f"{person_name}"
continue
# 其他用户映射
if replace_person_name not in name_mapping:
if msg_person.person_name not in name_mapping:
if current_user > "Z":
current_user = "A"
user_count += 1
name_mapping[replace_person_name] = f"用户{current_user}{user_count if user_count > 1 else ''}"
name_mapping[msg_person.person_name] = f"用户{current_user}{user_count if user_count > 1 else ''}"
current_user = chr(ord(current_user) + 1)
readable_messages = build_readable_messages(
@@ -420,23 +384,16 @@ class RelationshipManager:
# print(f"original_name: {original_name}, mapped_name: {mapped_name}")
readable_messages = readable_messages.replace(f"{original_name}", f"{mapped_name}")
print(name_mapping)
remaining_points = await self.get_points(person_name, nickname, readable_messages, name_mapping, timestamp, current_points)
attitude_to_me = await self.get_attitude_to_me(person_name, nickname, readable_messages, timestamp, attitude_to_me)
neuroticism = await self.get_neuroticism(person_name, nickname, readable_messages, timestamp, neuroticism)
person = await self.get_points(person_name, nickname, readable_messages, name_mapping, timestamp, person)
person = await self.get_attitude_to_me(person_name, nickname, readable_messages, timestamp, person)
person = await self.get_neuroticism(person_name, nickname, readable_messages, timestamp, person)
# 更新数据库
await person_info_manager.update_one_field(
person_id, "points", json.dumps(remaining_points, ensure_ascii=False, indent=None)
)
await person_info_manager.update_one_field(person_id, "neuroticism", neuroticism)
await person_info_manager.update_one_field(person_id, "attitude_to_me", attitude_to_me)
await person_info_manager.update_one_field(person_id, "know_times", know_times + 1)
await person_info_manager.update_one_field(person_id, "last_know", timestamp)
know_since = await person_info_manager.get_value(person_id, "know_since") or 0
if know_since == 0:
await person_info_manager.update_one_field(person_id, "know_since", timestamp)
person.know_times = know_times + 1
person.last_know = timestamp
person.sync_to_database()