Ви не можете вибрати більше 25 тем Теми мають розпочинатися з літери або цифри, можуть містити дефіси (-) і не повинні перевищувати 35 символів.

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205
  1. #
  2. # Copyright 2024 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. import json
  17. import traceback
  18. from flask import request, Response
  19. from flask_login import login_required, current_user
  20. from api.db.services.canvas_service import CanvasTemplateService, UserCanvasService
  21. from api.settings import RetCode
  22. from api.utils import get_uuid
  23. from api.utils.api_utils import get_json_result, server_error_response, validate_request, get_data_error_result
  24. from agent.canvas import Canvas
  25. from peewee import MySQLDatabase, PostgresqlDatabase
  26. @manager.route('/templates', methods=['GET'])
  27. @login_required
  28. def templates():
  29. return get_json_result(data=[c.to_dict() for c in CanvasTemplateService.get_all()])
  30. @manager.route('/list', methods=['GET'])
  31. @login_required
  32. def canvas_list():
  33. return get_json_result(data=sorted([c.to_dict() for c in \
  34. UserCanvasService.query(user_id=current_user.id)], key=lambda x: x["update_time"]*-1)
  35. )
  36. @manager.route('/rm', methods=['POST'])
  37. @validate_request("canvas_ids")
  38. @login_required
  39. def rm():
  40. for i in request.json["canvas_ids"]:
  41. if not UserCanvasService.query(user_id=current_user.id,id=i):
  42. return get_json_result(
  43. data=False, message='Only owner of canvas authorized for this operation.',
  44. code=RetCode.OPERATING_ERROR)
  45. UserCanvasService.delete_by_id(i)
  46. return get_json_result(data=True)
  47. @manager.route('/set', methods=['POST'])
  48. @validate_request("dsl", "title")
  49. @login_required
  50. def save():
  51. req = request.json
  52. req["user_id"] = current_user.id
  53. if not isinstance(req["dsl"], str):
  54. req["dsl"] = json.dumps(req["dsl"], ensure_ascii=False)
  55. req["dsl"] = json.loads(req["dsl"])
  56. if "id" not in req:
  57. if UserCanvasService.query(user_id=current_user.id, title=req["title"].strip()):
  58. return get_data_error_result(f"{req['title'].strip()} already exists.")
  59. req["id"] = get_uuid()
  60. if not UserCanvasService.save(**req):
  61. return get_data_error_result(message="Fail to save canvas.")
  62. else:
  63. if not UserCanvasService.query(user_id=current_user.id, id=req["id"]):
  64. return get_json_result(
  65. data=False, message='Only owner of canvas authorized for this operation.',
  66. code=RetCode.OPERATING_ERROR)
  67. UserCanvasService.update_by_id(req["id"], req)
  68. return get_json_result(data=req)
  69. @manager.route('/get/<canvas_id>', methods=['GET'])
  70. @login_required
  71. def get(canvas_id):
  72. e, c = UserCanvasService.get_by_id(canvas_id)
  73. if not e:
  74. return get_data_error_result(message="canvas not found.")
  75. return get_json_result(data=c.to_dict())
  76. @manager.route('/completion', methods=['POST'])
  77. @validate_request("id")
  78. @login_required
  79. def run():
  80. req = request.json
  81. stream = req.get("stream", True)
  82. e, cvs = UserCanvasService.get_by_id(req["id"])
  83. if not e:
  84. return get_data_error_result(message="canvas not found.")
  85. if not UserCanvasService.query(user_id=current_user.id, id=req["id"]):
  86. return get_json_result(
  87. data=False, message='Only owner of canvas authorized for this operation.',
  88. code=RetCode.OPERATING_ERROR)
  89. if not isinstance(cvs.dsl, str):
  90. cvs.dsl = json.dumps(cvs.dsl, ensure_ascii=False)
  91. final_ans = {"reference": [], "content": ""}
  92. message_id = req.get("message_id", get_uuid())
  93. try:
  94. canvas = Canvas(cvs.dsl, current_user.id)
  95. if "message" in req:
  96. canvas.messages.append({"role": "user", "content": req["message"], "id": message_id})
  97. canvas.add_user_input(req["message"])
  98. except Exception as e:
  99. return server_error_response(e)
  100. if stream:
  101. def sse():
  102. nonlocal answer, cvs
  103. try:
  104. for ans in canvas.run(stream=True):
  105. if ans.get("running_status"):
  106. yield "data:" + json.dumps({"code": 0, "message": "",
  107. "data": {"answer": ans["content"],
  108. "running_status": True}},
  109. ensure_ascii=False) + "\n\n"
  110. continue
  111. for k in ans.keys():
  112. final_ans[k] = ans[k]
  113. ans = {"answer": ans["content"], "reference": ans.get("reference", [])}
  114. yield "data:" + json.dumps({"code": 0, "message": "", "data": ans}, ensure_ascii=False) + "\n\n"
  115. canvas.messages.append({"role": "assistant", "content": final_ans["content"], "id": message_id})
  116. canvas.history.append(("assistant", final_ans["content"]))
  117. if final_ans.get("reference"):
  118. canvas.reference.append(final_ans["reference"])
  119. cvs.dsl = json.loads(str(canvas))
  120. UserCanvasService.update_by_id(req["id"], cvs.to_dict())
  121. except Exception as e:
  122. cvs.dsl = json.loads(str(canvas))
  123. UserCanvasService.update_by_id(req["id"], cvs.to_dict())
  124. traceback.print_exc()
  125. yield "data:" + json.dumps({"code": 500, "message": str(e),
  126. "data": {"answer": "**ERROR**: " + str(e), "reference": []}},
  127. ensure_ascii=False) + "\n\n"
  128. yield "data:" + json.dumps({"code": 0, "message": "", "data": True}, ensure_ascii=False) + "\n\n"
  129. resp = Response(sse(), mimetype="text/event-stream")
  130. resp.headers.add_header("Cache-control", "no-cache")
  131. resp.headers.add_header("Connection", "keep-alive")
  132. resp.headers.add_header("X-Accel-Buffering", "no")
  133. resp.headers.add_header("Content-Type", "text/event-stream; charset=utf-8")
  134. return resp
  135. for answer in canvas.run(stream=False):
  136. if answer.get("running_status"):
  137. continue
  138. final_ans["content"] = "\n".join(answer["content"]) if "content" in answer else ""
  139. canvas.messages.append({"role": "assistant", "content": final_ans["content"], "id": message_id})
  140. if final_ans.get("reference"):
  141. canvas.reference.append(final_ans["reference"])
  142. cvs.dsl = json.loads(str(canvas))
  143. UserCanvasService.update_by_id(req["id"], cvs.to_dict())
  144. return get_json_result(data={"answer": final_ans["content"], "reference": final_ans.get("reference", [])})
  145. @manager.route('/reset', methods=['POST'])
  146. @validate_request("id")
  147. @login_required
  148. def reset():
  149. req = request.json
  150. try:
  151. e, user_canvas = UserCanvasService.get_by_id(req["id"])
  152. if not e:
  153. return get_data_error_result(message="canvas not found.")
  154. if not UserCanvasService.query(user_id=current_user.id, id=req["id"]):
  155. return get_json_result(
  156. data=False, message='Only owner of canvas authorized for this operation.',
  157. code=RetCode.OPERATING_ERROR)
  158. canvas = Canvas(json.dumps(user_canvas.dsl), current_user.id)
  159. canvas.reset()
  160. req["dsl"] = json.loads(str(canvas))
  161. UserCanvasService.update_by_id(req["id"], {"dsl": req["dsl"]})
  162. return get_json_result(data=req["dsl"])
  163. except Exception as e:
  164. return server_error_response(e)
  165. @manager.route('/test_db_connect', methods=['POST'])
  166. @validate_request("db_type", "database", "username", "host", "port", "password")
  167. @login_required
  168. def test_db_connect():
  169. req = request.json
  170. try:
  171. if req["db_type"] in ["mysql", "mariadb"]:
  172. db = MySQLDatabase(req["database"], user=req["username"], host=req["host"], port=req["port"],
  173. password=req["password"])
  174. elif req["db_type"] == 'postgresql':
  175. db = PostgresqlDatabase(req["database"], user=req["username"], host=req["host"], port=req["port"],
  176. password=req["password"])
  177. db.connect()
  178. db.close()
  179. return get_json_result(data="Database Connection Successful!")
  180. except Exception as e:
  181. return server_error_response(e)