Du kannst nicht mehr als 25 Themen auswählen Themen müssen mit entweder einem Buchstaben oder einer Ziffer beginnen. Sie können Bindestriche („-“) enthalten und bis zu 35 Zeichen lang sein.

search_app.py 7.8KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188
  1. #
  2. # Copyright 2025 The InfiniFlow Authors. All Rights Reserved.
  3. #
  4. # Licensed under the Apache License, Version 2.0 (the "License");
  5. # you may not use this file except in compliance with the License.
  6. # You may obtain a copy of the License at
  7. #
  8. # http://www.apache.org/licenses/LICENSE-2.0
  9. #
  10. # Unless required by applicable law or agreed to in writing, software
  11. # distributed under the License is distributed on an "AS IS" BASIS,
  12. # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. # See the License for the specific language governing permissions and
  14. # limitations under the License.
  15. #
  16. from flask import request
  17. from flask_login import current_user, login_required
  18. from api import settings
  19. from api.constants import DATASET_NAME_LIMIT
  20. from api.db import StatusEnum
  21. from api.db.db_models import DB
  22. from api.db.services import duplicate_name
  23. from api.db.services.knowledgebase_service import KnowledgebaseService
  24. from api.db.services.search_service import SearchService
  25. from api.db.services.user_service import TenantService, UserTenantService
  26. from api.utils import get_uuid
  27. from api.utils.api_utils import get_data_error_result, get_json_result, not_allowed_parameters, server_error_response, validate_request
  28. @manager.route("/create", methods=["post"]) # noqa: F821
  29. @login_required
  30. @validate_request("name")
  31. def create():
  32. req = request.get_json()
  33. search_name = req["name"]
  34. description = req.get("description", "")
  35. if not isinstance(search_name, str):
  36. return get_data_error_result(message="Search name must be string.")
  37. if search_name.strip() == "":
  38. return get_data_error_result(message="Search name can't be empty.")
  39. if len(search_name.encode("utf-8")) > DATASET_NAME_LIMIT:
  40. return get_data_error_result(message=f"Search name length is {len(search_name)} which is large than {DATASET_NAME_LIMIT}")
  41. e, _ = TenantService.get_by_id(current_user.id)
  42. if not e:
  43. return get_data_error_result(message="Authorizationd identity.")
  44. search_name = search_name.strip()
  45. search_name = duplicate_name(KnowledgebaseService.query, name=search_name, tenant_id=current_user.id, status=StatusEnum.VALID.value)
  46. req["id"] = get_uuid()
  47. req["name"] = search_name
  48. req["description"] = description
  49. req["tenant_id"] = current_user.id
  50. req["created_by"] = current_user.id
  51. with DB.atomic():
  52. try:
  53. if not SearchService.save(**req):
  54. return get_data_error_result()
  55. return get_json_result(data={"search_id": req["id"]})
  56. except Exception as e:
  57. return server_error_response(e)
  58. @manager.route("/update", methods=["post"]) # noqa: F821
  59. @login_required
  60. @validate_request("search_id", "name", "search_config", "tenant_id")
  61. @not_allowed_parameters("id", "created_by", "create_time", "update_time", "create_date", "update_date", "created_by")
  62. def update():
  63. req = request.get_json()
  64. if not isinstance(req["name"], str):
  65. return get_data_error_result(message="Search name must be string.")
  66. if req["name"].strip() == "":
  67. return get_data_error_result(message="Search name can't be empty.")
  68. if len(req["name"].encode("utf-8")) > DATASET_NAME_LIMIT:
  69. return get_data_error_result(message=f"Search name length is {len(req['name'])} which is large than {DATASET_NAME_LIMIT}")
  70. req["name"] = req["name"].strip()
  71. tenant_id = req["tenant_id"]
  72. e, _ = TenantService.get_by_id(tenant_id)
  73. if not e:
  74. return get_data_error_result(message="Authorizationd identity.")
  75. search_id = req["search_id"]
  76. if not SearchService.accessible4deletion(search_id, current_user.id):
  77. return get_json_result(data=False, message="No authorization.", code=settings.RetCode.AUTHENTICATION_ERROR)
  78. try:
  79. search_app = SearchService.query(tenant_id=tenant_id, id=search_id)[0]
  80. if not search_app:
  81. return get_json_result(data=False, message=f"Cannot find search {search_id}", code=settings.RetCode.DATA_ERROR)
  82. if req["name"].lower() != search_app.name.lower() and len(SearchService.query(name=req["name"], tenant_id=tenant_id, status=StatusEnum.VALID.value)) >= 1:
  83. return get_data_error_result(message="Duplicated search name.")
  84. if "search_config" in req:
  85. current_config = search_app.search_config or {}
  86. new_config = req["search_config"]
  87. if not isinstance(new_config, dict):
  88. return get_data_error_result(message="search_config must be a JSON object")
  89. updated_config = {**current_config, **new_config}
  90. req["search_config"] = updated_config
  91. req.pop("search_id", None)
  92. req.pop("tenant_id", None)
  93. updated = SearchService.update_by_id(search_id, req)
  94. if not updated:
  95. return get_data_error_result(message="Failed to update search")
  96. e, updated_search = SearchService.get_by_id(search_id)
  97. if not e:
  98. return get_data_error_result(message="Failed to fetch updated search")
  99. return get_json_result(data=updated_search.to_dict())
  100. except Exception as e:
  101. return server_error_response(e)
  102. @manager.route("/detail", methods=["GET"]) # noqa: F821
  103. @login_required
  104. def detail():
  105. search_id = request.args["search_id"]
  106. try:
  107. tenants = UserTenantService.query(user_id=current_user.id)
  108. for tenant in tenants:
  109. if SearchService.query(tenant_id=tenant.tenant_id, id=search_id):
  110. break
  111. else:
  112. return get_json_result(data=False, message="Has no permission for this operation.", code=settings.RetCode.OPERATING_ERROR)
  113. search = SearchService.get_detail(search_id)
  114. if not search:
  115. return get_data_error_result(message="Can't find this Search App!")
  116. return get_json_result(data=search)
  117. except Exception as e:
  118. return server_error_response(e)
  119. @manager.route("/list", methods=["POST"]) # noqa: F821
  120. @login_required
  121. def list_search_app():
  122. keywords = request.args.get("keywords", "")
  123. page_number = int(request.args.get("page", 0))
  124. items_per_page = int(request.args.get("page_size", 0))
  125. orderby = request.args.get("orderby", "create_time")
  126. if request.args.get("desc", "true").lower() == "false":
  127. desc = False
  128. else:
  129. desc = True
  130. req = request.get_json()
  131. owner_ids = req.get("owner_ids", [])
  132. try:
  133. if not owner_ids:
  134. tenants = TenantService.get_joined_tenants_by_user_id(current_user.id)
  135. tenants = [m["tenant_id"] for m in tenants]
  136. search_apps, total = SearchService.get_by_tenant_ids(tenants, current_user.id, page_number, items_per_page, orderby, desc, keywords)
  137. else:
  138. tenants = owner_ids
  139. search_apps, total = SearchService.get_by_tenant_ids(tenants, current_user.id, 0, 0, orderby, desc, keywords)
  140. search_apps = [search_app for search_app in search_apps if search_app["tenant_id"] in tenants]
  141. total = len(search_apps)
  142. if page_number and items_per_page:
  143. search_apps = search_apps[(page_number - 1) * items_per_page : page_number * items_per_page]
  144. return get_json_result(data={"search_apps": search_apps, "total": total})
  145. except Exception as e:
  146. return server_error_response(e)
  147. @manager.route("/rm", methods=["post"]) # noqa: F821
  148. @login_required
  149. @validate_request("search_id")
  150. def rm():
  151. req = request.get_json()
  152. search_id = req["search_id"]
  153. if not SearchService.accessible4deletion(search_id, current_user.id):
  154. return get_json_result(data=False, message="No authorization.", code=settings.RetCode.AUTHENTICATION_ERROR)
  155. try:
  156. if not SearchService.delete_by_id(search_id):
  157. return get_data_error_result(message=f"Failed to delete search App {search_id}")
  158. return get_json_result(data=True)
  159. except Exception as e:
  160. return server_error_response(e)