Nevar pievienot vairāk kā 25 tēmas Tēmai ir jāsākas ar burtu vai ciparu, tā var saturēt domu zīmes ('-') un var būt līdz 35 simboliem gara.

remote_files.py 3.3KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990
  1. import urllib.parse
  2. from typing import cast
  3. import httpx
  4. from flask_login import current_user
  5. from flask_restx import Resource, marshal_with, reqparse
  6. import services
  7. from controllers.common import helpers
  8. from controllers.common.errors import (
  9. FileTooLargeError,
  10. RemoteFileUploadError,
  11. UnsupportedFileTypeError,
  12. )
  13. from core.file import helpers as file_helpers
  14. from core.helper import ssrf_proxy
  15. from extensions.ext_database import db
  16. from fields.file_fields import file_fields_with_signed_url, remote_file_info_fields
  17. from models.account import Account
  18. from services.file_service import FileService
  19. from . import console_ns
  20. @console_ns.route("/remote-files/<path:url>")
  21. class RemoteFileInfoApi(Resource):
  22. @marshal_with(remote_file_info_fields)
  23. def get(self, url):
  24. decoded_url = urllib.parse.unquote(url)
  25. resp = ssrf_proxy.head(decoded_url)
  26. if resp.status_code != httpx.codes.OK:
  27. # failed back to get method
  28. resp = ssrf_proxy.get(decoded_url, timeout=3)
  29. resp.raise_for_status()
  30. return {
  31. "file_type": resp.headers.get("Content-Type", "application/octet-stream"),
  32. "file_length": int(resp.headers.get("Content-Length", 0)),
  33. }
  34. @console_ns.route("/remote-files/upload")
  35. class RemoteFileUploadApi(Resource):
  36. @marshal_with(file_fields_with_signed_url)
  37. def post(self):
  38. parser = reqparse.RequestParser()
  39. parser.add_argument("url", type=str, required=True, help="URL is required")
  40. args = parser.parse_args()
  41. url = args["url"]
  42. try:
  43. resp = ssrf_proxy.head(url=url)
  44. if resp.status_code != httpx.codes.OK:
  45. resp = ssrf_proxy.get(url=url, timeout=3, follow_redirects=True)
  46. if resp.status_code != httpx.codes.OK:
  47. raise RemoteFileUploadError(f"Failed to fetch file from {url}: {resp.text}")
  48. except httpx.RequestError as e:
  49. raise RemoteFileUploadError(f"Failed to fetch file from {url}: {str(e)}")
  50. file_info = helpers.guess_file_info_from_response(resp)
  51. if not FileService.is_file_size_within_limit(extension=file_info.extension, file_size=file_info.size):
  52. raise FileTooLargeError
  53. content = resp.content if resp.request.method == "GET" else ssrf_proxy.get(url).content
  54. try:
  55. user = cast(Account, current_user)
  56. upload_file = FileService(db.engine).upload_file(
  57. filename=file_info.filename,
  58. content=content,
  59. mimetype=file_info.mimetype,
  60. user=user,
  61. source_url=url,
  62. )
  63. except services.errors.file.FileTooLargeError as file_too_large_error:
  64. raise FileTooLargeError(file_too_large_error.description)
  65. except services.errors.file.UnsupportedFileTypeError:
  66. raise UnsupportedFileTypeError()
  67. return {
  68. "id": upload_file.id,
  69. "name": upload_file.name,
  70. "size": upload_file.size,
  71. "extension": upload_file.extension,
  72. "url": file_helpers.get_signed_file_url(upload_file_id=upload_file.id),
  73. "mime_type": upload_file.mime_type,
  74. "created_by": upload_file.created_by,
  75. "created_at": upload_file.created_at,
  76. }, 201