remote_files.py 2.4 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071
  1. import urllib.parse
  2. from typing import cast
  3. from flask_login import current_user
  4. from flask_restful import Resource, marshal_with, reqparse
  5. from controllers.common import helpers
  6. from core.file import helpers as file_helpers
  7. from core.helper import ssrf_proxy
  8. from fields.file_fields import file_fields_with_signed_url, remote_file_info_fields
  9. from models.account import Account
  10. from services.file_service import FileService
  11. class RemoteFileInfoApi(Resource):
  12. @marshal_with(remote_file_info_fields)
  13. def get(self, url):
  14. decoded_url = urllib.parse.unquote(url)
  15. try:
  16. response = ssrf_proxy.head(decoded_url)
  17. return {
  18. "file_type": response.headers.get("Content-Type", "application/octet-stream"),
  19. "file_length": int(response.headers.get("Content-Length", 0)),
  20. }
  21. except Exception as e:
  22. return {"error": str(e)}, 400
  23. class RemoteFileUploadApi(Resource):
  24. @marshal_with(file_fields_with_signed_url)
  25. def post(self):
  26. parser = reqparse.RequestParser()
  27. parser.add_argument("url", type=str, required=True, help="URL is required")
  28. args = parser.parse_args()
  29. url = args["url"]
  30. response = ssrf_proxy.head(url)
  31. response.raise_for_status()
  32. file_info = helpers.guess_file_info_from_response(response)
  33. if not FileService.is_file_size_within_limit(extension=file_info.extension, file_size=file_info.size):
  34. return {"error": "File size exceeded"}, 400
  35. response = ssrf_proxy.get(url)
  36. response.raise_for_status()
  37. content = response.content
  38. try:
  39. user = cast(Account, current_user)
  40. upload_file = FileService.upload_file(
  41. filename=file_info.filename,
  42. content=content,
  43. mimetype=file_info.mimetype,
  44. user=user,
  45. source_url=url,
  46. )
  47. except Exception as e:
  48. return {"error": str(e)}, 400
  49. return {
  50. "id": upload_file.id,
  51. "name": upload_file.name,
  52. "size": upload_file.size,
  53. "extension": upload_file.extension,
  54. "url": file_helpers.get_signed_file_url(upload_file_id=upload_file.id),
  55. "mime_type": upload_file.mime_type,
  56. "created_by": upload_file.created_by,
  57. "created_at": upload_file.created_at,
  58. }, 201