message.py 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145
  1. # -*- coding:utf-8 -*-
  2. import logging
  3. import services
  4. from controllers.console import api
  5. from controllers.console.app.error import (CompletionRequestError, ProviderModelCurrentlyNotSupportError,
  6. ProviderNotInitializeError, ProviderQuotaExceededError)
  7. from controllers.console.explore.error import AppSuggestedQuestionsAfterAnswerDisabledError
  8. from controllers.console.universal_chat.wraps import UniversalChatResource
  9. from core.errors.error import ModelCurrentlyNotSupportError, ProviderTokenNotInitError, QuotaExceededError
  10. from core.model_runtime.errors.invoke import InvokeError
  11. from flask_login import current_user
  12. from flask_restful import fields, marshal_with, reqparse
  13. from flask_restful.inputs import int_range
  14. from libs.helper import TimestampField, uuid_value
  15. from services.errors.conversation import ConversationNotExistsError
  16. from services.errors.message import MessageNotExistsError, SuggestedQuestionsAfterAnswerDisabledError
  17. from services.message_service import MessageService
  18. from werkzeug.exceptions import InternalServerError, NotFound
  19. class UniversalChatMessageListApi(UniversalChatResource):
  20. feedback_fields = {
  21. 'rating': fields.String
  22. }
  23. agent_thought_fields = {
  24. 'id': fields.String,
  25. 'chain_id': fields.String,
  26. 'message_id': fields.String,
  27. 'position': fields.Integer,
  28. 'thought': fields.String,
  29. 'tool': fields.String,
  30. 'tool_input': fields.String,
  31. 'created_at': TimestampField
  32. }
  33. retriever_resource_fields = {
  34. 'id': fields.String,
  35. 'message_id': fields.String,
  36. 'position': fields.Integer,
  37. 'dataset_id': fields.String,
  38. 'dataset_name': fields.String,
  39. 'document_id': fields.String,
  40. 'document_name': fields.String,
  41. 'data_source_type': fields.String,
  42. 'segment_id': fields.String,
  43. 'score': fields.Float,
  44. 'hit_count': fields.Integer,
  45. 'word_count': fields.Integer,
  46. 'segment_position': fields.Integer,
  47. 'index_node_hash': fields.String,
  48. 'content': fields.String,
  49. 'created_at': TimestampField
  50. }
  51. message_fields = {
  52. 'id': fields.String,
  53. 'conversation_id': fields.String,
  54. 'inputs': fields.Raw,
  55. 'query': fields.String,
  56. 'answer': fields.String,
  57. 'feedback': fields.Nested(feedback_fields, attribute='user_feedback', allow_null=True),
  58. 'retriever_resources': fields.List(fields.Nested(retriever_resource_fields)),
  59. 'created_at': TimestampField,
  60. 'agent_thoughts': fields.List(fields.Nested(agent_thought_fields))
  61. }
  62. message_infinite_scroll_pagination_fields = {
  63. 'limit': fields.Integer,
  64. 'has_more': fields.Boolean,
  65. 'data': fields.List(fields.Nested(message_fields))
  66. }
  67. @marshal_with(message_infinite_scroll_pagination_fields)
  68. def get(self, universal_app):
  69. app_model = universal_app
  70. parser = reqparse.RequestParser()
  71. parser.add_argument('conversation_id', required=True, type=uuid_value, location='args')
  72. parser.add_argument('first_id', type=uuid_value, location='args')
  73. parser.add_argument('limit', type=int_range(1, 100), required=False, default=20, location='args')
  74. args = parser.parse_args()
  75. try:
  76. return MessageService.pagination_by_first_id(app_model, current_user,
  77. args['conversation_id'], args['first_id'], args['limit'])
  78. except services.errors.conversation.ConversationNotExistsError:
  79. raise NotFound("Conversation Not Exists.")
  80. except services.errors.message.FirstMessageNotExistsError:
  81. raise NotFound("First Message Not Exists.")
  82. class UniversalChatMessageFeedbackApi(UniversalChatResource):
  83. def post(self, universal_app, message_id):
  84. app_model = universal_app
  85. message_id = str(message_id)
  86. parser = reqparse.RequestParser()
  87. parser.add_argument('rating', type=str, choices=['like', 'dislike', None], location='json')
  88. args = parser.parse_args()
  89. try:
  90. MessageService.create_feedback(app_model, message_id, current_user, args['rating'])
  91. except services.errors.message.MessageNotExistsError:
  92. raise NotFound("Message Not Exists.")
  93. return {'result': 'success'}
  94. class UniversalChatMessageSuggestedQuestionApi(UniversalChatResource):
  95. def get(self, universal_app, message_id):
  96. app_model = universal_app
  97. message_id = str(message_id)
  98. try:
  99. questions = MessageService.get_suggested_questions_after_answer(
  100. app_model=app_model,
  101. user=current_user,
  102. message_id=message_id
  103. )
  104. except MessageNotExistsError:
  105. raise NotFound("Message not found")
  106. except ConversationNotExistsError:
  107. raise NotFound("Conversation not found")
  108. except SuggestedQuestionsAfterAnswerDisabledError:
  109. raise AppSuggestedQuestionsAfterAnswerDisabledError()
  110. except ProviderTokenNotInitError:
  111. raise ProviderNotInitializeError()
  112. except QuotaExceededError:
  113. raise ProviderQuotaExceededError()
  114. except ModelCurrentlyNotSupportError:
  115. raise ProviderModelCurrentlyNotSupportError()
  116. except InvokeError as e:
  117. raise CompletionRequestError(e.description)
  118. except Exception:
  119. logging.exception("internal server error.")
  120. raise InternalServerError()
  121. return {'data': questions}
  122. api.add_resource(UniversalChatMessageListApi, '/universal-chat/messages')
  123. api.add_resource(UniversalChatMessageFeedbackApi, '/universal-chat/messages/<uuid:message_id>/feedbacks')
  124. api.add_resource(UniversalChatMessageSuggestedQuestionApi, '/universal-chat/messages/<uuid:message_id>/suggested-questions')