django-distill


Namedjango-distill JSON
Version 3.2.7 PyPI version JSON
download
home_pagehttps://github.com/meeb/django-distill
SummaryStatic site renderer and publisher for Django.
upload_time2024-08-30 14:07:07
maintainerNone
docs_urlNone
authorhttps://github.com/meeb
requires_pythonNone
licenseMIT
keywords django distill static website jamstack s3 amazon s3 aws amazon google microsoft google cloud google cloud storage azure azure storage azure blob storage
VCS
bugtrack_url
requirements django requests
Travis-CI No Travis.
coveralls test coverage No coveralls.
            # django-distill

`django-distill` now has a website. Read more at:

## :link: https://django-distill.com/

`django-distill` is a minimal configuration static site generator and publisher
for Django. Most Django versions are supported, however up to date versions are
advised including the Django 3.x releases. `django-distill` as of the 1.7 release
only supports Python 3. Python 2 support has been dropped. If you require Python 2
support please pin `django-distill` to version 1.6 in your requirements.txt or
Pipfile. Python 3.6 or above is advised.

`django-distill` extends existing Django sites with the ability to export
fully functional static sites. It is suitable for sites such as blogs that have
a mostly static front end but you still want to use a CMS to manage the
content.

`django-distill` iterates over URLs in your Django project using easy to write
iterable functions to yield the parameters for whatever pages you want to save
as static HTML. These static files can be automatically uploaded to a bucket-style
remote container such as Amazon S3, Googe Cloud Files, Microsoft Azure Storage,
or, written to a local directory as a fully working local static version of
your project. The site generation, or distillation process, can be easily
integrated into CI/CD workflows to auto-deploy static sites on commit.
`django-distill` can be defined as an extension to Django to make Django
projects compatible with "Jamstack"-style site architecture.

`django-distill` plugs directly into the existing Django framework without the
need to write custom renderers or other more verbose code. You can also integrate
`django-distill` with existing dynamic sites and just generate static pages for
a small subsection of pages rather than the entire site.

For static files on CDNs you can use the following 'cache buster' library to
allow for fast static media updates when pushing changes:

