Skip to main content

Generate static pages from views

Project description

django-static-models

:warning: If you have tried this app This is a new version, can run older configs, but URL handling is new and much extended

Generate static pages from a Django Views. The static pages are gathered in a single configurable directory, with appropriate generated or configured subdirectories. The name of this app is historic, the app can now evoke pages from most Views.

This app is what I need it to be. It has no deployment code, as this is a separate step, and your business. However, the app is developing some generality.

What this app is and is not

Because most websites include some static files, the word ''static' has many uses,

This app generates pages from views and URLs in a Django site. The targetted Views will not have dynamic facilities such as logons or personalisation. Naturally, webpages delivered in this way will be fast, easy to deploy and, in terms of conventional concerns, have few security issues.

It should be noted that some dynamic facilities can still be added to such pages. Email handling, shopping, and search can be created using links to external sites. Javascript can be used to inject tailored information. Or Django can be enabled where necessary---for, say, admin pages,

This app is not for supplementing Django's existing tools for handling of static resources. Nor is it specifically for optimizing output.

Why you may or may not use it

Pros,

  • You want to output static pages from a Django project
  • Simple. Add some configuration

Cons,

  • No middleware solution. So generated pages are not deployed in the running app (though you can view them directly)
  • Deploy is not considered

Quickstart

Quickstart is not a good name as there is no quick start here. You need a model and a view, typically a DetailView or like, to work with. Or use some URLs,

  • Set a config
  • Generate some pages './manage.py -o viewstaticmerge'
  • Look in the 'site/' folder in the top level of the project,

Overview

The process for main app usage is... configure some settings which target a View, run the management command, do something with the generated pages.

The main work is in the config, which describes which views are served with what data, and where the generated pages are to be written.

This being Django, the action is not as simple as it sounds. What the app does is generate an HTML response, which in Django is an encoded stream. The stream is then written to an HTML file. There are various configurations possible to decide file placement and name.

The main code is in 'static_models.utils.ViewGenerator', which is well-documented. You may want a different kind of action, for example semi-automatic page generation. If so, look at the ViewGenerator class.

Install

The code can either be downloaded from Github, or installed using Pip,

pip3 install django-static-models

The code needs to be declared in settings.py,

INSTALLED_APPS = [
     'static_models.apps.StaticModelsConfig',
]

Configuration

You need to set a base directory,

STATICVIEWS_DIR = BASE_DIR / 'page_collection'

or similar.

Generating from data Models

A simple View configuration is,

STATIC_VIEWS = [
    {
    'query': 'all',
    'view' : 'page.views.PageDetailView',
    },
]
...

This will render the model on the PageDetailView through the view PageDetailView. It will search for all Page objects, then render them to an auto-named directory. The output files will be named by the 'pk' e.g. '3'.

Often you will not want to use the 'pk' data to name the files. Configure like this,

STATIC_VIEWS = [
    {
    'query': 'all',
    'view' : 'page.views.PageDetailView',
    'filename_from_attribute' : 'slug',
    },
]

This configuration presumes the model Pages has a field called 'slug'. 'filename_from_attribute' uses the 'slug' field to name the files. So now, in the 'sites' directory, the app may generate a file 'web-lunacy', not '3'.

Note that Django configures models so that the fields are acessible as Python attributes. 'filename_from_attribute' can also call a zero-argument callable (method/function whatever). This makes it possible to generate pages from different views of the same object---add a callable returning the secondary filenames/url_id to the model.

Sometimes you will not want to use the model name to name the pathroot of the files. Configure like this,

STATIC_VIEWS = [
    {
    'query': 'all',
    'view' : 'page.views.PageDetailView',
    'filename_from_attribute' : 'slug',
    'filepath' : 'article'
    },
]

Now the files go to 'STATICMODELS_DIR/article', not 'STATICMODELS_DIR/Page'. You can substitute any path in 'filepath', the filename is appended to the value.

Generating from URLs

