mongo.py 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144
  1. from pymongo import MongoClient
  2. from datetime import datetime
  3. from dotenv import load_dotenv
  4. import os
  5. load_dotenv()
  6. MONGO_URI = os.getenv("ARK_LOGS_MONGO_URI")
  7. client = MongoClient(MONGO_URI, serverSelectionTimeoutMS=5000)
  8. # 数据库
  9. db = client["arklogs"]
  10. # 豆包大模型的对话日志
  11. chat_logs = db["chat_logs"]
  12. # 聊天历史记录
  13. chat_history_col = db["chat_history"]
  14. # 兴趣圈集合
  15. circle_prompts = db["circle_prompt"]
  16. # 带下划线的表示私有方法(Private)
  17. def _ensure_index():
  18. try:
  19. chat_logs.create_index([("username", 1), ("asked_at", -1)])
  20. chat_history_col.create_index([("username", 1), ("timestamp", -1)])
  21. except Exception:
  22. pass
  23. def save_chat_log(
  24. username: str,
  25. question: str,
  26. stream_mode: bool,
  27. raw_response: str = None,
  28. status: str = "success",
  29. error: str = None,
  30. ):
  31. """
  32. 保存聊天原始响应日志到 MongoDB
  33. Args:
  34. username: 提问人
  35. question: 提问的问题
  36. stream_mode: 回答方式(流式或非流式)
  37. raw_response: API 原始响应的 repr 字符串
  38. status: 响应状态 success | error
  39. error: 异常时的错误信息
  40. """
  41. try:
  42. _ensure_index()
  43. chat_logs.insert_one({
  44. "username": username,
  45. "question": question,
  46. "stream_mode": stream_mode,
  47. "raw_response": raw_response,
  48. "status": status,
  49. "error": error,
  50. "asked_at": datetime.now(),
  51. })
  52. except Exception as e:
  53. print(f"MongoDB 日志写入失败: {e}")
  54. def save_chat_history(
  55. username: str,
  56. role: str,
  57. content: str,
  58. timestamp: datetime,
  59. response_id: str = None,
  60. thinking: str = None,
  61. searching: str = None,
  62. ):
  63. try:
  64. _ensure_index()
  65. chat_history_col.insert_one({
  66. "username": username,
  67. "role": role,
  68. "content": content,
  69. "thinking": thinking,
  70. "searching": searching,
  71. "response_id": response_id,
  72. "timestamp": timestamp,
  73. })
  74. except Exception as e:
  75. print(f"MongoDB 聊天历史写入失败: {e}")
  76. def get_chat_history(username: str) -> list:
  77. try:
  78. docs = chat_history_col.find(
  79. {"username": username},
  80. {"_id": 0}
  81. ).sort("timestamp", 1)
  82. return list(docs)
  83. except Exception as e:
  84. print(f"MongoDB 聊天历史读取失败: {e}")
  85. return []
  86. def get_last_response_id(username: str) -> str | None:
  87. try:
  88. doc = chat_history_col.find_one(
  89. {"username": username, "role": "assistant", "response_id": {"$ne": None}},
  90. {"response_id": 1, "_id": 0},
  91. sort=[("timestamp", -1)]
  92. )
  93. return doc["response_id"] if doc else None
  94. except Exception as e:
  95. print(f"MongoDB 查询 response_id 失败: {e}")
  96. return None
  97. def delete_chat_history(username: str) -> int:
  98. try:
  99. result = chat_history_col.delete_many({"username": username})
  100. return result.deleted_count
  101. except Exception as e:
  102. print(f"MongoDB 聊天历史删除失败: {e}")
  103. return 0
  104. _DEFAULT_PROMPT_CONFIG = {
  105. "name": "兴趣圈",
  106. "role": "活跃用户",
  107. "style": "自然亲切,有活人感",
  108. "keywords": [],
  109. "forbidden": [],
  110. }
  111. def get_circle_prompt(app_name: str) -> dict:
  112. try:
  113. doc = circle_prompts.find_one({"appName": app_name})
  114. return doc if doc else _DEFAULT_PROMPT_CONFIG
  115. except Exception:
  116. return _DEFAULT_PROMPT_CONFIG
  117. def upsert_circle_prompt(data: dict) -> None:
  118. circle_prompts.update_one(
  119. {"appName": data["appName"]},
  120. {"$set": data},
  121. upsert=True,
  122. )