[:link: meeb/django-cachekiller](https://github.com/meeb/django-cachekiller)

There is a complete example site that creates a static blog and uses
`django-distill` with `django-cachekiller` via continuous deployment on Netlify
available here:

[:link: meeb/django-distill-example](https://github.com/meeb/django-distill-example)


# Installation

Install from pip:

```bash
$ pip install django-distill
```

Add `django_distill` to your `INSTALLED_APPS` in your `settings.py`:

```python
INSTALLED_APPS = [
    # ... other apps here ...
    'django_distill',
]
```

That's it.


# Limitations

`django-distill` generates static pages and therefore only views which allow
`GET` requests that return an `HTTP 200` status code are supported.

It is assumed you are using URI parameters such as `/blog/123-abc` and not
querystring parameters such as `/blog?post_id=123&title=abc`. Querystring
parameters do not make sense for static page generation for obvious reasons.

Static media files such as images and style sheets are copied from your static
media directory defined in `STATIC_ROOT`. This means that you will want to run
`./manage.py collectstatic` **before** you run `./manage.py distill-local`
if you have made changes to static media. `django-distill` doesn't chain this
request by design, however you can enable it with the `--collectstatic`
argument.


# Usage

Assuming you have an existing Django project, edit a `urls.py` to include the
`distill_path` function which replaces Django's standard `path` function and
supports the new keyword arguments `distill_func` and `distill_file`.

The `distill_func` argument should be provided with a function or callable
class that returns an iterable or `None`.

The `distill_file` argument is entirely optional and allows you to override the
URL that would otherwise be generated from the reverse of the URL regex. This
allows you to rename URLs like `/example` to any other name like
`example.html`. As of v0.8 any URIs ending in a slash `/` are automatically
modified to end in `/index.html`. You can use format string parameters in the
`distill_file` to customise the file name, arg values from the URL will be
substituted in, for example `{}` for positional args or `{param_name}` for
named args.

An example distill setup for a theoretical blogging app would be:

```python
# Replaces the standard django.conf.path, identical syntax
from django_distill import distill_path

# Views and models from a theoretical blogging app
from blog.views import PostIndex, PostView, PostYear
from blog.models import Post

def get_index():
    # The index URI path, '', contains no parameters, named or otherwise.
    # You can simply just return nothing here.
    return None

def get_all_blogposts():
    # This function needs to return an iterable of dictionaries. Dictionaries
    # are required as the URL this distill function is for has named parameters.
    # You can just export a small subset of values here if you wish to
    # limit what pages will be generated.
    for post in Post.objects.all():
        yield {'blog_id': post.id, 'blog_title': post.title}

def get_years():
    # You can also just return an iterable containing static strings if the
    # URL only has one argument and you are using positional URL parameters:
    return (2014, 2015)
    # This is really just shorthand for ((2014,), (2015,))

urlpatterns = (
    # e.g. / the blog index
    distill_path('',
                 PostIndex.as_view(),
                 name='blog-index',
                 # Note that for paths which have no paramters
                 # distill_func is optional
                 distill_func=get_index,
                 # '' is not a valid file name! override it to index.html
                 distill_file='index.html'),
    # e.g. /post/123-some-post-title using named parameters
    distill_path('post/<int:blog_id>-<slug:blog_title>.html',
                 PostView.as_view(),
                 name='blog-post',
                 distill_func=get_all_blogposts),
    # e.g. /posts-by-year/2015 using positional parameters
    # url ends in / so file path will have /index.html appended
    distill_path('posts-by-year/<int:year>/',
                 PostYear.as_view(),
                 name='blog-year',
                 distill_func=get_years),
)
```

Your site will still function identically with the above changes. Internally
the `distill_func` and `distill_file` parameters are removed and the URL is
passed back to Django for normal processing. This has no runtime performance
impact as this happens only once upon starting the application.

If your path has no URI paramters, such as `/` or `/some-static-url` you do
not have to specify the `distill_func` parameter if you don't want to. As for
paths with no parameters the `distill_func` always returns `None`, this is set
as the default behaviour for `distill_func`s.

You can use the `distill_re_path` function as well, which replaces the default
`django.urls.re_path` function. Its usage is identical to the above:

```python
from django_distill import distill_re_path

urlpatterns = (
    distill_re_path(r'some/regex'
                    SomeOtherView.as_view(),
                    name='url-other-view',
                    distill_func=some_other_func),
)

```

If you are using an older version of Django in the 1.x series you can use the
`distill_url` function instead which replaces the `django.conf.urls.url` or
`django.urls.url` functions. Its usage is identical to the above:

```python
from django_distill import distill_url

urlpatterns = (
    distill_url(r'some/regex'
                SomeView.as_view(),
                name='url-view',
                distill_func=some_func),
)
```

### Parameters in file names

You can use standard Python string formatting in `distill_file` as well to enable
you to change the output file path for a file if you wish. Note this does not
update the URL used by Django so if you use this make sure your `path` pattern
matches the `distill_file` pattern or your links might not work in Django. An
example:

```python
# Override file path with parameters. Values are taken from the URL pattern
urlpatterns = (
    distill_path('post/<int:blog_id>-<slug:blog_title>.html',
                 PostView.as_view(),
                 name='blog-post',
                 distill_func=get_all_blogposts,
                 distill_file="post/{blog_id}-{blog_title}.html"
)
```

### Non-standard status codes

All views rendered by `django-distill` into static pages must return an HTTP 200 status
code. If for any reason you need to render a view which does not return an HTTP 200
status code, for example you also want to statically generate a 404 page which has a
view which (correctly) returns an HTTP 404 status code you can use the
`distill_status_codes` optional argument to a view. For example:

```python
from django_distill import distill_url

urlpatterns = (
    distill_url(r'some/regex'
                SomeView.as_view(),
                name='url-view',
                distill_status_codes=(200, 404),
                distill_func=some_func),
)
```

The optional `distill_status_codes` argument accepts a tuple of status codes as integers
which are permitted for the view to return without raising an error. By default this is
set to `(200,)` but you can override it if you need to for your site.

### Tracking Django's URL function support

`django-distill` will mirror whatever your installed version of Django supports,
therefore at some point the `distill_url` function will cease working in the future
when Django 2.x itself depreciates the `django.conf.urls.url` and `django.urls.url`
functions. You can use `distill_re_path` as a drop-in replacement. It is advisable to
use `distill_path` or `distill_re_path` if you're building a new site now.


### Internationalization

Internationalization is only supported for URLs, page content is unable to be
dynamically translated. By default your site will be generated using the
`LANGUAGE_CODE` value in your `settings.py`. If you also set `settings.USE_I18N` to
`True` then set other language codes in your `settings.DISTILL_LANGUAGES` value and register
URLs with `i18n_patterns(...)` then your site will be generated in multiple languges.
This assumes your multi-language site works as expected before adding `django-distill`.

For example if you set `settings.LANGUAGE_CODE = 'en'` your site will be
generated in one language.

If you have something like this in your `settings.py` instead:

```python
USE_I18N = True

DISTILL_LANGUAGES = [
    'en',
    'fr',
    'de',
]
```

While also using `i18n_patterns`in your `urls.py` like so:

```python
from django.conf.urls.i18n import i18n_patterns
from django_distill import distill_path

urlpatterns = i18n_patterns(
    distill_path('some-file.html',
                 SomeView.as_view(),
                 name='i18n-view',
                 distill_func=some_func
    )
)
```

Then your views will be generaged as `/en/some-file.html`, `/fr/some-file.html`
and `/de/some-file.html`. These URLs should work (and be translated) by your
site already. `django-distill` doesn't do any translation magic, it just
calls the URLs with the language code prefix.

**Note** While the default suggested method is to use `settings.DISTILL_LANGUAGES`
to keep things seperate `django-distill` will also check `settings.LANGUAGES` for
language codes.


### Sitemaps

You may need to generate a list of all the URLs registered with `django-distill`.
For example, you have a statically generated blog with a few hundred pages and
you want to list all of the URLs easily in a `sitemap.xml` or other similar list
of all URLs. You could wrap your sitemap view in `distill_path` then replicate
all of your URL generation logic by importing your views `distill_func`s from
your `urls.py` and generating these all manually, but given this is quite a hassle
there's a built-in helper to generate all your URLs that will be distilled for you.

```python
from django_distill import distilled_urls

for uri, file_name in distilled_urls():
    # URI is the generated, complete URI for the page
    print(uri)        # for example: /blog/my-post-123/
    # file_name is the actual file name on disk, this may be None or a string
    print(file_name)  # for example: /blog/my-post-123/index.html
```

**Note** that `distilled_urls()` will only return URLs after all of your URLs
in `urls.py` have been loaded with `distill_path(...)`.


# The `distill-local` command

Once you have wrapped the URLs you want to generate statically you can now
generate a complete functioning static site with:

```bash
$ ./manage.py distill-local [optional /path/to/export/directory]
```

Under the hood this simply iterates all URLs registered with `distill_url` and
generates the pages for them using parts of the Django testing framework to
spoof requests. Once the site pages have been rendered then files from the
`STATIC_ROOT` are copied over. Existing files with the same name are replaced in
the target directory and orphan files are deleted.

`distill-local` supports the following optional arguments:

`--collectstatic`: Automatically run `collectstatic` on your site before
rendering, this is just a shortcut to save you typing an extra command.

`--quiet`: Disable all output other than asking confirmation questions.

`--force`: Assume 'yes' to all confirmation questions.

`--exclude-staticfiles`: Do not copy any static files at all, only render output from
Django views.

`--parallel-render [number of threads]`: Render files in parallel on multiple
threads, this can speed up rendering. Defaults to `1` thread.

`--generate-redirects`: Attempt to generate static redirects stored in the
`django.contrib.redirects` app. If you have a redirect from `/old/` to `/new/` using
this flag will create a static HTML `<meta http-equiv="refresh" content="...">`
style redirect at `/old/index.html` to `/new/`.

**Note** If any of your views contain a Python error then rendering will fail
then the stack trace will be printed to the terminal and the rendering command
will exit with a status code of 1.


# The `distill-publish` command

```bash
$ ./manage.py distill-publish [optional destination here]
```

If you have configured at least one publishing destination (see below) you can
use the `distill-publish` command to publish the site to a remote location.

This will perform a full synchronisation, removing any remote files that are no
longer present in the generated static site and uploading any new or changed
files. The site will be built into a temporary directory locally first when
publishing which is deleted once the site has been published. Each file will be
checked that it has been published correctly by requesting it via the
`PUBLIC_URL`.

`distill-publish` supports the following optional arguments:

`--collectstatic`: Automatically run `collectstatic` on your site before
rendering, this is just a shortcut to save you typing an extra command.

`--quiet`: Disable all output other than asking confirmation questions.

`--force`: Assume 'yes' to all confirmation questions.

`--exclude-staticfiles`: Do not copy any static files at all, only render output from
Django views.

`--skip-verify`: Do not test if files are correctly uploaded on the server.

`--ignore-remote-content`: Do not fetch the list of remote files. It means that all
files will be uploaded, and no existing remote file will be  deleted. This can be
useful if you have a lot of files on the remote server, and you know that you want
to update most of them, and you don't care if old files remain on the server.

`--parallel-publish [number of threads]`: Publish files in parallel on multiple
threads, this can speed up publishing. Defaults to `1` thread.

`--parallel-render [number of threads]`: Render files in parallel on multiple
threads, this can speed up rendering. Defaults to `1` thread.

`--generate-redirects`: Attempt to generate static redirects stored in the
`django.contrib.redirects` app. If you have a redirect from `/old/` to `/new/` using
this flag will create a static HTML `<meta http-equiv="refresh" content="...">`
style redirect at `/old/index.html` to `/new/`.

**Note** that this means if you use `--force` and `--quiet` that the output
directory will have all files not part of the site export deleted without any
confirmation.

**Note**  If any of your views contain a Python error then rendering will fail
then the stack trace will be printed to the terminal and the rendering command
will exit with a status code of 1.


# The `distill-test-publish` command

```bash
$ ./manage.py distill-test-publish [optional destination here]
```

This will connect to your publishing target, authenticate to it, upload a
randomly named file, verify it exists on the `PUBLIC_URL` and then delete it
again. Use this to check your publishing settings are correct.

`distill-test-publish` has no arguments.


# Optional configuration settings

You can set the following optional `settings.py` variables:

**DISTILL_DIR**: string, default directory to export to:

```python
DISTILL_DIR = '/path/to/export/directory'
```

**DISTILL_PUBLISH**: dictionary, like Django's `settings.DATABASES`, supports
`default`:

```python
DISTILL_PUBLISH = {
    'default': {
        ... options ...
    },
    'some-other-target': {
        ... options ...
    },
}
```

**DISTILL_SKIP_ADMIN_DIRS**: bool, defaults to `True`

```python
DISTILL_SKIP_ADMIN_DIRS = True
```

Set `DISTILL_SKIP_ADMIN_DIRS` to `False` if you want `django-distill` to also copy over
static files in the `static/admin` directory. Usually, these are not required or
desired for statically generated sites. The default behaviour is to skip static admin
files.


**DISTILL_SKIP_STATICFILES_DIRS**: list, defaults to `[]`

```python
DISTILL_SKIP_STATICFILES_DIRS = ['some_dir']
```

Set `DISTILL_SKIP_STATICFILES_DIRS` to a list of directory names you want `django-distill`
to ignore directories in your defined `static/` directory. You can use this to ignore
copying directories containing files from apps you're not using that get bundled into your
`static/` directory by `collect-static`. For example if you set `DISTILL_SKIP_STATICFILES_DIRS`
to `['some_dir']` the static files directory `static/some_dir` would be skipped.


**DISTILL_LANGUAGES**: list, defaults to `[]`

```python
DISTILL_LANGUAGES = [
    'en',
    'fr',
    'de',
]
```

Set `DISTILL_LANGUAGES` to a list of language codes to attempt to render URLs with.
See the "Internationalization" section for more details.


# Developing locally with HTTPS

If you are using a local development environment which has HTTPS support you may need
to add `SECURE_SSL_REDIRECT = False` to your `settings.py` to prevent a `CommandError`
being raised when a request returns a 301 redirect instead of the expected HTTP/200
response code.


# Writing single files

As of `django-distill` version `3.0.0` you can use the
`django_distill.renderer.render_single_file` method to write out a single file
to disk using `django_distill`. This is useful for writing out single files to disk,
for example, you have a Django site which has some static files in a directory
written by `django_distill` but the rest of the site is a normal dynamic Django site.
You can update a static HTML file every time a model instance is saved. You can
use single file writing with signals to achieve this. For example:

```python
# in models.py
from django.db.models.signals import post_save
from django.dispatch import receiver
from django_distill.renderer import render_single_file

@receiver(post_save, sender=SomeBlogPostModel)
def write_blog_post_static_file_post_save(sender, **kwargs):
    render_single_file(
        '/path/to/output/directory',
        'blog-post-view-name',
        blog_id=sender.pk,
        blog_slug=sender.slug
    )
```

The syntax for `render_single_file` is similar to Django's `url.reverse`. The full
usage interface is:

```python
render_single_file(
    '/path/to/output/directory',
    'view-name-set-in-urls-py',
    *view_args,
    **view_kwargs
)
```

For example, if you had a blog post URL defined as:

```python
    # in urls.py
    distill_path('post/<int:blog_id>_<slug:blog_slug>.html',
                 PostView.as_view(),
                 name='blog-post',
                 distill_func=get_all_blogposts),
```

Your usage would be:

```python
render_single_file(
    '/path/to/output/directory',
    'blog-post',
    blog_id=123,
    blog_slug='blog-title-slug',
)
```

which would write out the contents of `/post/123_blog-title-slug.html` into
`/path/to/output/directory` as the file
`/path/to/output/directory/post/123_blog-title-slug.html`. Note any required
sub-directories (`/path/to/output/directory/post` in this example) will be
automatically created if they don't already exist. All `django-distill` rules
apply, such as URLs ending in `/` will be saved as `/index.html` to make sense
for a physical file on disk.

Also note that `render_single_file` can only be imported and used into an
initialised Django project.


# Publishing targets

You can automatically publish sites to various supported remote targets through
backends just like how you can use MySQL, SQLite, PostgreSQL etc. with
Django by changing the backend database engine. Currently the engines supported
by `django-distill` are:

**django_distill.backends.amazon_s3**: Publish to an Amazon S3 bucket. Requires
  the Python library `boto3` (`$ pip install django-distill[amazon]`). The bucket
  must already exist (use the AWS control panel). Options:

```python
'some-s3-container': {
    'ENGINE': 'django_distill.backends.amazon_s3',
    'PUBLIC_URL': 'http://.../',
    'ACCESS_KEY_ID': '...',
    'SECRET_ACCESS_KEY': '...',
    'BUCKET': '...',
    'ENDPOINT_URL': 'https://.../',  # Optional, set to use a different S3 endpoint
    'DEFAULT_CONTENT_TYPE': 'application/octet-stream',  # Optional
},
```

**django_distill.backends.google_storage**: Publish to a Google Cloud Storage
  bucket. Requires the Python libraries `google-api-python-client` and
  `google-cloud-storage`
  (`$ pip install django-distill[google]`). The bucket
  must already exist and be set up to host a public static website (use the
  Google Cloud control panel). Options:

```python
'some-google-storage-bucket': {
    'ENGINE': 'django_distill.backends.google_storage',
    'PUBLIC_URL': 'https://storage.googleapis.com/[bucket.name.here]/',
    'BUCKET': '[bucket.name.here]',
    'JSON_CREDENTIALS': '/path/to/some/credentials.json',
},
```

Note that `JSON_CREDENTIALS` is optional; if it is not specified, the google libraries
will try other authentication methods, in the search order described here:
https://cloud.google.com/docs/authentication/application-default-credentials (e.g. the
`GOOGLE_APPLICATION_CREDENTIALS` environment variable, attached service account, etc).


**django_distill.backends.microsoft_azure_storage**: Publish to a Microsoft
  Azure Blob Storage container. Requires the Python library
  `azure-storage-blob` (`$ pip install django-distill[microsoft]`). The storage
  account must already exist and be set up to host a public static website
  (use the Microsoft Azure control panel). Options:

```python
'some-microsoft-storage-account': {
    'ENGINE': 'django_distill.backends.microsoft_azure_storage',
    'PUBLIC_URL': 'https://[storage-account-name]...windows.net/',
    'CONNECTION_STRING': '...',
},
```

Note that each Azure storage account supports one static website using the
magic container `$web` which is where `django-distill` will attempt to
publish your site.


# Tests

There is a minimal test suite, you can run it by cloing this repository,
installing the required dependancies in `requirements.txt` then execuiting:

```bash
# ./run-tests.py
```


# Contributing

All properly formatted and sensible pull requests, issues and comments are
welcome.

            

Raw data

            {
    "_id": null,
    "home_page": "https://github.com/meeb/django-distill",
    "name": "django-distill",
    "maintainer": null,
    "docs_url": null,
    "requires_python": null,
    "maintainer_email": null,
    "keywords": "django, distill, static, website, jamstack, s3, amazon s3, aws, amazon, google, microsoft, google cloud, google cloud storage, azure, azure storage, azure blob storage",
    "author": "https://github.com/meeb",
    "author_email": "meeb@meeb.org",
    "download_url": "https://files.pythonhosted.org/packages/19/c8/ece20c91129314a1ec407ecc7df87102524a32615ea0c43be5b3ae54cc4b/django_distill-3.2.7.tar.gz",
    "platform": null,
    "description": "# django-distill\n\n`django-distill` now has a website. Read more at:\n\n## :link: https://django-distill.com/\n\n`django-distill` is a minimal configuration static site generator and publisher\nfor Django. Most Django versions are supported, however up to date versions are\nadvised including the Django 3.x releases. `django-distill` as of the 1.7 release\nonly supports Python 3. Python 2 support has been dropped. If you require Python 2\nsupport please pin `django-distill` to version 1.6 in your requirements.txt or\nPipfile. Python 3.6 or above is advised.\n\n`django-distill` extends existing Django sites with the ability to export\nfully functional static sites. It is suitable for sites such as blogs that have\na mostly static front end but you still want to use a CMS to manage the\ncontent.\n\n`django-distill` iterates over URLs in your Django project using easy to write\niterable functions to yield the parameters for whatever pages you want to save\nas static HTML. These static files can be automatically uploaded to a bucket-style\nremote container such as Amazon S3, Googe Cloud Files, Microsoft Azure Storage,\nor, written to a local directory as a fully working local static version of\nyour project. The site generation, or distillation process, can be easily\nintegrated into CI/CD workflows to auto-deploy static sites on commit.\n`django-distill` can be defined as an extension to Django to make Django\nprojects compatible with \"Jamstack\"-style site architecture.\n\n`django-distill` plugs directly into the existing Django framework without the\nneed to write custom renderers or other more verbose code. You can also integrate\n`django-distill` with existing dynamic sites and just generate static pages for\na small subsection of pages rather than the entire site.\n\nFor static files on CDNs you can use the following 'cache buster' library to\nallow for fast static media updates when pushing changes:\n\n[:link: meeb/django-cachekiller](https://github.com/meeb/django-cachekiller)\n\nThere is a complete example site that creates a static blog and uses\n`django-distill` with `django-cachekiller` via continuous deployment on Netlify\navailable here:\n\n[:link: meeb/django-distill-example](https://github.com/meeb/django-distill-example)\n\n\n# Installation\n\nInstall from pip:\n\n```bash\n$ pip install django-distill\n```\n\nAdd `django_distill` to your `INSTALLED_APPS` in your `settings.py`:\n\n```python\nINSTALLED_APPS = [\n    # ... other apps here ...\n    'django_distill',\n]\n```\n\nThat's it.\n\n\n# Limitations\n\n`django-distill` generates static pages and therefore only views which allow\n`GET` requests that return an `HTTP 200` status code are supported.\n\nIt is assumed you are using URI parameters such as `/blog/123-abc` and not\nquerystring parameters such as `/blog?post_id=123&title=abc`. Querystring\nparameters do not make sense for static page generation for obvious reasons.\n\nStatic media files such as images and style sheets are copied from your static\nmedia directory defined in `STATIC_ROOT`. This means that you will want to run\n`./manage.py collectstatic` **before** you run `./manage.py distill-local`\nif you have made changes to static media. `django-distill` doesn't chain this\nrequest by design, however you can enable it with the `--collectstatic`\nargument.\n\n\n# Usage\n\nAssuming you have an existing Django project, edit a `urls.py` to include the\n`distill_path` function which replaces Django's standard `path` function and\nsupports the new keyword arguments `distill_func` and `distill_file`.\n\nThe `distill_func` argument should be provided with a function or callable\nclass that returns an iterable or `None`.\n\nThe `distill_file` argument is entirely optional and allows you to override the\nURL that would otherwise be generated from the reverse of the URL regex. This\nallows you to rename URLs like `/example` to any other name like\n`example.html`. As of v0.8 any URIs ending in a slash `/` are automatically\nmodified to end in `/index.html`. You can use format string parameters in the\n`distill_file` to customise the file name, arg values from the URL will be\nsubstituted in, for example `{}` for positional args or `{param_name}` for\nnamed args.\n\nAn example distill setup for a theoretical blogging app would be:\n\n```python\n# Replaces the standard django.conf.path, identical syntax\nfrom django_distill import distill_path\n\n# Views and models from a theoretical blogging app\nfrom blog.views import PostIndex, PostView, PostYear\nfrom blog.models import Post\n\ndef get_index():\n    # The index URI path, '', contains no parameters, named or otherwise.\n    # You can simply just return nothing here.\n    return None\n\ndef get_all_blogposts():\n    # This function needs to return an iterable of dictionaries. Dictionaries\n    # are required as the URL this distill function is for has named parameters.\n    # You can just export a small subset of values here if you wish to\n    # limit what pages will be generated.\n    for post in Post.objects.all():\n        yield {'blog_id': post.id, 'blog_title': post.title}\n\ndef get_years():\n    # You can also just return an iterable containing static strings if the\n    # URL only has one argument and you are using positional URL parameters:\n    return (2014, 2015)\n    # This is really just shorthand for ((2014,), (2015,))\n\nurlpatterns = (\n    # e.g. / the blog index\n    distill_path('',\n                 PostIndex.as_view(),\n                 name='blog-index',\n                 # Note that for paths which have no paramters\n                 # distill_func is optional\n                 distill_func=get_index,\n                 # '' is not a valid file name! override it to index.html\n                 distill_file='index.html'),\n    # e.g. /post/123-some-post-title using named parameters\n    distill_path('post/<int:blog_id>-<slug:blog_title>.html',\n                 PostView.as_view(),\n                 name='blog-post',\n                 distill_func=get_all_blogposts),\n    # e.g. /posts-by-year/2015 using positional parameters\n    # url ends in / so file path will have /index.html appended\n    distill_path('posts-by-year/<int:year>/',\n                 PostYear.as_view(),\n                 name='blog-year',\n                 distill_func=get_years),\n)\n```\n\nYour site will still function identically with the above changes. Internally\nthe `distill_func` and `distill_file` parameters are removed and the URL is\npassed back to Django for normal processing. This has no runtime performance\nimpact as this happens only once upon starting the application.\n\nIf your path has no URI paramters, such as `/` or `/some-static-url` you do\nnot have to specify the `distill_func` parameter if you don't want to. As for\npaths with no parameters the `distill_func` always returns `None`, this is set\nas the default behaviour for `distill_func`s.\n\nYou can use the `distill_re_path` function as well, which replaces the default\n`django.urls.re_path` function. Its usage is identical to the above:\n\n```python\nfrom django_distill import distill_re_path\n\nurlpatterns = (\n    distill_re_path(r'some/regex'\n                    SomeOtherView.as_view(),\n                    name='url-other-view',\n                    distill_func=some_other_func),\n)\n\n```\n\nIf you are using an older version of Django in the 1.x series you can use the\n`distill_url` function instead which replaces the `django.conf.urls.url` or\n`django.urls.url` functions. Its usage is identical to the above:\n\n```python\nfrom django_distill import distill_url\n\nurlpatterns = (\n    distill_url(r'some/regex'\n                SomeView.as_view(),\n                name='url-view',\n                distill_func=some_func),\n)\n```\n\n### Parameters in file names\n\nYou can use standard Python string formatting in `distill_file` as well to enable\nyou to change the output file path for a file if you wish. Note this does not\nupdate the URL used by Django so if you use this make sure your `path` pattern\nmatches the `distill_file` pattern or your links might not work in Django. An\nexample:\n\n```python\n# Override file path with parameters. Values are taken from the URL pattern\nurlpatterns = (\n    distill_path('post/<int:blog_id>-<slug:blog_title>.html',\n                 PostView.as_view(),\n                 name='blog-post',\n                 distill_func=get_all_blogposts,\n                 distill_file=\"post/{blog_id}-{blog_title}.html\"\n)\n```\n\n### Non-standard status codes\n\nAll views rendered by `django-distill` into static pages must return an HTTP 200 status\ncode. If for any reason you need to render a view which does not return an HTTP 200\nstatus code, for example you also want to statically generate a 404 page which has a\nview which (correctly) returns an HTTP 404 status code you can use the\n`distill_status_codes` optional argument to a view. For example:\n\n```python\nfrom django_distill import distill_url\n\nurlpatterns = (\n    distill_url(r'some/regex'\n                SomeView.as_view(),\n                name='url-view',\n                distill_status_codes=(200, 404),\n                distill_func=some_func),\n)\n```\n\nThe optional `distill_status_codes` argument accepts a tuple of status codes as integers\nwhich are permitted for the view to return without raising an error. By default this is\nset to `(200,)` but you can override it if you need to for your site.\n\n### Tracking Django's URL function support\n\n`django-distill` will mirror whatever your installed version of Django supports,\ntherefore at some point the `distill_url` function will cease working in the future\nwhen Django 2.x itself depreciates the `django.conf.urls.url` and `django.urls.url`\nfunctions. You can use `distill_re_path` as a drop-in replacement. It is advisable to\nuse `distill_path` or `distill_re_path` if you're building a new site now.\n\n\n### Internationalization\n\nInternationalization is only supported for URLs, page content is unable to be\ndynamically translated. By default your site will be generated using the\n`LANGUAGE_CODE` value in your `settings.py`. If you also set `settings.USE_I18N` to\n`True` then set other language codes in your `settings.DISTILL_LANGUAGES` value and register\nURLs with `i18n_patterns(...)` then your site will be generated in multiple languges.\nThis assumes your multi-language site works as expected before adding `django-distill`.\n\nFor example if you set `settings.LANGUAGE_CODE = 'en'` your site will be\ngenerated in one language.\n\nIf you have something like this in your `settings.py` instead:\n\n```python\nUSE_I18N = True\n\nDISTILL_LANGUAGES = [\n    'en',\n    'fr',\n    'de',\n]\n```\n\nWhile also using `i18n_patterns`in your `urls.py` like so:\n\n```python\nfrom django.conf.urls.i18n import i18n_patterns\nfrom django_distill import distill_path\n\nurlpatterns = i18n_patterns(\n    distill_path('some-file.html',\n                 SomeView.as_view(),\n                 name='i18n-view',\n                 distill_func=some_func\n    )\n)\n```\n\nThen your views will be generaged as `/en/some-file.html`, `/fr/some-file.html`\nand `/de/some-file.html`. These URLs should work (and be translated) by your\nsite already. `django-distill` doesn't do any translation magic, it just\ncalls the URLs with the language code prefix.\n\n**Note** While the default suggested method is to use `settings.DISTILL_LANGUAGES`\nto keep things seperate `django-distill` will also check `settings.LANGUAGES` for\nlanguage codes.\n\n\n### Sitemaps\n\nYou may need to generate a list of all the URLs registered with `django-distill`.\nFor example, you have a statically generated blog with a few hundred pages and\nyou want to list all of the URLs easily in a `sitemap.xml` or other similar list\nof all URLs. You could wrap your sitemap view in `distill_path` then replicate\nall of your URL generation logic by importing your views `distill_func`s from\nyour `urls.py` and generating these all manually, but given this is quite a hassle\nthere's a built-in helper to generate all your URLs that will be distilled for you.\n\n```python\nfrom django_distill import distilled_urls\n\nfor uri, file_name in distilled_urls():\n    # URI is the generated, complete URI for the page\n    print(uri)        # for example: /blog/my-post-123/\n    # file_name is the actual file name on disk, this may be None or a string\n    print(file_name)  # for example: /blog/my-post-123/index.html\n```\n\n**Note** that `distilled_urls()` will only return URLs after all of your URLs\nin `urls.py` have been loaded with `distill_path(...)`.\n\n\n# The `distill-local` command\n\nOnce you have wrapped the URLs you want to generate statically you can now\ngenerate a complete functioning static site with:\n\n```bash\n$ ./manage.py distill-local [optional /path/to/export/directory]\n```\n\nUnder the hood this simply iterates all URLs registered with `distill_url` and\ngenerates the pages for them using parts of the Django testing framework to\nspoof requests. Once the site pages have been rendered then files from the\n`STATIC_ROOT` are copied over. Existing files with the same name are replaced in\nthe target directory and orphan files are deleted.\n\n`distill-local` supports the following optional arguments:\n\n`--collectstatic`: Automatically run `collectstatic` on your site before\nrendering, this is just a shortcut to save you typing an extra command.\n\n`--quiet`: Disable all output other than asking confirmation questions.\n\n`--force`: Assume 'yes' to all confirmation questions.\n\n`--exclude-staticfiles`: Do not copy any static files at all, only render output from\nDjango views.\n\n`--parallel-render [number of threads]`: Render files in parallel on multiple\nthreads, this can speed up rendering. Defaults to `1` thread.\n\n`--generate-redirects`: Attempt to generate static redirects stored in the\n`django.contrib.redirects` app. If you have a redirect from `/old/` to `/new/` using\nthis flag will create a static HTML `<meta http-equiv=\"refresh\" content=\"...\">`\nstyle redirect at `/old/index.html` to `/new/`.\n\n**Note** If any of your views contain a Python error then rendering will fail\nthen the stack trace will be printed to the terminal and the rendering command\nwill exit with a status code of 1.\n\n\n# The `distill-publish` command\n\n```bash\n$ ./manage.py distill-publish [optional destination here]\n```\n\nIf you have configured at least one publishing destination (see below) you can\nuse the `distill-publish` command to publish the site to a remote location.\n\nThis will perform a full synchronisation, removing any remote files that are no\nlonger present in the generated static site and uploading any new or changed\nfiles. The site will be built into a temporary directory locally first when\npublishing which is deleted once the site has been published. Each file will be\nchecked that it has been published correctly by requesting it via the\n`PUBLIC_URL`.\n\n`distill-publish` supports the following optional arguments:\n\n`--collectstatic`: Automatically run `collectstatic` on your site before\nrendering, this is just a shortcut to save you typing an extra command.\n\n`--quiet`: Disable all output other than asking confirmation questions.\n\n`--force`: Assume 'yes' to all confirmation questions.\n\n`--exclude-staticfiles`: Do not copy any static files at all, only render output from\nDjango views.\n\n`--skip-verify`: Do not test if files are correctly uploaded on the server.\n\n`--ignore-remote-content`: Do not fetch the list of remote files. It means that all\nfiles will be uploaded, and no existing remote file will be  deleted. This can be\nuseful if you have a lot of files on the remote server, and you know that you want\nto update most of them, and you don't care if old files remain on the server.\n\n`--parallel-publish [number of threads]`: Publish files in parallel on multiple\nthreads, this can speed up publishing. Defaults to `1` thread.\n\n`--parallel-render [number of threads]`: Render files in parallel on multiple\nthreads, this can speed up rendering. Defaults to `1` thread.\n\n`--generate-redirects`: Attempt to generate static redirects stored in the\n`django.contrib.redirects` app. If you have a redirect from `/old/` to `/new/` using\nthis flag will create a static HTML `<meta http-equiv=\"refresh\" content=\"...\">`\nstyle redirect at `/old/index.html` to `/new/`.\n\n**Note** that this means if you use `--force` and `--quiet` that the output\ndirectory will have all files not part of the site export deleted without any\nconfirmation.\n\n**Note**  If any of your views contain a Python error then rendering will fail\nthen the stack trace will be printed to the terminal and the rendering command\nwill exit with a status code of 1.\n\n\n# The `distill-test-publish` command\n\n```bash\n$ ./manage.py distill-test-publish [optional destination here]\n```\n\nThis will connect to your publishing target, authenticate to it, upload a\nrandomly named file, verify it exists on the `PUBLIC_URL` and then delete it\nagain. Use this to check your publishing settings are correct.\n\n`distill-test-publish` has no arguments.\n\n\n# Optional configuration settings\n\nYou can set the following optional `settings.py` variables:\n\n**DISTILL_DIR**: string, default directory to export to:\n\n```python\nDISTILL_DIR = '/path/to/export/directory'\n```\n\n**DISTILL_PUBLISH**: dictionary, like Django's `settings.DATABASES`, supports\n`default`:\n\n```python\nDISTILL_PUBLISH = {\n    'default': {\n        ... options ...\n    },\n    'some-other-target': {\n        ... options ...\n    },\n}\n```\n\n**DISTILL_SKIP_ADMIN_DIRS**: bool, defaults to `True`\n\n```python\nDISTILL_SKIP_ADMIN_DIRS = True\n```\n\nSet `DISTILL_SKIP_ADMIN_DIRS` to `False` if you want `django-distill` to also copy over\nstatic files in the `static/admin` directory. Usually, these are not required or\ndesired for statically generated sites. The default behaviour is to skip static admin\nfiles.\n\n\n**DISTILL_SKIP_STATICFILES_DIRS**: list, defaults to `[]`\n\n```python\nDISTILL_SKIP_STATICFILES_DIRS = ['some_dir']\n```\n\nSet `DISTILL_SKIP_STATICFILES_DIRS` to a list of directory names you want `django-distill`\nto ignore directories in your defined `static/` directory. You can use this to ignore\ncopying directories containing files from apps you're not using that get bundled into your\n`static/` directory by `collect-static`. For example if you set `DISTILL_SKIP_STATICFILES_DIRS`\nto `['some_dir']` the static files directory `static/some_dir` would be skipped.\n\n\n**DISTILL_LANGUAGES**: list, defaults to `[]`\n\n```python\nDISTILL_LANGUAGES = [\n    'en',\n    'fr',\n    'de',\n]\n```\n\nSet `DISTILL_LANGUAGES` to a list of language codes to attempt to render URLs with.\nSee the \"Internationalization\" section for more details.\n\n\n# Developing locally with HTTPS\n\nIf you are using a local development environment which has HTTPS support you may need\nto add `SECURE_SSL_REDIRECT = False` to your `settings.py` to prevent a `CommandError`\nbeing raised when a request returns a 301 redirect instead of the expected HTTP/200\nresponse code.\n\n\n# Writing single files\n\nAs of `django-distill` version `3.0.0` you can use the\n`django_distill.renderer.render_single_file` method to write out a single file\nto disk using `django_distill`. This is useful for writing out single files to disk,\nfor example, you have a Django site which has some static files in a directory\nwritten by `django_distill` but the rest of the site is a normal dynamic Django site.\nYou can update a static HTML file every time a model instance is saved. You can\nuse single file writing with signals to achieve this. For example:\n\n```python\n# in models.py\nfrom django.db.models.signals import post_save\nfrom django.dispatch import receiver\nfrom django_distill.renderer import render_single_file\n\n@receiver(post_save, sender=SomeBlogPostModel)\ndef write_blog_post_static_file_post_save(sender, **kwargs):\n    render_single_file(\n        '/path/to/output/directory',\n        'blog-post-view-name',\n        blog_id=sender.pk,\n        blog_slug=sender.slug\n    )\n```\n\nThe syntax for `render_single_file` is similar to Django's `url.reverse`. The full\nusage interface is:\n\n```python\nrender_single_file(\n    '/path/to/output/directory',\n    'view-name-set-in-urls-py',\n    *view_args,\n    **view_kwargs\n)\n```\n\nFor example, if you had a blog post URL defined as:\n\n```python\n    # in urls.py\n    distill_path('post/<int:blog_id>_<slug:blog_slug>.html',\n                 PostView.as_view(),\n                 name='blog-post',\n                 distill_func=get_all_blogposts),\n```\n\nYour usage would be:\n\n```python\nrender_single_file(\n    '/path/to/output/directory',\n    'blog-post',\n    blog_id=123,\n    blog_slug='blog-title-slug',\n)\n```\n\nwhich would write out the contents of `/post/123_blog-title-slug.html` into\n`/path/to/output/directory` as the file\n`/path/to/output/directory/post/123_blog-title-slug.html`. Note any required\nsub-directories (`/path/to/output/directory/post` in this example) will be\nautomatically created if they don't already exist. All `django-distill` rules\napply, such as URLs ending in `/` will be saved as `/index.html` to make sense\nfor a physical file on disk.\n\nAlso note that `render_single_file` can only be imported and used into an\ninitialised Django project.\n\n\n# Publishing targets\n\nYou can automatically publish sites to various supported remote targets through\nbackends just like how you can use MySQL, SQLite, PostgreSQL etc. with\nDjango by changing the backend database engine. Currently the engines supported\nby `django-distill` are:\n\n**django_distill.backends.amazon_s3**: Publish to an Amazon S3 bucket. Requires\n  the Python library `boto3` (`$ pip install django-distill[amazon]`). The bucket\n  must already exist (use the AWS control panel). Options:\n\n```python\n'some-s3-container': {\n    'ENGINE': 'django_distill.backends.amazon_s3',\n    'PUBLIC_URL': 'http://.../',\n    'ACCESS_KEY_ID': '...',\n    'SECRET_ACCESS_KEY': '...',\n    'BUCKET': '...',\n    'ENDPOINT_URL': 'https://.../',  # Optional, set to use a different S3 endpoint\n    'DEFAULT_CONTENT_TYPE': 'application/octet-stream',  # Optional\n},\n```\n\n**django_distill.backends.google_storage**: Publish to a Google Cloud Storage\n  bucket. Requires the Python libraries `google-api-python-client` and\n  `google-cloud-storage`\n  (`$ pip install django-distill[google]`). The bucket\n  must already exist and be set up to host a public static website (use the\n  Google Cloud control panel). Options:\n\n```python\n'some-google-storage-bucket': {\n    'ENGINE': 'django_distill.backends.google_storage',\n    'PUBLIC_URL': 'https://storage.googleapis.com/[bucket.name.here]/',\n    'BUCKET': '[bucket.name.here]',\n    'JSON_CREDENTIALS': '/path/to/some/credentials.json',\n},\n```\n\nNote that `JSON_CREDENTIALS` is optional; if it is not specified, the google libraries\nwill try other authentication methods, in the search order described here:\nhttps://cloud.google.com/docs/authentication/application-default-credentials (e.g. the\n`GOOGLE_APPLICATION_CREDENTIALS` environment variable, attached service account, etc).\n\n\n**django_distill.backends.microsoft_azure_storage**: Publish to a Microsoft\n  Azure Blob Storage container. Requires the Python library\n  `azure-storage-blob` (`$ pip install django-distill[microsoft]`). The storage\n  account must already exist and be set up to host a public static website\n  (use the Microsoft Azure control panel). Options:\n\n```python\n'some-microsoft-storage-account': {\n    'ENGINE': 'django_distill.backends.microsoft_azure_storage',\n    'PUBLIC_URL': 'https://[storage-account-name]...windows.net/',\n    'CONNECTION_STRING': '...',\n},\n```\n\nNote that each Azure storage account supports one static website using the\nmagic container `$web` which is where `django-distill` will attempt to\npublish your site.\n\n\n# Tests\n\nThere is a minimal test suite, you can run it by cloing this repository,\ninstalling the required dependancies in `requirements.txt` then execuiting:\n\n```bash\n# ./run-tests.py\n```\n\n\n# Contributing\n\nAll properly formatted and sensible pull requests, issues and comments are\nwelcome.\n",
    "bugtrack_url": null,
    "license": "MIT",
    "summary": "Static site renderer and publisher for Django.",
    "version": "3.2.7",
    "project_urls": {
        "Homepage": "https://github.com/meeb/django-distill"
    },
    "split_keywords": [
        "django",
        " distill",
        " static",
        " website",
        " jamstack",
        " s3",
        " amazon s3",
        " aws",
        " amazon",
        " google",
        " microsoft",
        " google cloud",
        " google cloud storage",
        " azure",
        " azure storage",
        " azure blob storage"
    ],
    "urls": [
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "19c8ece20c91129314a1ec407ecc7df87102524a32615ea0c43be5b3ae54cc4b",
                "md5": "7d67739b4dbb83d1ed623cbf256ed778",
                "sha256": "b0aaf6f9bb7a36cda5ce169fd51e862718a924c5a85e382a7c208c12ff8e3ee2"
            },
            "downloads": -1,
            "filename": "django_distill-3.2.7.tar.gz",
            "has_sig": false,
            "md5_digest": "7d67739b4dbb83d1ed623cbf256ed778",
            "packagetype": "sdist",
            "python_version": "source",
            "requires_python": null,
            "size": 38170,
            "upload_time": "2024-08-30T14:07:07",
            "upload_time_iso_8601": "2024-08-30T14:07:07.259678Z",
            "url": "https://files.pythonhosted.org/packages/19/c8/ece20c91129314a1ec407ecc7df87102524a32615ea0c43be5b3ae54cc4b/django_distill-3.2.7.tar.gz",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2024-08-30 14:07:07",
    "github": true,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "github_user": "meeb",
    "github_project": "django-distill",
    "travis_ci": false,
    "coveralls": false,
    "github_actions": true,
    "requirements": [
        {
            "name": "django",
            "specs": []
        },
        {
            "name": "requests",
            "specs": []
        }
    ],
    "lcname": "django-distill"
}
        
Elapsed time: 0.41043s