Asynchronous downloads scaffolding for Django projects
Project description
Django Async Downloads
Asynchronous downloads scaffolding for Django projects.
- TODO: describe cache mechanism
- TODO: how to customise the front end
Installation
# Without websocket support
pip install django-async-downloads
# With Websocket support
pip install django-async-downloads[ws]
Note at the present time on Windows, if you require websocket support,
you will need "Microsoft Visual C++ 14.0" installed (https://visualstudio.microsoft.com/visual-cpp-build-tools/)
due to hiredis
from the dependency chain (channels_redis
-> aioredis-py
-> hiredis
).
Add to your INSTALLED_APPS
:
INSTALLED_APPS = [
...
"async_downloads",
]
Add to your project's url_patterns
:
path("async_downloads/", include("async_downloads.urls"))
Add the CSS:
<link rel="stylesheet" type="text/css" href="{% static 'css/async_downloads.css' %}" />
Add the JS:
<script src="{% static "js/async_downloads.js" %}" id="async-downloads-script"
data-url="{% url 'async_downloads:ajax_update' %}"
data-clear-url="{% url 'async_downloads:ajax_clear_download' %}"></script>
Include the download centre nav-menu:
<ul class="navbar-nav">
...
{% include 'async_downloads/download_centre.html' %}
...
WS mode
Package can be used with websockets to provide asynchronous communication between frontend and backend. If package will be used in websockets mode additional settings must be applied.
-
Inside common settings
WS_MODE
must be toggled on.ASYNC_DOWNLOADS_WS_MODE = True
-
Application needs to be configured as a
ASGI
and proper WS urls need to be configured:# ws/urls.py from async_downloads.ws_consumers import DownloadsConsumer urlpatterns += [ re_path(r"ws/downloads/(?P<username>[\w.]+)/$", DownloadsConsumer.as_asgi()), ]
# example asgi.py from django.core.asgi import get_asgi_application from channels.auth import AuthMiddlewareStack from channels.routing import ProtocolTypeRouter, URLRouter import ws.urls application = ProtocolTypeRouter( { "http": get_asgi_application(), "websocket": AuthMiddlewareStack(URLRouter(ws.urls.urlpatterns)), } )
-
Change in application template async-downloads section to use WS js version.
# example: FULL_WS_URL = "http://app.com/ws/downloads/" <script src="{% static "js/ws_async_downloads.js" %}" id="async-downloads-script" data-url="{{ FULL_WS_URL }}" </script>
data-url
must be an absolute URL because this is required for a WebSocket connection, and it must include the protocol because thews_async_downloads.js
script will inspect it to determine which WebSockets protocol to use -ws
ifhttp
orwss
ifhttps
.
-
Configure
CHANNEL_LAYERS
inside common settings. Example config:CHANNEL_LAYERS = { "default": { "BACKEND": "channels_redis.core.RedisChannelLayer", "CONFIG": { "hosts": [("127.0.0.1", 6379)], }, }, }
Usage
TODO: using the JS
Cache Functions
cache.init_download
Initialise a download by preparing the cache entries. Returns a tuple of keys, the collection key and specific download key. You would typically want to call this within the web process, and pass the download key (and possibly the collection key) into the asynchronous function call so that the status can be updated.
Arguments:
user
: the unique identifier for a collection of downloads - this will typically be a user object but can be user PK or username.filename
: the name of the file being downloaded (does not need to be unique)name
: (optional) the name to associate with this download - defaults tofilename
cache.save_download
The asynchronous process should call this function when the iterable or file is prepared in order to save the output.
Arguments:
download_key
: the cache key of this particular downloaditerable
: (optional) an iterable of data rows to be writtenfile
: (optional) a BytesIO object
cache.update_percentage
The asynchronous process can call this function to calculate and update the completion percentage. (download_key, total, cur, resolution=10): Arguments:
download_key
: the cache key of this particular downloadtotal
: total to compare current progress againstcur
: current progress indexresolution
: resolution of the percentage calculation (make smaller for fewer updates, larger for more precision); default is10
, meaning it will increase in steps of 10%. The value is capped between1
and100
.
cache.set_percentage
The asynchronous process can call this function to directly set the completion percentage.
Arguments:
download_key
: the cache key of this particular downloadpercentage
: an number between 0 and 100 (inclusive)
cache.cleanup_collection
There can be a build up of expired download keys in a collection so it can be worth periodically
removing them using this function. (Note that the collection cache itself will expire if not touched
for ASYNC_DOWNLOADS_TIMEOUT
seconds, so it may not be critical to use this.)
Arguments:
collection_key
: the cache key of a collection of downloads
cache.set_error
The asynchronous process can call this function to set an error message on the download.
Arguments:
download_key
: the cache key of this particular downloaderror
: the error message to set
tasks.cleanup_expired_downloads
Delete expired downloads (where the download no longer exists in the cache). This is a clean up operation to prevent downloads that weren't manually deleted from building up, and should be run periodically to avoid bloating the server with files.
This is best setup as a periodic task in your project, which can be done by adding the following
to your project's celery.py
:
app.conf.beat_schedule = {
"async_downloads_cleanup": {
"task": "async_downloads.tasks.cleanup_expired_downloads",
"schedule": crontab(hour=0, minute=0, day_of_week=1)
}
}
Configurable Settings
ASYNC_DOWNLOADS_TIMEOUT
Default: 60 * 60 * 24
(1 day)
This is the cache timeout used for cache operations.
ASYNC_DOWNLOADS_DOWNLOAD_TEMPLATE
Default: "async_downloads/downloads.html"
This is the template that will be used by the ajax_update
view. You can override it by creating
a template in <project>/templates/async_downloads/downloads.html
, or else putting the template
wherever you choose and changing this setting.
ASYNC_DOWNLOADS_PATH_PREFIX
Default: "downloads"
The parent directory for all downloads in the MEDIA_ROOT
directory.
ASYNC_DOWNLOADS_COLLECTION_KEY_FORMAT
Default: "async_downloads/{}"
The collection key keeps track of the cache keys of a grouped collection of downloads. In the
unlikely event that this key format clashes with something in your project, you can change it.
The expectation is for the string to have a user primary key inserted with str.format
, so {}
is required to be present.
ASYNC_DOWNLOADS_WS_CHANNEL_NAME
Default: "downloads"
The channel name for all shared information about download in channels cache layer.
ASYNC_DOWNLOADS_CACHE_NAME
If this settings will be added default cache will be changed into new with provided name.
ASYNC_DOWNLOADS_WS_MODE
Default: False
If this flag will be set to True
package will be set to work with WebSockets in WS_MODE.
ASYNC_DOWNLOADS_CSS_CLASS
Default: None
Additional CSS class can be added to download-content
div.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
File details
Details for the file django-async-downloads-0.5.2.tar.gz
.
File metadata
- Download URL: django-async-downloads-0.5.2.tar.gz
- Upload date:
- Size: 16.5 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/5.0.0 CPython/3.12.2
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | 2188bc3852f8c5de40850523c7738aa9463c44b75a131247086dbbb58216db88 |
|
MD5 | 0f8500efb76db360e0f035a94f96fc04 |
|
BLAKE2b-256 | 6884a36e95b4e45bda60c2ebd797a78f6e4b8812086d7bb8cfd8ca46ed265abc |
File details
Details for the file django_async_downloads-0.5.2-py3-none-any.whl
.
File metadata
- Download URL: django_async_downloads-0.5.2-py3-none-any.whl
- Upload date:
- Size: 17.0 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/5.0.0 CPython/3.12.2
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | e0920278baf835c1b32e1b5fc35493eaf607de595a7a9d07ea110c7bf5737549 |
|
MD5 | 106368070d0250af420519d437ee8206 |
|
BLAKE2b-256 | c7e1840a8799ca3e8ab6216fcbbb36ae412778fccba995fe8720efb9300103b2 |