Skip to main content

Zero-config static site builder for markdown blogs

Project description

mdwiki

mdwiki is a small static site builder that turns a directory tree of markdown files into a blog-ready HTML site.

It keeps the authoring model simple:

  • directories become URL segments
  • each markdown file becomes one HTML page
  • local images referenced from markdown are copied into the output site
  • index pages and tag pages are generated automatically

Install

Use uv:

uv tool install mdwiki

Or install from source:

uv sync
uv run mdwiki_exec ./example-posts ./dist

CLI

Build a site from a markdown source tree:

mdwiki_exec <source_dir> <dist_dir>

Show the built-in help or version:

mdwiki_exec --help
mdwiki_exec --version

Run the safe incremental mode with a newline-separated changed file list:

mdwiki_exec --changed-files changed.txt ./post ./dist

Example:

mdwiki_exec ./post ./dist

source_dir usually contains:

  • markdown posts under dated folders such as post/2024/04/03/demo.md
  • optional local images next to each markdown file
  • optional CNAME
  • optional config.json

Safe Incremental Build

mdwiki now supports a safe incremental mode aimed at static blog workflows:

  • if only markdown files under post/ changed, it rebuilds the affected detail pages
  • if images or other local assets under post/ changed, it rebuilds posts in the same folder
  • it always refreshes index pages and tag pages so listing pages stay correct
  • if config.json, CNAME, or files outside the source tree changed, it falls back to a full rebuild

The builder writes a manifest file named .mdwiki-build.json into the output directory. If the previous output directory is restored in CI, the next run can reuse that manifest and avoid rewriting every article page.

Optional GoatCounter support

You can enable real pageview counters by adding goatcounter_script to your config.json:

{
  "goatcounter_script": "<script data-goatcounter=\"https://drunkpig.goatcounter.com/count\" async src=\"//gc.zgo.at/count.js\"></script>"
}

When this field is present:

  • article pages render a counter next to the slug
  • index pages fetch and display per-article pageviews

When it is absent, mdwiki renders 未知 instead of fake random numbers.

Optional source markdown link

You can also expose the original markdown source for crawlers and readers:

{
  "source_markdown_base_url": "https://raw.githubusercontent.com/drunkpig/drunkpig.github.io/master/post"
}

When this field is present, article pages will:

  • add a standard <link rel="alternate" type="text/markdown"> tag in <head>
  • keep the raw markdown URL available to crawlers without showing a visible banner in the page body

Build date in footer

The default footer template renders build_version as a YYYYMMDD date during site generation. Because GitHub Pages is built from source on every deploy, this date updates for all generated pages on each successful build.

GitHub Pages workflow

The recommended deployment model is:

  1. Keep markdown source in your repository.
  2. Restore the previous dist/ directory from cache.
  3. Build the site in GitHub Actions.
  4. Upload the generated dist/ directory as a Pages artifact.
  5. Deploy with actions/deploy-pages.

Example workflow:

name: Build and Deploy Pages

on:
  push:
    branches: ["master"]
  workflow_dispatch:

permissions:
  contents: read
  pages: write
  id-token: write

jobs:
  build:
    runs-on: ubuntu-latest
    steps:
      - uses: actions/checkout@v6
        with:
          fetch-depth: 0

      - uses: actions/setup-python@v5
        with:
          python-version: "3.11"

      - uses: actions/cache/restore@v4
        with:
          path: dist
          key: pages-dist-${{ github.ref_name }}-${{ github.run_id }}
          restore-keys: |
            pages-dist-${{ github.ref_name }}-

      - name: Detect changed files
        run: |
          if [ "${{ github.event.before }}" = "0000000000000000000000000000000000000000" ]; then
            : > changed.txt
          else
            git diff --name-only "${{ github.event.before }}" "${{ github.sha }}" > changed.txt
          fi

      - name: Install mdwiki
        run: pip install mdwiki

      - name: Build site
        run: |
          if [ -s changed.txt ]; then
            mdwiki_exec --changed-files changed.txt ./post ./dist
          else
            mdwiki_exec ./post ./dist
          fi

      - uses: actions/cache/save@v4
        if: github.event_name == 'push'
        with:
          path: dist
          key: pages-dist-${{ github.ref_name }}-${{ github.run_id }}

Develop with uv

uv sync
uv run mdwiki_exec ./post ./dist
uv build

Publish to PyPI:

git tag -a v0.3.7 -m "Release v0.3.7"
git push origin master v0.3.7

Pushing a new v* tag triggers the bundled GitHub Actions release workflow, which builds the package with uv and publishes it to PyPI.

Notes

  • CNAME is optional.
  • The bundled default theme is stored under mdwiki/templates/default.
  • mdwiki_exec is the supported public CLI entrypoint.

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

mdwiki-0.3.7.tar.gz (40.0 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

mdwiki-0.3.7-py3-none-any.whl (40.2 kB view details)

Uploaded Python 3

File details

Details for the file mdwiki-0.3.7.tar.gz.

File metadata

  • Download URL: mdwiki-0.3.7.tar.gz
  • Upload date:
  • Size: 40.0 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: uv/0.9.18 {"installer":{"name":"uv","version":"0.9.18","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":null,"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":null}

File hashes

Hashes for mdwiki-0.3.7.tar.gz
Algorithm Hash digest
SHA256 216607d4cc12bf957fe70c9aa10beb2ffb94a07dd3c440bd7d58dd8aac6d9b25
MD5 0ae47158fcc74b3b90a0a14b3bde93e5
BLAKE2b-256 d937d1f463398b3e8e682a3e5337b1810621b97889e1d93a3a8002ad30fd831f

See more details on using hashes here.

File details

Details for the file mdwiki-0.3.7-py3-none-any.whl.

File metadata

  • Download URL: mdwiki-0.3.7-py3-none-any.whl
  • Upload date:
  • Size: 40.2 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: uv/0.9.18 {"installer":{"name":"uv","version":"0.9.18","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":null,"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":null}

File hashes

Hashes for mdwiki-0.3.7-py3-none-any.whl
Algorithm Hash digest
SHA256 5b5d2688a1884e076458d6508b7ab10178a29140b0a51733f4b877c7b9ba3958
MD5 e823380608969ce54de589f95667285e
BLAKE2b-256 92414aaba39982102f266555479caebf5739ee9edbdd1b0b8bcef1ba1e0804c3

See more details on using hashes here.

Supported by

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