hcornet 506716e703
Some checks failed
Deployment Verification / deploy-and-test (push) Failing after 29s
first sync
2025-03-04 07:59:21 +01:00

456 lines
19 KiB
Python

# -*- coding: utf-8 -*-
"""
flask_dropzone
~~~~~~~~~~~~~~
:author: Grey Li <withlihui@gmail.com>
:copyright: (c) 2017 by Grey Li.
:license: MIT, see LICENSE for more details.
"""
import warnings
from flask import Blueprint
from flask import Markup
from flask import current_app
from flask import render_template_string
from flask import url_for
from .utils import get_url # noqa
from .utils import random_filename
#: defined normal file type
allowed_file_extensions = {
'default': 'image/*, audio/*, video/*, text/*, application/*',
'image': 'image/*',
'audio': 'audio/*',
'video': 'video/*',
'text': 'text/*',
'app': 'application/*'
}
class _Dropzone(object):
@staticmethod
def load(js_url='', css_url='', version='5.2.0'):
"""Load Dropzone resources with given version and init dropzone configuration.
.. versionchanged:: 1.4.3
Added ``js_url`` and ``css_url`` parameters to pass custom resource URL.
.. versionchanged:: 1.4.4
This method was deprecated due to inflexible. Now it's divided into three methods:
1. Use ``load_css()`` to load css resources.
2. Use ``load_js()`` to load js resources.
3. Use ``config()`` to configure Dropzone.
:param js_url: The JavaScript url for Dropzone.js.
:param css_url: The CSS url for Dropzone.js.
:param version: The version of Dropzone.js.
"""
warnings.warn('The method will be removed in 2.0, see docs for more details.')
js_filename = 'dropzone.min.js'
css_filename = 'dropzone.min.css'
upload_multiple = current_app.config['DROPZONE_UPLOAD_MULTIPLE']
parallel_uploads = current_app.config['DROPZONE_PARALLEL_UPLOADS']
if upload_multiple in [True, 'true', 'True', 1]:
upload_multiple = 'true'
else:
upload_multiple = 'false'
serve_local = current_app.config['DROPZONE_SERVE_LOCAL']
size = current_app.config['DROPZONE_MAX_FILE_SIZE']
param = current_app.config['DROPZONE_INPUT_NAME']
redirect_view = current_app.config['DROPZONE_REDIRECT_VIEW']
if redirect_view is not None:
redirect_js = '''
this.on("queuecomplete", function(file) {
// Called when all files in the queue finish uploading.
window.location = "%s";
});''' % url_for(redirect_view)
else:
redirect_js = ''
if not current_app.config['DROPZONE_ALLOWED_FILE_CUSTOM']:
allowed_type = allowed_file_extensions[
current_app.config['DROPZONE_ALLOWED_FILE_TYPE']]
else:
allowed_type = current_app.config['DROPZONE_ALLOWED_FILE_TYPE']
max_files = current_app.config['DROPZONE_MAX_FILES']
default_message = current_app.config['DROPZONE_DEFAULT_MESSAGE']
invalid_file_type = current_app.config['DROPZONE_INVALID_FILE_TYPE']
file_too_big = current_app.config['DROPZONE_FILE_TOO_BIG']
server_error = current_app.config['DROPZONE_SERVER_ERROR']
browser_unsupported = current_app.config['DROPZONE_BROWSER_UNSUPPORTED']
max_files_exceeded = current_app.config['DROPZONE_MAX_FILE_EXCEED']
cancelUpload = current_app.config['DROPZONE_CANCEL_UPLOAD']
removeFile = current_app.config['DROPZONE_REMOVE_FILE']
cancelConfirmation = current_app.config['DROPZONE_CANCEL_CONFIRMATION']
uploadCanceled = current_app.config['DROPZONE_UPLOAD_CANCELED']
timeout = current_app.config['DROPZONE_TIMEOUT']
if timeout:
timeout_js = 'timeout: %d,' % timeout
else:
timeout_js = ''
if serve_local:
js = '<script src="%s"></script>\n' % url_for('dropzone.static', filename=js_filename)
css = '<link rel="stylesheet" href="%s" type="text/css">\n' % \
url_for('dropzone.static', filename=css_filename)
else:
js = '<script src="https://cdn.jsdelivr.net/npm/dropzone@%s/dist/%s"></script>\n' % (version, js_filename)
css = '<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/dropzone@%s/dist/min/%s"' \
' type="text/css">\n' % (version, css_filename)
if js_url:
js = '<script src="%s"></script>\n' % js_url
if css_url:
css = '<link rel="stylesheet" href="%s" type="text/css">\n' % css_url
return Markup('''
%s%s<script>
Dropzone.options.myDropzone = {
init: function() {%s},
uploadMultiple: %s,
parallelUploads: %d,
paramName: "%s", // The name that will be used to transfer the file
maxFilesize: %d, // MB
acceptedFiles: "%s",
maxFiles: %s,
dictDefaultMessage: "%s", // message display on drop area
dictFallbackMessage: "%s",
dictInvalidFileType: "%s",
dictFileTooBig: "%s",
dictResponseError: "%s",
dictMaxFilesExceeded: "%s",
dictCancelUpload: "%s",
dictRemoveFile: "%s",
dictCancelUploadConfirmation: "%s",
dictUploadCanceled: "%s",
%s // timeout
};
</script>
''' % (css, js, redirect_js, upload_multiple, parallel_uploads, param, size, allowed_type, max_files,
default_message, browser_unsupported, invalid_file_type, file_too_big, server_error,
max_files_exceeded, cancelUpload, removeFile, cancelConfirmation, uploadCanceled, timeout_js))
@staticmethod
def load_css(css_url=None, version='5.2.0'):
"""Load Dropzone's css resources with given version.
.. versionadded:: 1.4.4
:param css_url: The CSS url for Dropzone.js.
:param version: The version of Dropzone.js.
"""
css_filename = 'dropzone.min.css'
serve_local = current_app.config['DROPZONE_SERVE_LOCAL']
if serve_local:
css = '<link rel="stylesheet" href="%s" type="text/css">\n' % \
url_for('dropzone.static', filename=css_filename)
else:
css = '<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/dropzone@%s/dist/min/%s"' \
' type="text/css">\n' % (version, css_filename)
if css_url:
css = '<link rel="stylesheet" href="%s" type="text/css">\n' % css_url
return Markup(css)
@staticmethod
def load_js(js_url=None, version='5.2.0'):
"""Load Dropzone's js resources with given version.
.. versionadded:: 1.4.4
:param js_url: The JS url for Dropzone.js.
:param version: The version of Dropzone.js.
"""
js_filename = 'dropzone.min.js'
serve_local = current_app.config['DROPZONE_SERVE_LOCAL']
if serve_local:
js = '<script src="%s"></script>\n' % url_for('dropzone.static', filename=js_filename)
else:
js = '<script src="https://cdn.jsdelivr.net/npm/dropzone@%s/dist/%s"></script>\n' % (version, js_filename)
if js_url:
js = '<script src="%s"></script>\n' % js_url
return Markup(js)
@staticmethod
def config(redirect_url=None, custom_init='', custom_options='', **kwargs):
"""Initialize dropzone configuration.
.. versionadded:: 1.4.4
:param redirect_url: The URL to redirect when upload complete.
:param custom_init: Custom javascript code in ``init: function() {}``.
:param custom_options: Custom javascript code in ``Dropzone.options.myDropzone = {}``.
:param **kwargs: Mirror configuration variable, lowercase and without prefix.
For example, ``DROPZONE_UPLOAD_MULTIPLE`` becomes ``upload_multiple`` here.
"""
if custom_init and not custom_init.strip().endswith(';'):
custom_init += ';'
if custom_options and not custom_options.strip().endswith(','):
custom_options += ','
upload_multiple = kwargs.get('upload_multiple', current_app.config['DROPZONE_UPLOAD_MULTIPLE'])
parallel_uploads = kwargs.get('parallel_uploads', current_app.config['DROPZONE_PARALLEL_UPLOADS'])
if upload_multiple in [True, 'true', 'True', 1]:
upload_multiple = 'true'
else:
upload_multiple = 'false'
size = kwargs.get('max_file_size', current_app.config['DROPZONE_MAX_FILE_SIZE'])
param = kwargs.get('input_name', current_app.config['DROPZONE_INPUT_NAME'])
redirect_view = kwargs.get('redirect_view', current_app.config['DROPZONE_REDIRECT_VIEW'])
if redirect_view is not None or redirect_url is not None:
redirect_url = redirect_url or url_for(redirect_view)
redirect_js = '''
this.on("queuecomplete", function(file) {
// Called when all files in the queue finish uploading.
window.location = "%s";
});''' % redirect_url
else:
redirect_js = ''
max_files = kwargs.get('max_files', current_app.config['DROPZONE_MAX_FILES'])
click_upload = kwargs.get('upload_on_click', current_app.config['DROPZONE_UPLOAD_ON_CLICK'])
button_id = kwargs.get('upload_btn_id', current_app.config['DROPZONE_UPLOAD_BTN_ID'])
in_form = kwargs.get('in_form', current_app.config['DROPZONE_IN_FORM'])
cancelUpload = kwargs.get('cancel_upload', current_app.config['DROPZONE_CANCEL_UPLOAD'])
removeFile = kwargs.get('remove_file', current_app.config['DROPZONE_REMOVE_FILE'])
cancelConfirmation = kwargs.get('cancel_confirmation', current_app.config['DROPZONE_CANCEL_CONFIRMATION'])
uploadCanceled = kwargs.get('upload_canceled', current_app.config['DROPZONE_UPLOAD_CANCELED'])
if click_upload:
if in_form:
action = get_url(kwargs.get('upload_action', current_app.config['DROPZONE_UPLOAD_ACTION']))
click_listener = '''
dz = this; // Makes sure that 'this' is understood inside the functions below.
document.getElementById("%s").addEventListener("click", function handler(e) {
e.currentTarget.removeEventListener(e.type, handler);
e.preventDefault();
e.stopPropagation();
dz.processQueue();
});
this.on("queuecomplete", function(file) {
// Called when all files in the queue finish uploading.
document.getElementById("%s").click();
});
''' % (button_id, button_id)
click_option = '''
url: "%s",
autoProcessQueue: false,
// addRemoveLinks: true,
''' % action
else:
click_listener = '''
dz = this;
document.getElementById("%s").addEventListener("click", function handler(e) {dz.processQueue();});
''' % button_id
click_option = '''
autoProcessQueue: false,
// addRemoveLinks: true,
'''
upload_multiple = 'true'
parallel_uploads = max_files if isinstance(max_files, int) else parallel_uploads
else:
click_listener = ''
click_option = ''
allowed_file_type = kwargs.get('allowed_file_type', current_app.config['DROPZONE_ALLOWED_FILE_TYPE'])
allowed_file_custom = kwargs.get('allowed_file_custom', current_app.config['DROPZONE_ALLOWED_FILE_CUSTOM'])
if allowed_file_custom:
allowed_type = allowed_file_type
else:
allowed_type = allowed_file_extensions[allowed_file_type]
default_message = kwargs.get('default_message', current_app.config['DROPZONE_DEFAULT_MESSAGE'])
invalid_file_type = kwargs.get('invalid_file_type', current_app.config['DROPZONE_INVALID_FILE_TYPE'])
file_too_big = kwargs.get('file_too_big', current_app.config['DROPZONE_FILE_TOO_BIG'])
server_error = kwargs.get('server_error', current_app.config['DROPZONE_SERVER_ERROR'])
browser_unsupported = kwargs.get('browser_unsupported', current_app.config['DROPZONE_BROWSER_UNSUPPORTED'])
max_files_exceeded = kwargs.get('max_file_exceeded', current_app.config['DROPZONE_MAX_FILE_EXCEED'])
timeout = kwargs.get('timeout', current_app.config['DROPZONE_TIMEOUT'])
if timeout:
custom_options += 'timeout: %d,' % timeout
enable_csrf = kwargs.get('enable_csrf', current_app.config['DROPZONE_ENABLE_CSRF'])
if enable_csrf:
if 'csrf' not in current_app.extensions:
raise RuntimeError("CSRFProtect is not initialized. It's required to enable CSRF protect, \
see docs for more details.")
csrf_token = render_template_string('{{ csrf_token() }}')
custom_options += 'headers: {"X-CSRF-Token": "%s"},' % csrf_token
return Markup('''<script>
Dropzone.options.myDropzone = {
init: function() {
%s // redirect after queue complete
%s // upload queue when button click
%s // custom init code
},
%s // click upload options
uploadMultiple: %s,
parallelUploads: %d,
paramName: "%s", // The name that will be used to transfer the file
maxFilesize: %d, // MB
acceptedFiles: "%s",
maxFiles: %s,
dictDefaultMessage: `%s`, // message display on drop area
dictFallbackMessage: "%s",
dictInvalidFileType: "%s",
dictFileTooBig: "%s",
dictResponseError: "%s",
dictMaxFilesExceeded: "%s",
dictCancelUpload: "%s",
dictRemoveFile: "%s",
dictCancelUploadConfirmation: "%s",
dictUploadCanceled: "%s",
%s // custom options code
};
</script>
''' % (redirect_js, click_listener, custom_init, click_option,
upload_multiple, parallel_uploads, param, size, allowed_type, max_files,
default_message, browser_unsupported, invalid_file_type, file_too_big,
server_error, max_files_exceeded, cancelUpload, removeFile, cancelConfirmation,
uploadCanceled, custom_options))
@staticmethod
def create(action='', csrf=False, action_view='', **kwargs):
"""Create a Dropzone form with given action.
.. versionchanged:: 1.4.2
Added ``csrf`` parameter to enable CSRF protect.
.. versionchanged:: 1.4.3
Added ``action`` parameter to replace ``action_view``, ``action_view`` was deprecated now.
.. versionchanged:: 1.5.0
If ``DROPZONE_IN_FORM`` set to ``True``, create ``<div>`` instead of ``<form>``.
.. versionchanged:: 1.5.4
``csrf`` was deprecated now.
:param action: The action attribute in ``<form>``, pass the url which handle uploads.
:param csrf: Enable CSRF protect or not, same with ``DROPZONE_ENABLE_CSRF``, deprecated since 1.5.4.
:param action_view: The view which handle the post data, deprecated since 1.4.2.
"""
if current_app.config['DROPZONE_IN_FORM']:
return Markup('<div class="dropzone" id="myDropzone"></div>')
if action:
action_url = get_url(action)
else:
warnings.warn('The argument was renamed to "action" and will be removed in 2.0.')
action_url = url_for(action_view, **kwargs)
if csrf:
warnings.warn('The argument was deprecated and will be removed in 2.0, use DROPZONE_ENABLE_CSRF instead.')
return Markup('''<form action="%s" method="post" class="dropzone" id="myDropzone"
enctype="multipart/form-data"></form>''' % action_url)
@staticmethod
def style(css):
"""Add css to dropzone.
:param css: style sheet code.
"""
return Markup('<style>\n.dropzone{%s}\n</style>' % css)
class Dropzone(object):
def __init__(self, app=None):
if app is not None:
self.init_app(app)
def init_app(self, app):
blueprint = Blueprint('dropzone', __name__)
app.register_blueprint(blueprint)
if not hasattr(app, 'extensions'):
app.extensions = {}
app.extensions['dropzone'] = _Dropzone
app.context_processor(self.context_processor)
# settings
app.config.setdefault('DROPZONE_SERVE_LOCAL', False)
app.config.setdefault('DROPZONE_MAX_FILE_SIZE', 3) # MB
app.config.setdefault('DROPZONE_INPUT_NAME', 'file')
app.config.setdefault('DROPZONE_ALLOWED_FILE_CUSTOM', False)
app.config.setdefault('DROPZONE_ALLOWED_FILE_TYPE', 'default')
app.config.setdefault('DROPZONE_MAX_FILES', 'null')
# The timeout to cancel upload request in millisecond, default to 30000 (30 second).
# Set a large number if you need to upload large file.
# .. versionadded: 1.5.0
app.config.setdefault('DROPZONE_TIMEOUT', None) # millisecond, default to 30000 (30 second)
# The view to redirect when upload was completed.
# .. versionadded:: 1.4.1
app.config.setdefault('DROPZONE_REDIRECT_VIEW', None)
# Whether to send multiple files in one request.
# In default, each file will send with a request.
# Then you can use ``request.files.getlist('paramName')`` to
# get a list of uploads.
# .. versionadded:: 1.4.1
app.config.setdefault('DROPZONE_UPLOAD_MULTIPLE', False)
# When ``DROPZONE_UPLOAD_MULTIPLE`` set to True, this will
# defined how many uploads will handled in per request.
# .. versionadded:: 1.4.1
app.config.setdefault('DROPZONE_PARALLEL_UPLOADS', 2)
# When set to ``True``, it will add a csrf_token hidden field in upload form.
# You have to install Flask-WTF to make it work properly, see details in docs.
# .. versionadded:: 1.4.2
app.config.setdefault('DROPZONE_ENABLE_CSRF', False)
# Add support to upload files when button was clicked.
# .. versionadded:: 1.5.0
app.config.setdefault('DROPZONE_UPLOAD_ACTION', '')
app.config.setdefault('DROPZONE_UPLOAD_ON_CLICK', False)
app.config.setdefault('DROPZONE_UPLOAD_BTN_ID', 'upload')
# Add support to create dropzone inside ``<form>``.
# .. versionadded:: 1.5.0
app.config.setdefault('DROPZONE_IN_FORM', False)
# messages
app.config.setdefault('DROPZONE_DEFAULT_MESSAGE', "Drop files here or click to upload.")
app.config.setdefault('DROPZONE_INVALID_FILE_TYPE', "You can't upload files of this type.")
app.config.setdefault('DROPZONE_FILE_TOO_BIG',
"File is too big {{filesize}}. Max filesize: {{maxFilesize}}MiB.")
app.config.setdefault('DROPZONE_SERVER_ERROR', "Server error: {{statusCode}}")
app.config.setdefault('DROPZONE_BROWSER_UNSUPPORTED',
"Your browser does not support drag'n'drop file uploads.")
app.config.setdefault('DROPZONE_MAX_FILE_EXCEED', "You can't upload any more files.")
app.config.setdefault('DROPZONE_CANCEL_UPLOAD', "Cancel upload")
app.config.setdefault('DROPZONE_REMOVE_FILE', "Remove file")
app.config.setdefault('DROPZONE_CANCEL_CONFIRMATION', "You really want to delete this file?")
app.config.setdefault('DROPZONE_UPLOAD_CANCELED', "Upload canceled")
@staticmethod
def context_processor():
return {
'dropzone': current_app.extensions['dropzone']
}