file.py 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105
  1. import urllib.parse
  2. from flask import request
  3. from flask_login import current_user
  4. from flask_restful import Resource, marshal_with, reqparse
  5. import services
  6. from configs import dify_config
  7. from constants import DOCUMENT_EXTENSIONS
  8. from controllers.console import api
  9. from controllers.console.datasets.error import (
  10. FileTooLargeError,
  11. NoFileUploadedError,
  12. TooManyFilesError,
  13. UnsupportedFileTypeError,
  14. )
  15. from controllers.console.setup import setup_required
  16. from controllers.console.wraps import account_initialization_required, cloud_edition_billing_resource_check
  17. from core.helper import ssrf_proxy
  18. from fields.file_fields import file_fields, remote_file_info_fields, upload_config_fields
  19. from libs.login import login_required
  20. from services.file_service import FileService
  21. PREVIEW_WORDS_LIMIT = 3000
  22. class FileApi(Resource):
  23. @setup_required
  24. @login_required
  25. @account_initialization_required
  26. @marshal_with(upload_config_fields)
  27. def get(self):
  28. return {
  29. "file_size_limit": dify_config.UPLOAD_FILE_SIZE_LIMIT,
  30. "batch_count_limit": dify_config.UPLOAD_FILE_BATCH_LIMIT,
  31. "image_file_size_limit": dify_config.UPLOAD_IMAGE_FILE_SIZE_LIMIT,
  32. "video_file_size_limit": dify_config.UPLOAD_VIDEO_FILE_SIZE_LIMIT,
  33. "audio_file_size_limit": dify_config.UPLOAD_AUDIO_FILE_SIZE_LIMIT,
  34. }, 200
  35. @setup_required
  36. @login_required
  37. @account_initialization_required
  38. @marshal_with(file_fields)
  39. @cloud_edition_billing_resource_check("documents")
  40. def post(self):
  41. # get file from request
  42. file = request.files["file"]
  43. parser = reqparse.RequestParser()
  44. parser.add_argument("source", type=str, required=False, location="args")
  45. source = parser.parse_args().get("source")
  46. # check file
  47. if "file" not in request.files:
  48. raise NoFileUploadedError()
  49. if len(request.files) > 1:
  50. raise TooManyFilesError()
  51. try:
  52. upload_file = FileService.upload_file(file=file, user=current_user, source=source)
  53. except services.errors.file.FileTooLargeError as file_too_large_error:
  54. raise FileTooLargeError(file_too_large_error.description)
  55. except services.errors.file.UnsupportedFileTypeError:
  56. raise UnsupportedFileTypeError()
  57. return upload_file, 201
  58. class FilePreviewApi(Resource):
  59. @setup_required
  60. @login_required
  61. @account_initialization_required
  62. def get(self, file_id):
  63. file_id = str(file_id)
  64. text = FileService.get_file_preview(file_id)
  65. return {"content": text}
  66. class FileSupportTypeApi(Resource):
  67. @setup_required
  68. @login_required
  69. @account_initialization_required
  70. def get(self):
  71. return {"allowed_extensions": DOCUMENT_EXTENSIONS}
  72. class RemoteFileInfoApi(Resource):
  73. @marshal_with(remote_file_info_fields)
  74. def get(self, url):
  75. decoded_url = urllib.parse.unquote(url)
  76. try:
  77. response = ssrf_proxy.head(decoded_url)
  78. return {
  79. "file_type": response.headers.get("Content-Type", "application/octet-stream"),
  80. "file_length": int(response.headers.get("Content-Length", 0)),
  81. }
  82. except Exception as e:
  83. return {"error": str(e)}, 400
  84. api.add_resource(FileApi, "/files/upload")
  85. api.add_resource(FilePreviewApi, "/files/<uuid:file_id>/preview")
  86. api.add_resource(FileSupportTypeApi, "/files/support-type")
  87. api.add_resource(RemoteFileInfoApi, "/remote-files/<path:url>")