If you are generating from database models, I recomment the approach above. However, some views get their information from URLs only e.g. ListViews contain all their information inside them, they only need evoking. Presuming a suitable view, the key is the URL, the value is the filename, { 'urls' : {'products/': 'products'}, 'filepath' : 'products' },

If the filename (the value) is None, the generator makes a filename from the URL. May work in many instances,

{
'urls' : { 'home': 'index', 'about': None, 'contact': None},
'view' : 'hp_reviews.views.SitePageDetailView',
'filepath' : ''
},

You can state many URLs at once,

{
'urls' : { 'home': 'index', 'about': 'about', 'contact': 'contact'},
'view' : 'hp_reviews.views.SitePageDetailView',
'filepath' : ''
},

Generating one-off pages

Some views do nothing but generate single pages. You can 'filename' these,

{
'view' : 'homepage.views.HomepageView',
'filename' : 'index',
}

Management commands

The main way of invoking the configuration. Typically,

./manage.py viewstaticmerge

By default 'modelstaticmerge' will not touch files if it finds the generated files work are the same size as the existing file. This is similar to 'rsync' behaviour. Also, it will not add a file extension. But you have these options,

  • -o, --overwrite Replace currently existing files (default is to ignore if unchanged)
  • -e, --html_extension Add '.html' extension to generated files.

So,

./manage.py modelstaticmerge -e

Will generate HTML files from the configuration. From the second configuration above the files will be put in site/page/ and will be named from the slug data + '.html'.

The management command is a little stripped down. You can do the same, with a few more options, by using the shell to import the ViewGenerator class from static_models.utils.

ViewGenerator

Offers options which may be of use. All options are exposed in the settings or the management command,

Viewing files

I suggest to start you go look at the generated file. Try loading to a browser, see what happens. Straight away you will see the issue.

The generated pages will have correct internal links ie. for resources like CSS or image links. But...

Servers and browsers default files without extensions

This applies to Django development servers and to deployment servers like Apache and Nginx.

If you used this app's default setup, you will generate files without an 'htm/html' extension. The files are named,

site/page/many-wonders

Not,

site/page/many-wonders.html

A browser or server will usually read these as an 'octet-stream', not a HTML file. They will not show the page, they will offer a download.

This is an issue with static page serving, not this app or Django. There are ways round this.

Use the app facilities to generate extensions

That way, every page will load and display. The issue with this is that links between pages will be broken. And your project will loose any pretty URL setup. So, of course, you can rewrite your templates to add extensions to links. Or you may not care about dev viewing. Remember, these files are not what the dev server will be using for viewing, unless you set up a URL.

Coax the server into using a different default type

Browsers will not be able to read the files, they almost always use the extension, not the MIME type of the file. However, coaxing the server into seeing files as a different type is not as difficult as it sounds. It depends on your server. But if you can ask it to accept file types without an extension as MIME type 'text/html', you are up and running.

As for delivery, there are another option. Many deployment servers would allow you to rewrite the filename with an extension. Which is web-classic form, and guaranteed.

Viewing files in development

Django static app has a middleware solution, where any URL looking for a file seeks first in the 'static' folder, and if that fails, goes hunting round the apps.

It would be interesting to have a parallel process for the static pages, but I've not implemented this. However, I have bashed together a View that can deliver the pages, even with no extension. Put this in ''urls.py',

from django.conf import settings
from static_models.views import StaticView
    ...
path('site/<path:path>/', StaticView.as_view(path_root= settings.BASE_DIR  + '/site/'), name='site-detail'),

This code uses a setting BASE_DIR. StaticView has no opinion on the pathstyle. Now you can see all the static files on the URL 'site/...'. Such as 'site/page/many-wonders/'.

If there is a problem with the above, it is that links between pages will not work. All pages have been re-rooted to a URL root 'site'. A link like 'page/many-wonders' will now be broken. It needs to be 'site/page/many-wonders'. And we don't want to configure a URL for web root ('/') because likely you have some home page or other good use for that.

