You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

canvas_app.py 18KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474
  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 logging
  18. import re
  19. import sys
  20. from functools import partial
  21. import trio
  22. from flask import request, Response
  23. from flask_login import login_required, current_user
  24. from agent.component import LLM
  25. from api.db import FileType
  26. from api.db.services.canvas_service import CanvasTemplateService, UserCanvasService, API4ConversationService
  27. from api.db.services.document_service import DocumentService
  28. from api.db.services.file_service import FileService
  29. from api.db.services.user_service import TenantService
  30. from api.db.services.user_canvas_version import UserCanvasVersionService
  31. from api.settings import RetCode
  32. from api.utils import get_uuid
  33. from api.utils.api_utils import get_json_result, server_error_response, validate_request, get_data_error_result
  34. from agent.canvas import Canvas
  35. from peewee import MySQLDatabase, PostgresqlDatabase
  36. from api.db.db_models import APIToken
  37. import time
  38. from api.utils.file_utils import filename_type, read_potential_broken_pdf
  39. from rag.utils.redis_conn import REDIS_CONN
  40. @manager.route('/templates', methods=['GET']) # noqa: F821
  41. @login_required
  42. def templates():
  43. return get_json_result(data=[c.to_dict() for c in CanvasTemplateService.get_all()])
  44. @manager.route('/list', methods=['GET']) # noqa: F821
  45. @login_required
  46. def canvas_list():
  47. return get_json_result(data=sorted([c.to_dict() for c in \
  48. UserCanvasService.query(user_id=current_user.id)], key=lambda x: x["update_time"]*-1)
  49. )
  50. @manager.route('/rm', methods=['POST']) # noqa: F821
  51. @validate_request("canvas_ids")
  52. @login_required
  53. def rm():
  54. for i in request.json["canvas_ids"]:
  55. if not UserCanvasService.accessible(i, current_user.id):
  56. return get_json_result(
  57. data=False, message='Only owner of canvas authorized for this operation.',
  58. code=RetCode.OPERATING_ERROR)
  59. UserCanvasService.delete_by_id(i)
  60. return get_json_result(data=True)
  61. @manager.route('/set', methods=['POST']) # noqa: F821
  62. @validate_request("dsl", "title")
  63. @login_required
  64. def save():
  65. req = request.json
  66. if not isinstance(req["dsl"], str):
  67. req["dsl"] = json.dumps(req["dsl"], ensure_ascii=False)
  68. req["dsl"] = json.loads(req["dsl"])
  69. if "id" not in req:
  70. req["user_id"] = current_user.id
  71. if UserCanvasService.query(user_id=current_user.id, title=req["title"].strip()):
  72. return get_data_error_result(message=f"{req['title'].strip()} already exists.")
  73. req["id"] = get_uuid()
  74. if not UserCanvasService.save(**req):
  75. return get_data_error_result(message="Fail to save canvas.")
  76. else:
  77. if not UserCanvasService.accessible(req["id"], current_user.id):
  78. return get_json_result(
  79. data=False, message='Only owner of canvas authorized for this operation.',
  80. code=RetCode.OPERATING_ERROR)
  81. UserCanvasService.update_by_id(req["id"], req)
  82. # save version
  83. UserCanvasVersionService.insert( user_canvas_id=req["id"], dsl=req["dsl"], title="{0}_{1}".format(req["title"], time.strftime("%Y_%m_%d_%H_%M_%S")))
  84. UserCanvasVersionService.delete_all_versions(req["id"])
  85. return get_json_result(data=req)
  86. @manager.route('/get/<canvas_id>', methods=['GET']) # noqa: F821
  87. @login_required
  88. def get(canvas_id):
  89. if not UserCanvasService.accessible(canvas_id, current_user.id):
  90. return get_data_error_result(message="canvas not found.")
  91. e, c = UserCanvasService.get_by_tenant_id(canvas_id)
  92. return get_json_result(data=c)
  93. @manager.route('/getsse/<canvas_id>', methods=['GET']) # type: ignore # noqa: F821
  94. def getsse(canvas_id):
  95. token = request.headers.get('Authorization').split()
  96. if len(token) != 2:
  97. return get_data_error_result(message='Authorization is not valid!"')
  98. token = token[1]
  99. objs = APIToken.query(beta=token)
  100. if not objs:
  101. return get_data_error_result(message='Authentication error: API key is invalid!"')
  102. tenant_id = objs[0].tenant_id
  103. if not UserCanvasService.query(user_id=tenant_id, id=canvas_id):
  104. return get_json_result(
  105. data=False,
  106. message='Only owner of canvas authorized for this operation.',
  107. code=RetCode.OPERATING_ERROR
  108. )
  109. e, c = UserCanvasService.get_by_id(canvas_id)
  110. if not e or c.user_id != tenant_id:
  111. return get_data_error_result(message="canvas not found.")
  112. return get_json_result(data=c.to_dict())
  113. @manager.route('/completion', methods=['POST']) # noqa: F821
  114. @validate_request("id")
  115. @login_required
  116. def run():
  117. req = request.json
  118. query = req.get("query", "")
  119. files = req.get("files", [])
  120. inputs = req.get("inputs", {})
  121. user_id = req.get("user_id", current_user.id)
  122. if not UserCanvasService.accessible(req["id"], current_user.id):
  123. return get_json_result(
  124. data=False, message='Only owner of canvas authorized for this operation.',
  125. code=RetCode.OPERATING_ERROR)
  126. e, cvs = UserCanvasService.get_by_id(req["id"])
  127. if not e:
  128. return get_data_error_result(message="canvas not found.")
  129. if not isinstance(cvs.dsl, str):
  130. cvs.dsl = json.dumps(cvs.dsl, ensure_ascii=False)
  131. try:
  132. canvas = Canvas(cvs.dsl, current_user.id, req["id"])
  133. except Exception as e:
  134. return server_error_response(e)
  135. def sse():
  136. nonlocal canvas, user_id
  137. try:
  138. for ans in canvas.run(query=query, files=files, user_id=user_id, inputs=inputs):
  139. yield "data:" + json.dumps(ans, ensure_ascii=False) + "\n\n"
  140. cvs.dsl = json.loads(str(canvas))
  141. UserCanvasService.update_by_id(req["id"], cvs.to_dict())
  142. except Exception as e:
  143. logging.exception(e)
  144. yield "data:" + json.dumps({"code": 500, "message": str(e), "data": False}, ensure_ascii=False) + "\n\n"
  145. resp = Response(sse(), mimetype="text/event-stream")
  146. resp.headers.add_header("Cache-control", "no-cache")
  147. resp.headers.add_header("Connection", "keep-alive")
  148. resp.headers.add_header("X-Accel-Buffering", "no")
  149. resp.headers.add_header("Content-Type", "text/event-stream; charset=utf-8")
  150. return resp
  151. @manager.route('/reset', methods=['POST']) # noqa: F821
  152. @validate_request("id")
  153. @login_required
  154. def reset():
  155. req = request.json
  156. if not UserCanvasService.accessible(req["id"], current_user.id):
  157. return get_json_result(
  158. data=False, message='Only owner of canvas authorized for this operation.',
  159. code=RetCode.OPERATING_ERROR)
  160. try:
  161. e, user_canvas = UserCanvasService.get_by_id(req["id"])
  162. if not e:
  163. return get_data_error_result(message="canvas not found.")
  164. canvas = Canvas(json.dumps(user_canvas.dsl), current_user.id)
  165. canvas.reset()
  166. req["dsl"] = json.loads(str(canvas))
  167. UserCanvasService.update_by_id(req["id"], {"dsl": req["dsl"]})
  168. return get_json_result(data=req["dsl"])
  169. except Exception as e:
  170. return server_error_response(e)
  171. @manager.route("/upload/<canvas_id>", methods=["POST"]) # noqa: F821
  172. def upload(canvas_id):
  173. e, cvs = UserCanvasService.get_by_tenant_id(canvas_id)
  174. if not e:
  175. return get_data_error_result(message="canvas not found.")
  176. user_id = cvs["user_id"]
  177. def structured(filename, filetype, blob, content_type):
  178. nonlocal user_id
  179. if filetype == FileType.PDF.value:
  180. blob = read_potential_broken_pdf(blob)
  181. location = get_uuid()
  182. FileService.put_blob(user_id, location, blob)
  183. return {
  184. "id": location,
  185. "name": filename,
  186. "size": sys.getsizeof(blob),
  187. "extension": filename.split(".")[-1].lower(),
  188. "mime_type": content_type,
  189. "created_by": user_id,
  190. "created_at": time.time(),
  191. "preview_url": None
  192. }
  193. if request.args.get("url"):
  194. from crawl4ai import (
  195. AsyncWebCrawler,
  196. BrowserConfig,
  197. CrawlerRunConfig,
  198. DefaultMarkdownGenerator,
  199. PruningContentFilter,
  200. CrawlResult
  201. )
  202. try:
  203. url = request.args.get("url")
  204. filename = re.sub(r"\?.*", "", url.split("/")[-1])
  205. async def adownload():
  206. browser_config = BrowserConfig(
  207. headless=True,
  208. verbose=False,
  209. )
  210. async with AsyncWebCrawler(config=browser_config) as crawler:
  211. crawler_config = CrawlerRunConfig(
  212. markdown_generator=DefaultMarkdownGenerator(
  213. content_filter=PruningContentFilter()
  214. ),
  215. pdf=True,
  216. screenshot=False
  217. )
  218. result: CrawlResult = await crawler.arun(
  219. url=url,
  220. config=crawler_config
  221. )
  222. return result
  223. page = trio.run(adownload())
  224. if page.pdf:
  225. if filename.split(".")[-1].lower() != "pdf":
  226. filename += ".pdf"
  227. return get_json_result(data=structured(filename, "pdf", page.pdf, page.response_headers["content-type"]))
  228. return get_json_result(data=structured(filename, "html", str(page.markdown).encode("utf-8"), page.response_headers["content-type"], user_id))
  229. except Exception as e:
  230. return server_error_response(e)
  231. file = request.files['file']
  232. try:
  233. DocumentService.check_doc_health(user_id, file.filename)
  234. return get_json_result(data=structured(file.filename, filename_type(file.filename), file.read(), file.content_type))
  235. except Exception as e:
  236. return server_error_response(e)
  237. @manager.route('/input_form', methods=['GET']) # noqa: F821
  238. @login_required
  239. def input_form():
  240. cvs_id = request.args.get("id")
  241. cpn_id = request.args.get("component_id")
  242. try:
  243. e, user_canvas = UserCanvasService.get_by_id(cvs_id)
  244. if not e:
  245. return get_data_error_result(message="canvas not found.")
  246. if not UserCanvasService.query(user_id=current_user.id, id=cvs_id):
  247. return get_json_result(
  248. data=False, message='Only owner of canvas authorized for this operation.',
  249. code=RetCode.OPERATING_ERROR)
  250. canvas = Canvas(json.dumps(user_canvas.dsl), current_user.id)
  251. return get_json_result(data=canvas.get_component_input_form(cpn_id))
  252. except Exception as e:
  253. return server_error_response(e)
  254. @manager.route('/debug', methods=['POST']) # noqa: F821
  255. @validate_request("id", "component_id", "params")
  256. @login_required
  257. def debug():
  258. req = request.json
  259. if not UserCanvasService.accessible(req["id"], current_user.id):
  260. return get_json_result(
  261. data=False, message='Only owner of canvas authorized for this operation.',
  262. code=RetCode.OPERATING_ERROR)
  263. try:
  264. e, user_canvas = UserCanvasService.get_by_id(req["id"])
  265. canvas = Canvas(json.dumps(user_canvas.dsl), current_user.id)
  266. canvas.reset()
  267. canvas.message_id = get_uuid()
  268. component = canvas.get_component(req["component_id"])["obj"]
  269. component.reset()
  270. if isinstance(component, LLM):
  271. component.set_debug_inputs(req["params"])
  272. component.invoke(**{k: o["value"] for k,o in req["params"].items()})
  273. outputs = component.output()
  274. for k in outputs.keys():
  275. if isinstance(outputs[k], partial):
  276. txt = ""
  277. for c in outputs[k]():
  278. txt += c
  279. outputs[k] = txt
  280. return get_json_result(data=outputs)
  281. except Exception as e:
  282. return server_error_response(e)
  283. @manager.route('/test_db_connect', methods=['POST']) # noqa: F821
  284. @validate_request("db_type", "database", "username", "host", "port", "password")
  285. @login_required
  286. def test_db_connect():
  287. req = request.json
  288. try:
  289. if req["db_type"] in ["mysql", "mariadb"]:
  290. db = MySQLDatabase(req["database"], user=req["username"], host=req["host"], port=req["port"],
  291. password=req["password"])
  292. elif req["db_type"] == 'postgresql':
  293. db = PostgresqlDatabase(req["database"], user=req["username"], host=req["host"], port=req["port"],
  294. password=req["password"])
  295. elif req["db_type"] == 'mssql':
  296. import pyodbc
  297. connection_string = (
  298. f"DRIVER={{ODBC Driver 17 for SQL Server}};"
  299. f"SERVER={req['host']},{req['port']};"
  300. f"DATABASE={req['database']};"
  301. f"UID={req['username']};"
  302. f"PWD={req['password']};"
  303. )
  304. db = pyodbc.connect(connection_string)
  305. cursor = db.cursor()
  306. cursor.execute("SELECT 1")
  307. cursor.close()
  308. else:
  309. return server_error_response("Unsupported database type.")
  310. if req["db_type"] != 'mssql':
  311. db.connect()
  312. db.close()
  313. return get_json_result(data="Database Connection Successful!")
  314. except Exception as e:
  315. return server_error_response(e)
  316. #api get list version dsl of canvas
  317. @manager.route('/getlistversion/<canvas_id>', methods=['GET']) # noqa: F821
  318. @login_required
  319. def getlistversion(canvas_id):
  320. try:
  321. list =sorted([c.to_dict() for c in UserCanvasVersionService.list_by_canvas_id(canvas_id)], key=lambda x: x["update_time"]*-1)
  322. return get_json_result(data=list)
  323. except Exception as e:
  324. return get_data_error_result(message=f"Error getting history files: {e}")
  325. #api get version dsl of canvas
  326. @manager.route('/getversion/<version_id>', methods=['GET']) # noqa: F821
  327. @login_required
  328. def getversion( version_id):
  329. try:
  330. e, version = UserCanvasVersionService.get_by_id(version_id)
  331. if version:
  332. return get_json_result(data=version.to_dict())
  333. except Exception as e:
  334. return get_json_result(data=f"Error getting history file: {e}")
  335. @manager.route('/listteam', methods=['GET']) # noqa: F821
  336. @login_required
  337. def list_canvas():
  338. keywords = request.args.get("keywords", "")
  339. page_number = int(request.args.get("page", 1))
  340. items_per_page = int(request.args.get("page_size", 150))
  341. orderby = request.args.get("orderby", "create_time")
  342. desc = request.args.get("desc", True)
  343. try:
  344. tenants = TenantService.get_joined_tenants_by_user_id(current_user.id)
  345. canvas, total = UserCanvasService.get_by_tenant_ids(
  346. [m["tenant_id"] for m in tenants], current_user.id, page_number,
  347. items_per_page, orderby, desc, keywords)
  348. return get_json_result(data={"canvas": canvas, "total": total})
  349. except Exception as e:
  350. return server_error_response(e)
  351. @manager.route('/setting', methods=['POST']) # noqa: F821
  352. @validate_request("id", "title", "permission")
  353. @login_required
  354. def setting():
  355. req = request.json
  356. req["user_id"] = current_user.id
  357. if not UserCanvasService.accessible(req["id"], current_user.id):
  358. return get_json_result(
  359. data=False, message='Only owner of canvas authorized for this operation.',
  360. code=RetCode.OPERATING_ERROR)
  361. e,flow = UserCanvasService.get_by_id(req["id"])
  362. if not e:
  363. return get_data_error_result(message="canvas not found.")
  364. flow = flow.to_dict()
  365. flow["title"] = req["title"]
  366. if req["description"]:
  367. flow["description"] = req["description"]
  368. if req["permission"]:
  369. flow["permission"] = req["permission"]
  370. if req["avatar"]:
  371. flow["avatar"] = req["avatar"]
  372. num= UserCanvasService.update_by_id(req["id"], flow)
  373. return get_json_result(data=num)
  374. @manager.route('/trace', methods=['GET']) # noqa: F821
  375. def trace():
  376. cvs_id = request.args.get("canvas_id")
  377. msg_id = request.args.get("message_id")
  378. try:
  379. bin = REDIS_CONN.get(f"{cvs_id}-{msg_id}-logs")
  380. if not bin:
  381. return get_json_result(data={})
  382. return get_json_result(data=json.loads(bin.encode("utf-8")))
  383. except Exception as e:
  384. logging.exception(e)
  385. @manager.route('/<canvas_id>/sessions', methods=['GET']) # noqa: F821
  386. @login_required
  387. def sessions(canvas_id):
  388. tenant_id = current_user.id
  389. if not UserCanvasService.accessible(canvas_id, tenant_id):
  390. return get_json_result(
  391. data=False, message='Only owner of canvas authorized for this operation.',
  392. code=RetCode.OPERATING_ERROR)
  393. user_id = request.args.get("user_id")
  394. page_number = int(request.args.get("page", 1))
  395. items_per_page = int(request.args.get("page_size", 30))
  396. keywords = request.args.get("keywords")
  397. from_date = request.args.get("from_date")
  398. to_date = request.args.get("to_date")
  399. orderby = request.args.get("orderby", "update_time")
  400. if request.args.get("desc") == "False" or request.args.get("desc") == "false":
  401. desc = False
  402. else:
  403. desc = True
  404. # dsl defaults to True in all cases except for False and false
  405. include_dsl = request.args.get("dsl") != "False" and request.args.get("dsl") != "false"
  406. total, sess = API4ConversationService.get_list(canvas_id, tenant_id, page_number, items_per_page, orderby, desc,
  407. None, user_id, include_dsl, keywords, from_date, to_date)
  408. try:
  409. return get_json_result(data={"total": total, "sessions": sess})
  410. except Exception as e:
  411. return server_error_response(e)