To use the Upload component you need to two things
- Configure dash-uploader with
du.configure_upload
- Create the upload component with
du.Upload
Typically you also would like to define callbacks (functions that are called automatically when upload finishes).
⚠️ Security note: The Upload component allows uploads of arbitrary files to the server harddisk and one should take this into account (with user token checking etc.) if used as part of a public website! Particularly, theconfigure_upload
opens a route forPOST
HTTP requests. Use thehttp_request_handler
argument for defining your custom validation logic.
4 Custom handling of HTTP Requests
5 How dash-uploader works internally?
You need to configure the dash uploader after you created your dash application instance (app
) and before you create any Upload components
Example
import dash_uploader as du
du.configure_upload(app, r'C:\tmp\uploads')
configure_upload(app, folder, use_upload_id=True, upload_api=None, http_request_handler=None)
The application instance. Usually created using
app = dash.Dash(__name__)
The folder where to upload files. Can be relative
("uploads"
) or absolute (r"C:\tmp\my_uploads"
).
If the folder does not exist, it will be created
automatically.
New in version 0.2.0
Determines if the uploads are put into
folders defined by a "upload id". To define an upload id, use the upload_id
parameter of the du.Upload
component. In typical use case, upload id's are unique for different users. In that case, you must use a callable as the app.layout
.
If True, uploads will be put into <folder>/<upload_id>/
;
that is, every user (for example with different
session id) will use their own folder. If False,
all files from all sessions are uploaded into
same folder (not recommended).
The upload api endpoint to use; the url that is used
internally for the upload component POST and GET HTTP
requests. For example, using upload_api="/API/dash-uploader"
would create api endpoint and use address
http://<myhost>[<requests_pathname_prefix>]/API/dash-uploader
for the communication between the front-end and the server. The requests_pathname_prefix
is added automatically, if the dash app
instance has requests_pathname_prefix
. (used with proxies)
New in version 0.5.0
Used for custom configuration on the HTTP POST and GET requests. This can be used to add validation for the HTTP requests (du.HttpRequestHandler
.
See the documentation of @du.HttpRequestHandler
for
more details.
Below are the arguments for the du.Upload
component and their default values.
Upload(
id='dash-uploader',
text='Drag and Drop Here to upload!',
text_completed='Uploaded: ',
text_disabled='The uploader is disabled.',
cancel_button=True,
pause_button=False,
disabled=False,
filetypes=None,
max_file_size=1024,
chunk_size=1,
default_style=None,
upload_id=None,
max_files=1,
)
The html id for the component. This is needed when defining callbacks. Note that ids must be unique in a dash application.
The text to be shown in the upload "Drag and Drop" area. Optional.
The text to show in the upload area after upload has completed succesfully before the name of the uploaded file.
For example, if user
uploaded "data.zip" and text_completed
is
"Ready! ", then user would see text "Ready!
data.zip".
New in version [0.6.0]
The text to show in the upload area when the the component is disabled.
If True, shows a cancel button.
If True, shows a pause button.
New in version [0.6.0]
If True, the file is not allowed to be uploaded.
The filetypes that can be uploaded.
For example ['zip', 'rar']
.
Note that this just checks the extension of the
filename, and user might still upload any kind
of file (by renaming)!
By default, all filetypes are accepted.
The maximum file size in Megabytes. Default: 1024 (=1Gb).
New in version [0.6.0]
The chunk size in Megabytes. Optional. Default: 1 (=1MB).
Inline CSS styling for the main div element. If None, use the default style of the component. If dict, will use the union on the given dict and the default style. (you may override part of the style by giving a dictionary) More styling options through the CSS classes.
The upload id, created with uuid.uuid1()
or uuid.uuid4(), for example. If None
, creates random session id with uuid.uuid1()
. Defines a subfolder where the files are to be uploaded.
Only used, if use_upload_id
parameter is set to True
in du.configure_upload
.
⚠️ Experimental feature. Read below. For bulletproof implementation, force usage of zip files and keep max_files = 1.
The number of files that can be added to the upload field simultaneously.
Notes (not up to date: resumable.js -> flow.js change):
(1) If even a single file which is not supported file
type (i.e. missing in filetypes
), is added to the upload queue, upload process of all files will be permanently interrupted.
(2) Use reasonably small number in max_files
.
(3) When uploading two (or more) folders with Chrome, there is
a bug in resumable.js which makes only one of the
folders to be uploaded. See:
23/resumable.js#416
(4) When uploading folders, note that the subdirectories
are not created -> All files in the folders will
be uploaded to the single upload folder.
In the following example it is assumed that the du.Upload
component id is dash-uploader
, i.e. the component was created with:
du.Upload(
id='dash-uploader',
)
New in version 0.3.0. Syntax Updated in version 0.7.0.
To syntax for calling a callback after uploading file/files would be (version >=0.7.0):
@du.callback(
output=<Output>,
id=<Upload_id>,
)
def call_me(status):
# Do some processing
return dash_component
This is the Output object, just as in regular dash callbacks.
This must be the same id
as used when initiating your du.Upload
component.
A function that takes exactly one argument: status
, and returns a valid dash component, like html.Div("hello")
, as in the regular dash callbacks.
The status
is a du.UploadStatus
object.
A simple example using the du.callback
is:
@du.callback(
output=Output("callback-output", "children"),
id="dash-uploader",
)
def callback_on_completion(status: du.UploadStatus):
return html.Ul([html.Li(str(x)) for x in status.uploaded_files])
In older dash-uploader versions (0.3.0 to 0.6.0), the one would use filenames
(list of str) instead of status
;
@du.callback(
output=Output('callback-output', 'children'),
id='dash-uploader',
)
def get_a_list(filenames):
return html.Ul([html.Li(filenames)])
A list of strings. These will be the uploaded files. For example:
['C:\tmp\Uploads\166d6f24-a80d-11ea-aa00-f48c5012fb50\dataset.csv']
New in version 0.7.0
The status
argument expected by callbacks created with @du.callback
are of type du.UploadStatus
. It has following attributes:
status.latest_file
(pathlib.Path): The full file path to the file that has been latest uploaded.status.uploaded_files
(list of pathlib.Path): The list of full file paths to all of the uploaded files. (uploaded in this session)status.is_completed
(bool): True if all the files have been uploadedstatus.n_uploaded
(int): The number of files already uploaded in this sessionstatus.n_total
(int): The number of files to be uploaded.status.uploaded_size_mb
(float): Size of files uploaded in Megabytesstatus.total_size_mb
(float): Total size of files to be uploaded in Megabytesstatus.upload_id
(str or None): The upload id used in the upload process, if any.status.progress
(float): From 0 to 1, indicating the current upload progress of all files. Fromflow.progress()
. This is based on the total file size.
Example of du.UploadStatus
attributes:
status.latest_file = WindowsPath("C:\tmp\Uploads\f4605571-c0a0-11ec-930f-2016b9d15494\bootstrap-3.3.7.min.css")
status.uploaded_files = [WindowsPath("C:\tmp\Uploads\f4605571-c0a0-11ec-930f-2016b9d15494\base.css"), WindowsPath("C:\tmp\Uploads\f4605571-c0a0-11ec-930f-2016b9d15494\bootstrap-3.3.7.min.css")]
status.is_completed = False
status.n_uploaded = 2
status.n_total = 4
status.uploaded_size_mb = 0.11781978607177734
status.total_size_mb = 0.24998188018798828
status.progress = 0.47131330472103006
status.upload_id = f4605571-c0a0-11ec-930f-2016b9d15494
New in version 0.5.0
The HttpRequestHandler
is a class that is meant to be subclassed. It is used as an argument for configure_upload
, and it makes custom HTTP POST and GET request handling possible. For example, if you run your dash app publicly, you should use some validation logic to validate the HTTP requests from the users!
class HttpRequestHandler(BaseHttpRequestHandler):
# You may use the flask.request
# and flask.session inside the methods of this
# class when needed.
def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)
def post_before(self):
pass
def post(self):
self.post_before()
returnvalue = super().post()
self.post_after()
return returnvalue
def post_after(self):
pass
def get_before(self):
pass
def get(self):
self.get_before()
returnvalue = super().get()
self.get_after()
return returnvalue
def get_after(self):
pass
- The React Component of
dash-uploader
sends HTTP POST requests. It could in the future send also HTTP GET requests, or other HTTP requests. - The Flask server (Dash uses Flask underneath) is configured in
configure_upload
to call thepost()
function of thehttp_request_handler
on every HTTP POST request directed to the API endpoint of the dash-uploader. - The values in the POST request are listed in Documentation of flow.js. Most interesting from these is probably the filename (
resumableFilename
). In addition to these, there isupload_id
added by thedash-uploader
, ifuse_upload_id=True
when callingconfigure_upload
. - You can use the flask.request and flask.session proxies as you like. There you get access to all the HTTP Request parameters and Cookies, for example. As an quick example, to get the request filename, upload_id and some cookie value, you can use:
from flask import request
filename = request.form.get("flowFilename", default="error", type=str)
upload_id = request.form.get("upload_id", default="", type=str)
cookie_value = request.cookies.get('some_cookie')
- Please refer to the developer documentation for the details.