Super-flexible Static Site Generator
Project description
Alteza
Alteza is a static site generator driven by PyPage. Examples of other static site generators can be found here.
The differentiator with Alteza is that the site author (if familiar with Python) will have a lot more fine-grained control over the output, than what (as far as I'm aware) any of the existing options offer.
The learning curve is also shorter with Alteza. I've tried to follow part of xmonad's philosophy of keeping things small and simple. Alteza doesn't try to do a lot of things; instead it simply offers the core crucial functionality that is common to most static site generators.
Alteza also imposes very little required structure or a particular "way of doing things" on your website (other than requiring unique names). You retain the freedom to organize your website as you wish. (The name Alteza comes from a word that may be translated to illustriousness in Español.)
A key design aspect of Alteza is writing little scripts and executing such code to generate your website. Your static site can contain arbitrary Python that is executed at the time of site generation. PyPage, in particular, makes it seamless to include actual Python code inside page templates. (This of course means that you must run Alteza with trusted code, or in an isolated container.)
Installation
You can install Alteza easily with pip:
pip install alteza
Try running alteza -h
to see the command-line options available.
User Guide
-
The directory structure is generally mirrored in the generated site.
-
By default, nothing is copied/published to the generated site.
- A file must explicitly indicate using a
public: true
variable/field that it is to be published.- So directories with no public files, are non-existent in the generated site.
- Files reachable from marked-as-public files will also be publicly accessible.
- Here, reachability is discovered when a provided
link
function is used to link to other files.
- Here, reachability is discovered when a provided
- A file must explicitly indicate using a
-
There are two kinds of files that are subject to processing with PyPage: Markdown files (ending with
.md
) and any file with a.py
before its actual extension.- Markdown Files:
- Markdown files are first processed to have their "front matter" extracted using Meta-Data.
- The first blank line or
---
ends the front matter section. - The front matter is processed as YAML, and the fields are injected into the
pypage
environment.
- The first blank line or
The Markdown file is processed using.pypage
, with its Python environment enhanced by the YAML fields from the front matter- The environment dictionary after the Markdown is processed by pypage is treated as the "return value" of this
.md
file.This "return value" dictionary has a.content
key added to it which maps to thepypage
output for this.md
file - This Markdown file is passed to a template specified in configuration, for a second round of processing by PyPage.
- Templates are HTML files processed by PyPage. The PyPage-processed Markdown HTML output is passed to the template as the variable
body
variable. The template itself is executed by PyPage.- The template should use this
body
value via PyPage (with{{ boydy }}
in order to render thebody
's contents.
- The template should use this
- (See more on configuration files in the next section.)
- The template is defined using a
template
variable declared in a__config__.py
file. - The
template
's value must be the entire contents of a template HTML file. A convenience functionreadfile
is provided for this. So you can writetemplate = readfile('some_template.html')
in a config file. - Templates may be overriden in descendant
__config__.py
files, or in the Markdown itself using a PyPage multiline code tag (not inline code tag).
- Templates are HTML files processed by PyPage. The PyPage-processed Markdown HTML output is passed to the template as the variable
- Markdown files result in a directory, with an
index.html
file containing the Markdown's output.
- Markdown files are first processed to have their "front matter" extracted using Meta-Data.
- Other Dynamic Files (i.e. any file with a
.py
before the last.
in its file name):- These files are processed with PyPage once with no template application step afterward.
- Other content files are not read. They are selectively either symlinked or copied.
- Markdown Files:
-
Python Environment and Configuration:
- Note: Python code in both
.md
and other.py.*
files are run using Python's built-inexec
(andeval
) functions, and when they're run, we passed in a dictionary for theirglobals
argument. We call that dict the environment, orenv
. - Configuration is done through file(s) called
__config__.py
.- First, we recursively go through all directories top-down.
- At each directory (descending downward), we execute an
__config__.py
file, if one is present. After execution, we absorb any variables in it that do not start with a_
into theenv
dict.- This behavior cna be used to override values. For example a top-level directory can define a
default_template
, which can then be overriden by inner directories.
- This behavior cna be used to override values. For example a top-level directory can define a
- The deepest
.md
/.py.*
files get executed first. After it executes, we check if aenv
contains a fieldpublic
that is set asTrue
. If it does, we mark that file for publication. Other than recording the value ofpublic
after each dynamic file is executed, any modification toenv
made by a dynamic file are discarded (and not absorbed, unlike with__config__.py
).- I would recommend not using
__config__.py
to setpublic
asTrue
, as that would make the entire directory and all its descendants public (unless that behavior is exactly what is desired). Reachability withlink
(described below) is, in my opinion, a better way to make only reachable content public.
- I would recommend not using
- Note: Python code in both
-
Name Registry and
link
.- The name of every file in the input content is stored in a "name registry" of sorts that's used by
link
.- Currently, names, without their file extension, have to be unique across input content. This might change in the future.
- The Name Registry will error out if it encounters any non-unique names. (I understand this is a significant limitation, so I might support marking this simply opt-in behavior with a
--unique
flag in the future.)
- Any non-dynamic content file that has been
link
-ed to is marked for publication (i.e. copying or symlinking). - A Python function named
link
is injected into the top levelenv
.- This function can be used to get relative links to any other file.
link
will automatically determine & return the relative path to a file.- For example, one can do
<a href="{{link('some-other-blog-post')}}">
, and the generated site will have a relative link to it (i.e. to its directory if a Markdown file, and to the file itself otherwise).
- For example, one can do
- Reachability of files is determined using this function, and unreachable files will be treated as non-public (and thus not exist in the generated site).
- This function can be used to get relative links to any other file.
- Extensions may be omitted for dynamic files (i.e.
.md
for Markdown, and.py*
for any file with.py
before its extension).- I.e. one can write both
link('magic-turtle')
orlink('magic-turtle.md')
for the filemagic-turtle.md
, andlink('pygments-styles')
orlink('pygments-styles.py.css')
for the filepygments-styles.py.css
.
- I.e. one can write both
- The name of every file in the input content is stored in a "name registry" of sorts that's used by
Usage, Testing & Development
Running
If you've installed Alteza with pip, you can just run alteza
, e.g.:
alteza -h
If you're working on Alteza itself, then run the alteza
module itself, from the project directory directly, e.g. python3 -m alteza -h
.
Command-line Arguments
The -h
argument above will print the list of available arguments:
usage: __main__.py [--copy_assets] [--trailing_slash] [--content CONTENT] [--output OUTPUT] [-h]
options:
--copy_assets (bool, default=False) Copy assets instead of symlinking to them
--trailing_slash (bool, default=False) Include a trailing slash in links to markdown pages
--content CONTENT
(str, default=test_content) Directory to read the input content from.
--output OUTPUT
(str, default=test_output) Directory to send the output. WARNING: This will be deleted first.
-h, --help show this help message and exit
As might be obvious above, you set the content
to your content directory. The output directory will be deleted entirely, before being written to.
To test against test_content
(and generate output to test_output
), run it like this:
python -m alteza --content test_content --output test_output
Code Style
I'm using black
. To re-format the code, just run: black alteza
.
Fwiw, I've configured my IDE (PyCharm) to always auto-format with black
.
Type Checking
To ensure better code quality, Alteza is type-checked with five different type checking systems: Mypy, Meta's Pyre, Microsoft's Pyright, Google's Pytype, and Pyflakes; as well as linted with Pylint.
To run some type checks:
mypy alteza # should have zero errors
pyflakes alteza # should have zero errors
pyre check # should have zero errors as well
pyright alteza # should have zero errors also
pytype alteza # should have zero errors too
Or, all at once with: mypy alteza ; pyre check ; pyright alteza ; pytype alteza ; pyflakes alteza
.
Linting
Linting policy is very strict. Pylint must issue a perfect 10/10 score, otherwise the Pylint CI check will fail.
To test whether lints are passing, simply run:
pylint -j 0 alteza
Of course, when it makes sense, lints are suppressed next to the relevant line, in code. Also, unlike typical Python code, the naming convention generally-followed in this codebase is camelCase
. Pylint checks have been mostly disabled for names.
Dependencies
To install dependencies for development, run:
python3 -m pip install -r requirements.txt
python3 -m pip install -r requirements-dev.txt
To use a virtual environment (after creating one with python3 -m venv venv
):
source venv/bin/activate
# ... install requirements ...
# ... do some development ...
deactive # end the venv
License
This project is licensed under the AGPL v3, but I'm reserving the right to re-license it under a license with fewer restrictions, e.g. the Apache License 2.0, and any PRs constitute consent to re-license as such.
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.