As a partial solution, StaticView also accepts 'pk' and 'slug' segment captures. So you can generate static pages, then rewrite urls.py, e.g.

from django.conf import settings
from static_models.views import StaticView
    ...

# New URL for static pages
path('page/<slug:slug>/', StaticView.as_view(path_root= settings.MEDIA_ROOT + '/site/page/'), name='page-detail'),

# original URL for dynamic pages
#path('page/<slug:slug>/',  PageDetailView.as_view(), name='page-detail'), 

Now links between pages work. So reverse URLs. But you will need to set up a URL path for every model you generate static files from.

Static definition, in the View, of path_root

This is not something that would be of interest to many.

You can define path_root using a custom class declaration, With that you can configure fancy URL resolution/HTTP responses etc. e.g.

from django.conf import settings
from static_models.views import StaticView

class PageStaticView(StaticView):
    path_root = settings.MEDIA_ROOT + '/site/page/'
    ...

And in ''urls.py',

from page.views import PageStaticView
    ...
    path('page/<int:pk>/', PageStaticView.as_view(), name='page-detail'),

Generating URL'/' root

If you are planning a complete static site, as opposed to boosting part of your site, you may run into the Django root abstraction. Django serves static files from the 'static/' directory, not the root. It has no analogy for a physical base 'root' directory. It either errors, or returns a configured URL. So what will you do with '/'?

Well, most visual websites will have a 'home' page of some kind. You could cook something up in deployment, and ignore anything that doesn't work through page generation. Or you could take advantage of this app's one-off generation, and generate an 'index.html' page from a project's 'home' page view.

Maintaining a tree of static webpages

Run the management command whenever you update. Or automate the process in a deploy script. Another way is to run a maintenance command periodically.

If you want to merge eagerly and automatically, there's a few ways, none are wonderful. I'd favour using the post_save signal in 'apps.py'. That will work with all model changes, and contains the logic within an app,

To auto-merge filesystem storage,

def static_merge(sender, instance=None, **kwargs):
    from image.views import ArticleDetailView
    from static_models.utils import file_static_merge
    file_static_merge(instance, ArticleDetailView)

class ImageConfig(AppConfig):
    ...
    def ready(self):
        ...
        from django.db.models.signals import post_save
        from image.models import Page
        post_save.connect(static_merge, sender=Page)

For more options, run full static handling,

def static_merge(sender, instance=None, **kwargs):
    from article.views import ArticleDetailView
    from static_models.utils import ModelGenerator
    g = ModelGenerator(sender, ArticleDetailView, overwrite=True)
    g.obj_create(instance)

class ImageConfig(AppConfig):
    ...
    def ready(self):
        ...
        from django.db.models.signals import post_save
        from article.models import Page
        post_save.connect(static_merge, sender=Page)

g.obj_delete() can be used in the same way on post_delete, but be careful if you want to share the static pages, that you do not delete data expected elsewhere.

Alternatives

You could use some outside tool to grab pages from the server. In the same way a web-cache like Squid works.

Closer to the codebase, Wagtail CMS have been talking about using Gatsby for the purpose of storing static pages.

Deploy Django without admin. No logons then, of course, but removes many potential security violations while allowing dynamic templates and form-handling.

django-static-sites Deployment options built in.

django-static-delivery 0.0.1 A middleware solution, which makes sense. Probably more of an optimiser than a generator, but untested.

django-static-pages Uses a test client to generate pages, Not clear if it is only for testing?

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

django-static-models-1.1.1.tar.gz (21.7 kB view hashes)

Uploaded Source

Built Distribution

django_static_models-1.1.1-py2.py3-none-any.whl (19.4 kB view hashes)

Uploaded Python 2 Python 3

Supported by

AWS AWS Cloud computing and Security Sponsor Datadog Datadog Monitoring Fastly Fastly CDN Google Google Download Analytics Microsoft Microsoft PSF Sponsor Pingdom Pingdom Monitoring Sentry Sentry Error logging StatusPage StatusPage Status page