external.py 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264
  1. from flask import request
  2. from flask_login import current_user
  3. from flask_restful import Resource, marshal, reqparse
  4. from werkzeug.exceptions import Forbidden, InternalServerError, NotFound
  5. import services
  6. from controllers.console import api
  7. from controllers.console.datasets.error import DatasetNameDuplicateError
  8. from controllers.console.setup import setup_required
  9. from controllers.console.wraps import account_initialization_required
  10. from fields.dataset_fields import dataset_detail_fields
  11. from libs.login import login_required
  12. from services.dataset_service import DatasetService
  13. from services.external_knowledge_service import ExternalDatasetService
  14. from services.hit_testing_service import HitTestingService
  15. from services.knowledge_service import ExternalDatasetTestService
  16. def _validate_name(name):
  17. if not name or len(name) < 1 or len(name) > 100:
  18. raise ValueError("Name must be between 1 to 100 characters.")
  19. return name
  20. def _validate_description_length(description):
  21. if description and len(description) > 400:
  22. raise ValueError("Description cannot exceed 400 characters.")
  23. return description
  24. class ExternalApiTemplateListApi(Resource):
  25. @setup_required
  26. @login_required
  27. @account_initialization_required
  28. def get(self):
  29. page = request.args.get("page", default=1, type=int)
  30. limit = request.args.get("limit", default=20, type=int)
  31. search = request.args.get("keyword", default=None, type=str)
  32. external_knowledge_apis, total = ExternalDatasetService.get_external_knowledge_apis(
  33. page, limit, current_user.current_tenant_id, search
  34. )
  35. response = {
  36. "data": [item.to_dict() for item in external_knowledge_apis],
  37. "has_more": len(external_knowledge_apis) == limit,
  38. "limit": limit,
  39. "total": total,
  40. "page": page,
  41. }
  42. return response, 200
  43. @setup_required
  44. @login_required
  45. @account_initialization_required
  46. def post(self):
  47. parser = reqparse.RequestParser()
  48. parser.add_argument(
  49. "name",
  50. nullable=False,
  51. required=True,
  52. help="Name is required. Name must be between 1 to 100 characters.",
  53. type=_validate_name,
  54. )
  55. parser.add_argument(
  56. "settings",
  57. type=dict,
  58. location="json",
  59. nullable=False,
  60. required=True,
  61. )
  62. args = parser.parse_args()
  63. ExternalDatasetService.validate_api_list(args["settings"])
  64. # The role of the current user in the ta table must be admin, owner, or editor, or dataset_operator
  65. if not current_user.is_dataset_editor:
  66. raise Forbidden()
  67. try:
  68. external_knowledge_api = ExternalDatasetService.create_external_knowledge_api(
  69. tenant_id=current_user.current_tenant_id, user_id=current_user.id, args=args
  70. )
  71. except services.errors.dataset.DatasetNameDuplicateError:
  72. raise DatasetNameDuplicateError()
  73. return external_knowledge_api.to_dict(), 201
  74. class ExternalApiTemplateApi(Resource):
  75. @setup_required
  76. @login_required
  77. @account_initialization_required
  78. def get(self, external_knowledge_api_id):
  79. external_knowledge_api_id = str(external_knowledge_api_id)
  80. external_knowledge_api = ExternalDatasetService.get_external_knowledge_api(external_knowledge_api_id)
  81. if external_knowledge_api is None:
  82. raise NotFound("API template not found.")
  83. return external_knowledge_api.to_dict(), 200
  84. @setup_required
  85. @login_required
  86. @account_initialization_required
  87. def patch(self, external_knowledge_api_id):
  88. external_knowledge_api_id = str(external_knowledge_api_id)
  89. parser = reqparse.RequestParser()
  90. parser.add_argument(
  91. "name",
  92. nullable=False,
  93. required=True,
  94. help="type is required. Name must be between 1 to 100 characters.",
  95. type=_validate_name,
  96. )
  97. parser.add_argument(
  98. "settings",
  99. type=dict,
  100. location="json",
  101. nullable=False,
  102. required=True,
  103. )
  104. args = parser.parse_args()
  105. ExternalDatasetService.validate_api_list(args["settings"])
  106. external_knowledge_api = ExternalDatasetService.update_external_knowledge_api(
  107. tenant_id=current_user.current_tenant_id,
  108. user_id=current_user.id,
  109. external_knowledge_api_id=external_knowledge_api_id,
  110. args=args,
  111. )
  112. return external_knowledge_api.to_dict(), 200
  113. @setup_required
  114. @login_required
  115. @account_initialization_required
  116. def delete(self, external_knowledge_api_id):
  117. external_knowledge_api_id = str(external_knowledge_api_id)
  118. # The role of the current user in the ta table must be admin, owner, or editor
  119. if not current_user.is_editor or current_user.is_dataset_operator:
  120. raise Forbidden()
  121. ExternalDatasetService.delete_external_knowledge_api(current_user.current_tenant_id, external_knowledge_api_id)
  122. return {"result": "success"}, 200
  123. class ExternalApiUseCheckApi(Resource):
  124. @setup_required
  125. @login_required
  126. @account_initialization_required
  127. def get(self, external_knowledge_api_id):
  128. external_knowledge_api_id = str(external_knowledge_api_id)
  129. external_knowledge_api_is_using, count = ExternalDatasetService.external_knowledge_api_use_check(
  130. external_knowledge_api_id
  131. )
  132. return {"is_using": external_knowledge_api_is_using, "count": count}, 200
  133. class ExternalDatasetCreateApi(Resource):
  134. @setup_required
  135. @login_required
  136. @account_initialization_required
  137. def post(self):
  138. # The role of the current user in the ta table must be admin, owner, or editor
  139. if not current_user.is_editor:
  140. raise Forbidden()
  141. parser = reqparse.RequestParser()
  142. parser.add_argument("external_knowledge_api_id", type=str, required=True, nullable=False, location="json")
  143. parser.add_argument("external_knowledge_id", type=str, required=True, nullable=False, location="json")
  144. parser.add_argument(
  145. "name",
  146. nullable=False,
  147. required=True,
  148. help="name is required. Name must be between 1 to 100 characters.",
  149. type=_validate_name,
  150. )
  151. parser.add_argument("description", type=str, required=False, nullable=True, location="json")
  152. parser.add_argument("external_retrieval_model", type=dict, required=False, location="json")
  153. args = parser.parse_args()
  154. # The role of the current user in the ta table must be admin, owner, or editor, or dataset_operator
  155. if not current_user.is_dataset_editor:
  156. raise Forbidden()
  157. try:
  158. dataset = ExternalDatasetService.create_external_dataset(
  159. tenant_id=current_user.current_tenant_id,
  160. user_id=current_user.id,
  161. args=args,
  162. )
  163. except services.errors.dataset.DatasetNameDuplicateError:
  164. raise DatasetNameDuplicateError()
  165. return marshal(dataset, dataset_detail_fields), 201
  166. class ExternalKnowledgeHitTestingApi(Resource):
  167. @setup_required
  168. @login_required
  169. @account_initialization_required
  170. def post(self, dataset_id):
  171. dataset_id_str = str(dataset_id)
  172. dataset = DatasetService.get_dataset(dataset_id_str)
  173. if dataset is None:
  174. raise NotFound("Dataset not found.")
  175. try:
  176. DatasetService.check_dataset_permission(dataset, current_user)
  177. except services.errors.account.NoPermissionError as e:
  178. raise Forbidden(str(e))
  179. parser = reqparse.RequestParser()
  180. parser.add_argument("query", type=str, location="json")
  181. parser.add_argument("external_retrieval_model", type=dict, required=False, location="json")
  182. args = parser.parse_args()
  183. HitTestingService.hit_testing_args_check(args)
  184. try:
  185. response = HitTestingService.external_retrieve(
  186. dataset=dataset,
  187. query=args["query"],
  188. account=current_user,
  189. external_retrieval_model=args["external_retrieval_model"],
  190. )
  191. return response
  192. except Exception as e:
  193. raise InternalServerError(str(e))
  194. class BedrockRetrievalApi(Resource):
  195. # this api is only for internal testing
  196. def post(self):
  197. parser = reqparse.RequestParser()
  198. parser.add_argument("retrieval_setting", nullable=False, required=True, type=dict, location="json")
  199. parser.add_argument(
  200. "query",
  201. nullable=False,
  202. required=True,
  203. type=str,
  204. )
  205. parser.add_argument("knowledge_id", nullable=False, required=True, type=str)
  206. args = parser.parse_args()
  207. # Call the knowledge retrieval service
  208. result = ExternalDatasetTestService.knowledge_retrieval(
  209. args["retrieval_setting"], args["query"], args["knowledge_id"]
  210. )
  211. return result, 200
  212. api.add_resource(ExternalKnowledgeHitTestingApi, "/datasets/<uuid:dataset_id>/external-hit-testing")
  213. api.add_resource(ExternalDatasetCreateApi, "/datasets/external")
  214. api.add_resource(ExternalApiTemplateListApi, "/datasets/external-knowledge-api")
  215. api.add_resource(ExternalApiTemplateApi, "/datasets/external-knowledge-api/<uuid:external_knowledge_api_id>")
  216. api.add_resource(ExternalApiUseCheckApi, "/datasets/external-knowledge-api/<uuid:external_knowledge_api_id>/use-check")
  217. # this api is only for internal test
  218. api.add_resource(BedrockRetrievalApi, "/test/retrieval")