message.py 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285
  1. from typing import List, Optional
  2. from fastapi import HTTPException
  3. from sqlalchemy.ext.asyncio import AsyncSession
  4. from sqlalchemy.orm import Session
  5. from sqlmodel import select
  6. from app.exceptions.exception import ResourceNotFoundError
  7. from app.models import MessageFile
  8. from app.models.message import Message, MessageCreate, MessageUpdate
  9. from app.services.thread.thread import ThreadService
  10. from app.models.token_relation import RelationType
  11. from app.providers.auth_provider import auth_policy
  12. from app.schemas.common import DeleteResponse
  13. class MessageService:
  14. @staticmethod
  15. def format_message_content(message_create: MessageCreate) -> List:
  16. content = []
  17. if isinstance(message_create.content, str):
  18. content.append(
  19. {
  20. "type": "text",
  21. "text": {"value": message_create.content, "annotations": []},
  22. }
  23. )
  24. elif isinstance(message_create.content, list):
  25. for msg in message_create.content:
  26. if msg.get("type") == "text":
  27. msg_value = msg.get("text")
  28. content.append(
  29. {
  30. "type": "text",
  31. "text": {"value": msg_value, "annotations": []},
  32. }
  33. )
  34. elif msg.get("type") == "image_file" or msg.get("type") == "image_url":
  35. content.append(msg)
  36. elif msg.get("type") == "input_audio":
  37. content.append(msg)
  38. return content
  39. @staticmethod
  40. def new_message(
  41. *, session: Session, content, role, assistant_id, thread_id, run_id
  42. ) -> Message:
  43. message = Message(
  44. content=[{"type": "text", "text": {"value": content, "annotations": []}}],
  45. role=role,
  46. assistant_id=assistant_id,
  47. thread_id=thread_id,
  48. run_id=run_id,
  49. )
  50. session.add(message)
  51. session.commit()
  52. session.refresh(message)
  53. return message
  54. @staticmethod
  55. def get_message_list(*, session: Session, thread_id) -> List[Message]:
  56. statement = (
  57. select(Message)
  58. .where(Message.thread_id == thread_id)
  59. .order_by(Message.created_at)
  60. )
  61. return session.execute(statement).scalars().all()
  62. @staticmethod
  63. async def create_message(
  64. *, session: AsyncSession, body: MessageCreate, thread_id: str
  65. ) -> Message:
  66. # get thread
  67. thread = await ThreadService.get_thread(session=session, thread_id=thread_id)
  68. print(
  69. "create_messagecreate_messagecreate_messagecreate_messagecreate_messagecreate_message"
  70. )
  71. # print(thread)
  72. # print(body)
  73. # TODO message annotations
  74. body_file_ids = body.file_ids
  75. if body.attachments:
  76. body_file_ids = [a.get("file_id") for a in body.attachments]
  77. # print(body_file_ids)
  78. if body_file_ids:
  79. thread_file_ids = []
  80. if thread.tool_resources and "file_search" in thread.tool_resources:
  81. thread_file_ids = (
  82. thread.tool_resources.get("file_search")
  83. .get("vector_stores")[0]
  84. .get("file_ids")
  85. )
  86. for file_id in body_file_ids:
  87. if file_id not in thread_file_ids:
  88. thread_file_ids.append(file_id)
  89. print(thread_file_ids)
  90. # if thread_file_ids:
  91. if not thread.tool_resources:
  92. thread.tool_resources = {}
  93. if "file_search" not in thread.tool_resources:
  94. thread.tool_resources["file_search"] = {
  95. "vector_stores": [{"file_ids": []}]
  96. }
  97. thread.tool_resources.get("file_search").get("vector_stores")[0][
  98. "file_ids"
  99. ] = thread_file_ids
  100. setattr(thread, "tool_resources", thread.tool_resources)
  101. # thread.tool_resources = thread.tool_resources
  102. print(thread)
  103. session.add(thread)
  104. await session.commit()
  105. await session.refresh(thread)
  106. # session.add(thread)
  107. # await session.commit()
  108. # await session.refresh(thread)
  109. content = MessageService.format_message_content(body)
  110. db_message = Message.model_validate(
  111. body.model_dump(by_alias=True),
  112. update={"thread_id": thread_id, "content": content},
  113. from_attributes=True,
  114. )
  115. session.add(db_message)
  116. await session.commit()
  117. await session.refresh(db_message)
  118. return db_message
  119. @staticmethod
  120. def get_message_sync(
  121. *, session: Session, thread_id: str, message_id: str
  122. ) -> Message:
  123. statement = (
  124. select(Message)
  125. .where(Message.thread_id == thread_id)
  126. .where(Message.id == message_id)
  127. )
  128. result = session.execute(statement)
  129. message = result.scalars().one_or_none()
  130. if message is None:
  131. raise HTTPException(status_code=404, detail="Message not found")
  132. return message
  133. @staticmethod
  134. def modify_message_sync(
  135. *, session: Session, thread_id: str, message_id: str, body: MessageUpdate
  136. ) -> Message:
  137. if body.content:
  138. body.content = [
  139. {"type": "text", "text": {"value": body.content, "annotations": []}}
  140. ]
  141. # get thread
  142. ThreadService.get_thread_sync(thread_id=thread_id, session=session)
  143. # get message
  144. db_message = MessageService.get_message_sync(
  145. session=session, thread_id=thread_id, message_id=message_id
  146. )
  147. update_data = body.dict(exclude_unset=True)
  148. for key, value in update_data.items():
  149. setattr(db_message, key, value)
  150. session.add(db_message)
  151. session.commit()
  152. session.refresh(db_message)
  153. return db_message
  154. @staticmethod
  155. async def modify_message(
  156. *, session: AsyncSession, thread_id: str, message_id: str, body: MessageUpdate
  157. ) -> Message:
  158. if body.content:
  159. body.content = [
  160. {"type": "text", "text": {"value": body.content, "annotations": []}}
  161. ]
  162. # get thread
  163. await ThreadService.get_thread(thread_id=thread_id, session=session)
  164. # get message
  165. db_message = await MessageService.get_message(
  166. session=session, thread_id=thread_id, message_id=message_id
  167. )
  168. update_data = body.dict(exclude_unset=True)
  169. for key, value in update_data.items():
  170. setattr(db_message, key, value)
  171. session.add(db_message)
  172. await session.commit()
  173. await session.refresh(db_message)
  174. return db_message
  175. @staticmethod
  176. async def delete_message(
  177. *, session: AsyncSession, thread_id: str, message_id: str
  178. ) -> Message:
  179. message = await MessageService.get_message(
  180. session=session, thread_id=thread_id, message_id=message_id
  181. )
  182. await session.delete(message)
  183. await auth_policy.delete_token_rel(
  184. session=session, relation_type=RelationType.Message, relation_id=message_id
  185. )
  186. await session.commit()
  187. return DeleteResponse(id=message_id, object="message.deleted", deleted=True)
  188. @staticmethod
  189. async def get_message(
  190. *, session: AsyncSession, thread_id: str, message_id: str
  191. ) -> Message:
  192. statement = (
  193. select(Message)
  194. .where(Message.thread_id == thread_id)
  195. .where(Message.id == message_id)
  196. )
  197. result = await session.execute(statement)
  198. message = result.scalars().one_or_none()
  199. if message is None:
  200. raise HTTPException(status_code=404, detail="Message not found")
  201. return message
  202. @staticmethod
  203. async def get_message_file(
  204. *, session: AsyncSession, thread_id: str, message_id: str, file_id: str
  205. ) -> MessageFile:
  206. await MessageService.get_message(
  207. session=session, thread_id=thread_id, message_id=message_id
  208. )
  209. # get message files
  210. statement = (
  211. select(MessageFile)
  212. .where(MessageFile.id == file_id)
  213. .where(MessageFile.message_id == message_id)
  214. )
  215. result = await session.execute(statement)
  216. msg_file = result.scalars().one_or_none()
  217. if msg_file is None:
  218. raise ResourceNotFoundError(message="Message file not found")
  219. return msg_file
  220. @staticmethod
  221. async def copy_messages(
  222. *,
  223. session: AsyncSession,
  224. from_thread_id: str,
  225. to_thread_id: str,
  226. end_message_id: str
  227. ):
  228. """
  229. copy thread messages to another thread
  230. """
  231. statement = select(Message).where(Message.thread_id == from_thread_id)
  232. if end_message_id:
  233. statement = statement.where(Message.id <= end_message_id)
  234. result = await session.execute(statement.order_by(Message.id))
  235. original_messages = result.scalars().all()
  236. for original_message in original_messages:
  237. new_message = Message(
  238. thread_id=to_thread_id,
  239. **original_message.model_dump(
  240. exclude={"id", "created_at", "updated_at", "thread_id"}
  241. ),
  242. )
  243. session.add(new_message)
  244. await session.commit()
  245. @staticmethod
  246. async def create_messages(
  247. *,
  248. session: AsyncSession,
  249. thread_id: str,
  250. run_id: str,
  251. assistant_id: str,
  252. messages: list
  253. ):
  254. for original_message in messages:
  255. content = MessageService.format_message_content(original_message)
  256. new_message = Message.model_validate(
  257. original_message.model_dump(by_alias=True),
  258. update={
  259. "thread_id": thread_id,
  260. "run_id": run_id,
  261. "assistant_id": assistant_id,
  262. "content": content,
  263. "role": original_message.role,
  264. },
  265. )
  266. session.add(new_message)