conversation.py 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129
  1. from flask_restful import marshal_with, reqparse # type: ignore
  2. from flask_restful.inputs import int_range # type: ignore
  3. from sqlalchemy.orm import Session
  4. from werkzeug.exceptions import NotFound
  5. from controllers.web import api
  6. from controllers.web.error import NotChatAppError
  7. from controllers.web.wraps import WebApiResource
  8. from core.app.entities.app_invoke_entities import InvokeFrom
  9. from extensions.ext_database import db
  10. from fields.conversation_fields import conversation_infinite_scroll_pagination_fields, simple_conversation_fields
  11. from libs.helper import uuid_value
  12. from models.model import AppMode
  13. from services.conversation_service import ConversationService
  14. from services.errors.conversation import ConversationNotExistsError, LastConversationNotExistsError
  15. from services.web_conversation_service import WebConversationService
  16. class ConversationListApi(WebApiResource):
  17. @marshal_with(conversation_infinite_scroll_pagination_fields)
  18. def get(self, app_model, end_user):
  19. app_mode = AppMode.value_of(app_model.mode)
  20. if app_mode not in {AppMode.CHAT, AppMode.AGENT_CHAT, AppMode.ADVANCED_CHAT}:
  21. raise NotChatAppError()
  22. parser = reqparse.RequestParser()
  23. parser.add_argument("last_id", type=uuid_value, location="args")
  24. parser.add_argument("limit", type=int_range(1, 100), required=False, default=20, location="args")
  25. parser.add_argument("pinned", type=str, choices=["true", "false", None], location="args")
  26. parser.add_argument(
  27. "sort_by",
  28. type=str,
  29. choices=["created_at", "-created_at", "updated_at", "-updated_at"],
  30. required=False,
  31. default="-updated_at",
  32. location="args",
  33. )
  34. args = parser.parse_args()
  35. pinned = None
  36. if "pinned" in args and args["pinned"] is not None:
  37. pinned = args["pinned"] == "true"
  38. try:
  39. with Session(db.engine) as session:
  40. return WebConversationService.pagination_by_last_id(
  41. session=session,
  42. app_model=app_model,
  43. user=end_user,
  44. last_id=args["last_id"],
  45. limit=args["limit"],
  46. invoke_from=InvokeFrom.WEB_APP,
  47. pinned=pinned,
  48. sort_by=args["sort_by"],
  49. )
  50. except LastConversationNotExistsError:
  51. raise NotFound("Last Conversation Not Exists.")
  52. class ConversationApi(WebApiResource):
  53. def delete(self, app_model, end_user, c_id):
  54. app_mode = AppMode.value_of(app_model.mode)
  55. if app_mode not in {AppMode.CHAT, AppMode.AGENT_CHAT, AppMode.ADVANCED_CHAT}:
  56. raise NotChatAppError()
  57. conversation_id = str(c_id)
  58. try:
  59. ConversationService.delete(app_model, conversation_id, end_user)
  60. except ConversationNotExistsError:
  61. raise NotFound("Conversation Not Exists.")
  62. WebConversationService.unpin(app_model, conversation_id, end_user)
  63. return {"result": "success"}, 204
  64. class ConversationRenameApi(WebApiResource):
  65. @marshal_with(simple_conversation_fields)
  66. def post(self, app_model, end_user, c_id):
  67. app_mode = AppMode.value_of(app_model.mode)
  68. if app_mode not in {AppMode.CHAT, AppMode.AGENT_CHAT, AppMode.ADVANCED_CHAT}:
  69. raise NotChatAppError()
  70. conversation_id = str(c_id)
  71. parser = reqparse.RequestParser()
  72. parser.add_argument("name", type=str, required=False, location="json")
  73. parser.add_argument("auto_generate", type=bool, required=False, default=False, location="json")
  74. args = parser.parse_args()
  75. try:
  76. return ConversationService.rename(app_model, conversation_id, end_user, args["name"], args["auto_generate"])
  77. except ConversationNotExistsError:
  78. raise NotFound("Conversation Not Exists.")
  79. class ConversationPinApi(WebApiResource):
  80. def patch(self, app_model, end_user, c_id):
  81. app_mode = AppMode.value_of(app_model.mode)
  82. if app_mode not in {AppMode.CHAT, AppMode.AGENT_CHAT, AppMode.ADVANCED_CHAT}:
  83. raise NotChatAppError()
  84. conversation_id = str(c_id)
  85. try:
  86. WebConversationService.pin(app_model, conversation_id, end_user)
  87. except ConversationNotExistsError:
  88. raise NotFound("Conversation Not Exists.")
  89. return {"result": "success"}
  90. class ConversationUnPinApi(WebApiResource):
  91. def patch(self, app_model, end_user, c_id):
  92. app_mode = AppMode.value_of(app_model.mode)
  93. if app_mode not in {AppMode.CHAT, AppMode.AGENT_CHAT, AppMode.ADVANCED_CHAT}:
  94. raise NotChatAppError()
  95. conversation_id = str(c_id)
  96. WebConversationService.unpin(app_model, conversation_id, end_user)
  97. return {"result": "success"}
  98. api.add_resource(ConversationRenameApi, "/conversations/<uuid:c_id>/name", endpoint="web_conversation_name")
  99. api.add_resource(ConversationListApi, "/conversations")
  100. api.add_resource(ConversationApi, "/conversations/<uuid:c_id>")
  101. api.add_resource(ConversationPinApi, "/conversations/<uuid:c_id>/pin")
  102. api.add_resource(ConversationUnPinApi, "/conversations/<uuid:c_id>/unpin")