mirror of
https://git.mirrors.martin98.com/https://github.com/langgenius/dify.git
synced 2025-08-12 10:08:58 +08:00
fix(remote-files): fallback to get when remote server not support head method (#10370)
This commit is contained in:
parent
f8c958a409
commit
823ae03a08
@ -62,3 +62,27 @@ class EmailSendIpLimitError(BaseHTTPException):
|
|||||||
error_code = "email_send_ip_limit"
|
error_code = "email_send_ip_limit"
|
||||||
description = "Too many emails have been sent from this IP address recently. Please try again later."
|
description = "Too many emails have been sent from this IP address recently. Please try again later."
|
||||||
code = 429
|
code = 429
|
||||||
|
|
||||||
|
|
||||||
|
class FileTooLargeError(BaseHTTPException):
|
||||||
|
error_code = "file_too_large"
|
||||||
|
description = "File size exceeded. {message}"
|
||||||
|
code = 413
|
||||||
|
|
||||||
|
|
||||||
|
class UnsupportedFileTypeError(BaseHTTPException):
|
||||||
|
error_code = "unsupported_file_type"
|
||||||
|
description = "File type not allowed."
|
||||||
|
code = 415
|
||||||
|
|
||||||
|
|
||||||
|
class TooManyFilesError(BaseHTTPException):
|
||||||
|
error_code = "too_many_files"
|
||||||
|
description = "Only one file is allowed."
|
||||||
|
code = 400
|
||||||
|
|
||||||
|
|
||||||
|
class NoFileUploadedError(BaseHTTPException):
|
||||||
|
error_code = "no_file_uploaded"
|
||||||
|
description = "Please upload your file."
|
||||||
|
code = 400
|
||||||
|
@ -15,7 +15,7 @@ from fields.file_fields import file_fields, upload_config_fields
|
|||||||
from libs.login import login_required
|
from libs.login import login_required
|
||||||
from services.file_service import FileService
|
from services.file_service import FileService
|
||||||
|
|
||||||
from .errors import (
|
from .error import (
|
||||||
FileTooLargeError,
|
FileTooLargeError,
|
||||||
NoFileUploadedError,
|
NoFileUploadedError,
|
||||||
TooManyFilesError,
|
TooManyFilesError,
|
@ -1,25 +0,0 @@
|
|||||||
from libs.exception import BaseHTTPException
|
|
||||||
|
|
||||||
|
|
||||||
class FileTooLargeError(BaseHTTPException):
|
|
||||||
error_code = "file_too_large"
|
|
||||||
description = "File size exceeded. {message}"
|
|
||||||
code = 413
|
|
||||||
|
|
||||||
|
|
||||||
class UnsupportedFileTypeError(BaseHTTPException):
|
|
||||||
error_code = "unsupported_file_type"
|
|
||||||
description = "File type not allowed."
|
|
||||||
code = 415
|
|
||||||
|
|
||||||
|
|
||||||
class TooManyFilesError(BaseHTTPException):
|
|
||||||
error_code = "too_many_files"
|
|
||||||
description = "Only one file is allowed."
|
|
||||||
code = 400
|
|
||||||
|
|
||||||
|
|
||||||
class NoFileUploadedError(BaseHTTPException):
|
|
||||||
error_code = "no_file_uploaded"
|
|
||||||
description = "Please upload your file."
|
|
||||||
code = 400
|
|
@ -1,9 +1,11 @@
|
|||||||
import urllib.parse
|
import urllib.parse
|
||||||
from typing import cast
|
from typing import cast
|
||||||
|
|
||||||
|
import httpx
|
||||||
from flask_login import current_user
|
from flask_login import current_user
|
||||||
from flask_restful import Resource, marshal_with, reqparse
|
from flask_restful import Resource, marshal_with, reqparse
|
||||||
|
|
||||||
|
import services
|
||||||
from controllers.common import helpers
|
from controllers.common import helpers
|
||||||
from core.file import helpers as file_helpers
|
from core.file import helpers as file_helpers
|
||||||
from core.helper import ssrf_proxy
|
from core.helper import ssrf_proxy
|
||||||
@ -11,19 +13,25 @@ from fields.file_fields import file_fields_with_signed_url, remote_file_info_fie
|
|||||||
from models.account import Account
|
from models.account import Account
|
||||||
from services.file_service import FileService
|
from services.file_service import FileService
|
||||||
|
|
||||||
|
from .error import (
|
||||||
|
FileTooLargeError,
|
||||||
|
UnsupportedFileTypeError,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
class RemoteFileInfoApi(Resource):
|
class RemoteFileInfoApi(Resource):
|
||||||
@marshal_with(remote_file_info_fields)
|
@marshal_with(remote_file_info_fields)
|
||||||
def get(self, url):
|
def get(self, url):
|
||||||
decoded_url = urllib.parse.unquote(url)
|
decoded_url = urllib.parse.unquote(url)
|
||||||
try:
|
resp = ssrf_proxy.head(decoded_url)
|
||||||
response = ssrf_proxy.head(decoded_url)
|
if resp.status_code != httpx.codes.OK:
|
||||||
return {
|
# failed back to get method
|
||||||
"file_type": response.headers.get("Content-Type", "application/octet-stream"),
|
resp = ssrf_proxy.get(decoded_url, timeout=3)
|
||||||
"file_length": int(response.headers.get("Content-Length", 0)),
|
resp.raise_for_status()
|
||||||
}
|
return {
|
||||||
except Exception as e:
|
"file_type": resp.headers.get("Content-Type", "application/octet-stream"),
|
||||||
return {"error": str(e)}, 400
|
"file_length": int(resp.headers.get("Content-Length", 0)),
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
class RemoteFileUploadApi(Resource):
|
class RemoteFileUploadApi(Resource):
|
||||||
@ -35,17 +43,17 @@ class RemoteFileUploadApi(Resource):
|
|||||||
|
|
||||||
url = args["url"]
|
url = args["url"]
|
||||||
|
|
||||||
response = ssrf_proxy.head(url)
|
resp = ssrf_proxy.head(url=url)
|
||||||
response.raise_for_status()
|
if resp.status_code != httpx.codes.OK:
|
||||||
|
resp = ssrf_proxy.get(url=url, timeout=3)
|
||||||
|
resp.raise_for_status()
|
||||||
|
|
||||||
file_info = helpers.guess_file_info_from_response(response)
|
file_info = helpers.guess_file_info_from_response(resp)
|
||||||
|
|
||||||
if not FileService.is_file_size_within_limit(extension=file_info.extension, file_size=file_info.size):
|
if not FileService.is_file_size_within_limit(extension=file_info.extension, file_size=file_info.size):
|
||||||
return {"error": "File size exceeded"}, 400
|
raise FileTooLargeError
|
||||||
|
|
||||||
response = ssrf_proxy.get(url)
|
content = resp.content if resp.request.method == "GET" else ssrf_proxy.get(url).content
|
||||||
response.raise_for_status()
|
|
||||||
content = response.content
|
|
||||||
|
|
||||||
try:
|
try:
|
||||||
user = cast(Account, current_user)
|
user = cast(Account, current_user)
|
||||||
@ -56,8 +64,10 @@ class RemoteFileUploadApi(Resource):
|
|||||||
user=user,
|
user=user,
|
||||||
source_url=url,
|
source_url=url,
|
||||||
)
|
)
|
||||||
except Exception as e:
|
except services.errors.file.FileTooLargeError as file_too_large_error:
|
||||||
return {"error": str(e)}, 400
|
raise FileTooLargeError(file_too_large_error.description)
|
||||||
|
except services.errors.file.UnsupportedFileTypeError:
|
||||||
|
raise UnsupportedFileTypeError()
|
||||||
|
|
||||||
return {
|
return {
|
||||||
"id": upload_file.id,
|
"id": upload_file.id,
|
||||||
|
@ -1,7 +1,9 @@
|
|||||||
import urllib.parse
|
import urllib.parse
|
||||||
|
|
||||||
|
import httpx
|
||||||
from flask_restful import marshal_with, reqparse
|
from flask_restful import marshal_with, reqparse
|
||||||
|
|
||||||
|
import services
|
||||||
from controllers.common import helpers
|
from controllers.common import helpers
|
||||||
from controllers.web.wraps import WebApiResource
|
from controllers.web.wraps import WebApiResource
|
||||||
from core.file import helpers as file_helpers
|
from core.file import helpers as file_helpers
|
||||||
@ -9,19 +11,22 @@ from core.helper import ssrf_proxy
|
|||||||
from fields.file_fields import file_fields_with_signed_url, remote_file_info_fields
|
from fields.file_fields import file_fields_with_signed_url, remote_file_info_fields
|
||||||
from services.file_service import FileService
|
from services.file_service import FileService
|
||||||
|
|
||||||
|
from .error import FileTooLargeError, UnsupportedFileTypeError
|
||||||
|
|
||||||
|
|
||||||
class RemoteFileInfoApi(WebApiResource):
|
class RemoteFileInfoApi(WebApiResource):
|
||||||
@marshal_with(remote_file_info_fields)
|
@marshal_with(remote_file_info_fields)
|
||||||
def get(self, app_model, end_user, url):
|
def get(self, app_model, end_user, url):
|
||||||
decoded_url = urllib.parse.unquote(url)
|
decoded_url = urllib.parse.unquote(url)
|
||||||
try:
|
resp = ssrf_proxy.head(decoded_url)
|
||||||
response = ssrf_proxy.head(decoded_url)
|
if resp.status_code != httpx.codes.OK:
|
||||||
return {
|
# failed back to get method
|
||||||
"file_type": response.headers.get("Content-Type", "application/octet-stream"),
|
resp = ssrf_proxy.get(decoded_url, timeout=3)
|
||||||
"file_length": int(response.headers.get("Content-Length", -1)),
|
resp.raise_for_status()
|
||||||
}
|
return {
|
||||||
except Exception as e:
|
"file_type": resp.headers.get("Content-Type", "application/octet-stream"),
|
||||||
return {"error": str(e)}, 400
|
"file_length": int(resp.headers.get("Content-Length", -1)),
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
class RemoteFileUploadApi(WebApiResource):
|
class RemoteFileUploadApi(WebApiResource):
|
||||||
@ -33,28 +38,30 @@ class RemoteFileUploadApi(WebApiResource):
|
|||||||
|
|
||||||
url = args["url"]
|
url = args["url"]
|
||||||
|
|
||||||
response = ssrf_proxy.head(url)
|
resp = ssrf_proxy.head(url=url)
|
||||||
response.raise_for_status()
|
if resp.status_code != httpx.codes.OK:
|
||||||
|
resp = ssrf_proxy.get(url=url, timeout=3)
|
||||||
|
resp.raise_for_status()
|
||||||
|
|
||||||
file_info = helpers.guess_file_info_from_response(response)
|
file_info = helpers.guess_file_info_from_response(resp)
|
||||||
|
|
||||||
if not FileService.is_file_size_within_limit(extension=file_info.extension, file_size=file_info.size):
|
if not FileService.is_file_size_within_limit(extension=file_info.extension, file_size=file_info.size):
|
||||||
return {"error": "File size exceeded"}, 400
|
raise FileTooLargeError
|
||||||
|
|
||||||
response = ssrf_proxy.get(url)
|
content = resp.content if resp.request.method == "GET" else ssrf_proxy.get(url).content
|
||||||
response.raise_for_status()
|
|
||||||
content = response.content
|
|
||||||
|
|
||||||
try:
|
try:
|
||||||
upload_file = FileService.upload_file(
|
upload_file = FileService.upload_file(
|
||||||
filename=file_info.filename,
|
filename=file_info.filename,
|
||||||
content=content,
|
content=content,
|
||||||
mimetype=file_info.mimetype,
|
mimetype=file_info.mimetype,
|
||||||
user=end_user, # Use end_user instead of current_user
|
user=end_user,
|
||||||
source_url=url,
|
source_url=url,
|
||||||
)
|
)
|
||||||
except Exception as e:
|
except services.errors.file.FileTooLargeError as file_too_large_error:
|
||||||
return {"error": str(e)}, 400
|
raise FileTooLargeError(file_too_large_error.description)
|
||||||
|
except services.errors.file.UnsupportedFileTypeError:
|
||||||
|
raise UnsupportedFileTypeError
|
||||||
|
|
||||||
return {
|
return {
|
||||||
"id": upload_file.id,
|
"id": upload_file.id,
|
||||||
|
Loading…
x
Reference in New Issue
Block a user