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.

tool_files.py 2.0KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657
  1. from urllib.parse import quote
  2. from flask import Response
  3. from flask_restful import Resource, reqparse
  4. from werkzeug.exceptions import Forbidden, NotFound
  5. from controllers.files import api
  6. from controllers.files.error import UnsupportedFileTypeError
  7. from core.tools.signature import verify_tool_file_signature
  8. from core.tools.tool_file_manager import ToolFileManager
  9. from models import db as global_db
  10. class ToolFilePreviewApi(Resource):
  11. def get(self, file_id, extension):
  12. file_id = str(file_id)
  13. parser = reqparse.RequestParser()
  14. parser.add_argument("timestamp", type=str, required=True, location="args")
  15. parser.add_argument("nonce", type=str, required=True, location="args")
  16. parser.add_argument("sign", type=str, required=True, location="args")
  17. parser.add_argument("as_attachment", type=bool, required=False, default=False, location="args")
  18. args = parser.parse_args()
  19. if not verify_tool_file_signature(
  20. file_id=file_id, timestamp=args["timestamp"], nonce=args["nonce"], sign=args["sign"]
  21. ):
  22. raise Forbidden("Invalid request.")
  23. try:
  24. tool_file_manager = ToolFileManager(engine=global_db.engine)
  25. stream, tool_file = tool_file_manager.get_file_generator_by_tool_file_id(
  26. file_id,
  27. )
  28. if not stream or not tool_file:
  29. raise NotFound("file is not found")
  30. except Exception:
  31. raise UnsupportedFileTypeError()
  32. response = Response(
  33. stream,
  34. mimetype=tool_file.mimetype,
  35. direct_passthrough=True,
  36. headers={},
  37. )
  38. if tool_file.size > 0:
  39. response.headers["Content-Length"] = str(tool_file.size)
  40. if args["as_attachment"]:
  41. encoded_filename = quote(tool_file.name)
  42. response.headers["Content-Disposition"] = f"attachment; filename*=UTF-8''{encoded_filename}"
  43. return response
  44. api.add_resource(ToolFilePreviewApi, "/files/tools/<uuid:file_id>.<string:extension>")