fastapi:~/app$ cat fastapi-htmx.md

FastAPI + HTMX: Der Full-Stack ohne Build-Prozess

# Erstellen Sie produktionsreife Web-Apps ohne React oder webpack: FastAPI, HTMX, Alpine.js, Jinja2, reines CSS, Bootstrap-Muster, i18n, Deployment, SEO und Performance.

words: 8343 read_time: 42m updated: 2026-06-22 16:59
$ less fastapi-htmx.md

TL;DR: FastAPI + HTMX + Alpine.js + Jinja2 + reines CSS erzeugt produktionsreife Webanwendungen ohne Build-Tools, ohne node_modules/ und mit perfekten Lighthouse-Werten. Dieser Leitfaden behandelt das gesamte System von der Architektur bis zum Deployment und nutzt blakecrosley.com als Produktionsreferenz: 210 Blogbeiträge, interaktive JavaScript-Komponenten, 11 Kernleitfäden, 48 Designstudien sowie Englisch plus 9 übersetzte Locales, ohne einen einzigen Bundler, Compiler oder Transpiler.1

Der moderne Webentwicklungs-Stack setzt voraus, dass Sie React, webpack, TypeScript und eine Build-Pipeline benötigen. Für eine große Kategorie von Anwendungen — contentgetriebene Websites, interne Tools, CRUD-Anwendungen, Portfolio-Websites, Dokumentationsplattformen — ist diese Annahme falsch. Der in diesem Leitfaden beschriebene Stack eliminiert die gesamte Frontend-Build-Toolchain und erzeugt zugleich Websites, die bei Lighthouse 100/100/100/100 erreichen.2

Das ist kein Plädoyer. Es ist eine Messung. Die hier beschriebene Architektur läuft in Produktion, bedient echte Benutzer in zehn Sprachen, und die Zahlen sind überprüfbar.


Wichtigste Erkenntnisse

  • Server-gerendertes HTML eliminiert drei ganze Problemkategorien: Client-State-Management, JSON-Serialisierungsgrenzen und Hydration-Mismatches. HTMX macht Server-Antworten zur finalen Ausgabe — kein clientseitiger Rendering-Schritt.
  • Keine Build-Tools bedeuten keine Build-Fehler. Keine npm install-Konflikte bei Peer-Dependencies, keine TypeScript-Compiler-Fehler in Dateien, die Sie nicht angefasst haben, keine Dependabot-PRs für transitive Abhängigkeiten, die Sie nie importiert haben. Die Deploy-Pipeline ist git push.
  • Alpine.js übernimmt clientseitigen State, den HTMX nicht abbilden kann. Dropdowns, Modals, mobile Navigations-Toggles und sämtlicher UI-State, der ausschließlich im Browser existiert, gehören zu Alpine.js. Die Grenze ist klar: Wenn der State den Server benötigt, verwenden Sie HTMX. Falls nicht, verwenden Sie Alpine.js.
  • Plain CSS mit Custom Properties ersetzt Sass und Tailwind. CSS Custom Properties kaskadieren, vererben sich und reagieren zur Laufzeit auf Media Queries. Präprozessor-Variablen kompilieren zu statischen Werten und verschwinden. Der Browser liest Custom Properties direkt — kein Kompilierungsschritt.
  • Dieser Ansatz hat klare Grenzen. Er ist falsch für große Teams, die Komponenten-Schnittstellen teilen, für SaaS-Produkte mit komplexem clientseitigen State und für Anwendungen, die auf npm-Ökosystem-Bibliotheken angewiesen sind. Der Entscheidungsrahmen in Abschnitt 15 identifiziert die Grenze präzise.
  • blakecrosley.com ist der Beweis. Die Kernmuster in diesem Leitfaden (HTMX, Alpine.js, Jinja2, plain CSS) laufen produktiv auf blakecrosley.com. Die Bootstrap- und SQLAlchemy-Abschnitte behandeln Standardmuster für den Stack, die auf dieser konkreten Seite nicht verwendet werden. Jede Aussage hat einen Dateipfad, einen Konfigurationsblock oder ein Lighthouse-Audit, das Sie selbst unter PageSpeed Insights verifizieren können.2

So nutzen Sie diesen Leitfaden

Dies ist eine umfassende Referenz. Beginnen Sie dort, wo Ihr Erfahrungsniveau passt:

Erfahrung Hier starten Anschließend erkunden
Python-Entwickler, neu bei HTMX The No-Build ThesisArchitecture OverviewHTMX Deep Dive Alpine.js Patterns, Security
React/Vue-Entwickler, der Alternativen evaluiert The No-Build ThesisDecision Framework Architecture Overview, Performance
FastAPI-Entwickler, der Interaktivität hinzufügt HTMX Deep DiveAlpine.js Patterns i18n and Localization, Deployment
Full-Stack-Entwickler, der von Grund auf baut Sequenziell ab Architecture Overview lesen Quick Reference Card zur laufenden Verwendung

Verwenden Sie Strg+F / Cmd+F, um nach bestimmten Mustern oder Attributen zu suchen. Die Quick Reference Card am Ende bietet eine überfliegbare Zusammenfassung.


Die No-Build-These

Die These ist eng und konkret: Für content-getriebene Seiten mit einem einzelnen Entwickler oder kleinen Team lösen Build-Tools Probleme, die Sie nicht haben, und schaffen dabei welche, die Sie schon haben.

Hier sind die echten Metriken von blakecrosley.com:

Metrik blakecrosley.com (No-Build) Typisches Next.js-Projekt3
Abhängigkeiten 17 Python-Pakete 311+ npm-Pakete
Build-Konfigurationsdateien 0 5-8 (next.config, tsconfig, postcss, tailwind, etc.)
Größe von node_modules/ Existiert nicht 187 MB Baseline, 250-400 MB mit Erweiterungen
Installationszeit pip install: 8 Sekunden npm install: 30-90 Sekunden
Build-Schritt Keiner next build: 15-60 Sekunden
Deploy-Pipeline git push → live in ~40 Sekunden Installieren → bauen → deployen: 2-5 Minuten
Lighthouse Performance 100 70-90 ohne explizite Optimierung4

Die 17 Python-Pakete umfassen FastAPI, Jinja2, Pydantic, uvicorn, nh3 und 12 weitere. Keines ist ein Build-Tool. Keines ist ein Compiler. Keines ist ein Bundler.5

Worauf Sie verzichten

Ehrlichkeit verlangt, die echten Kosten zu benennen:

Kein TypeScript. Jede .js-Datei ist Vanilla-JavaScript. Typfehler werden durch Testing und Code-Analyse erkannt, nicht durch einen Compiler. Das funktioniert für einen Solo-Entwickler. Für ein Team von 10 Personen, die Komponenten-Schnittstellen teilen, würde es nicht funktionieren.

Kein Hot Module Replacement. CSS-Änderungen erfordern einen manuellen Browser-Refresh. HTMXs hx-boost macht die Navigation schnell genug, dass volle Refreshes tolerierbar sind, aber bei engen visuellen Iterationszyklen spart HMR Zeit.

Kein Tree Shaking. Jedes Byte JavaScript, das Sie schreiben, wird an den Browser ausgeliefert. Diese Einschränkung erzwingt Disziplin: kleine, fokussierte Dateien statt großer Utility-Module.

Keine npm-Komponentenbibliotheken. Kein Radix, kein shadcn/ui, kein Headless UI. Jedes interaktive Element ist handgebaut oder verwendet die eingebauten Komponenten von Bootstrap 5.

Keine Design-System-Tokens aus npm. Das Design-System lebt in CSS Custom Properties. Es kann nicht als Paket in ein anderes Projekt importiert werden.

Diese Kompromisse sind für eine content-getriebene Seite mit ein bis drei Entwicklern akzeptabel. Für ein SaaS-Produkt mit einem 15-köpfigen Engineering-Team wären sie inakzeptabel. Abschnitt 15 liefert den Entscheidungsrahmen.

Was Sie gewinnen

Keine Build-Fehler. Kein npm install kann an Peer-Dependency-Konflikten scheitern. Kein next build kann an einem TypeScript-Fehler in einer Datei scheitern, die Sie nicht angefasst haben.6

Debugging mit View Source. Das JavaScript, das im Browser läuft, ist das JavaScript, das Sie geschrieben haben. Keine Source Maps erforderlich.

Sofortiger lokaler Start. uvicorn app.main:app --reload startet in unter 2 Sekunden.

Konkreter Request-Wasserfall. Ein erster Besuch lädt: ein HTML-Dokument (~15KB gzipped), eine CSS-Datei (~8KB), HTMX (~16KB, gecacht), Alpine.js (~15KB, gecacht) und das interaktive JS der Seite (~4-8KB). Insgesamt: ungefähr 55-65KB beim ersten Besuch.1

Zukunftssicheres Frontend. Der clientseitige Code verwendet HTML, CSS und JavaScript — Standards, die seit 30 Jahren Abwärtskompatibilität bewahren.7 Keine Webpack-4-→-5-Migration, keine Create-React-App-Abkündigung, keine Next.js-App-Router-Migration.

Stack-Vergleich

Wie sich der No-Build-Stack auf messbaren Dimensionen mit gängigen Alternativen vergleicht:

Dimension FastAPI+HTMX (dieser Leitfaden) Next.js (React) Astro 11ty
An den Browser ausgeliefertes JS 35-40KB (HTMX+Alpine+kleine Seitenskripte) 85-250KB+ (React-Runtime) 0KB Standard, optionale Islands 0KB Standard
Build-Schritt Keiner Erforderlich (webpack/turbopack) Erforderlich (Vite) Erforderlich (custom)
Konfigurationsdateien 0 5-8 (next.config, tsconfig, etc.) 1-3 (astro.config, tsconfig) 1-2 (.eleventy.js)
Deploy-Pipeline git push (40s) Install+Build+Deploy (2-5min) Install+Build+Deploy (1-3min) Install+Build+Deploy (1-2min)
Serverseitige Interaktivität Nativ (HTMX) API-Routes + Client-Fetch Begrenzt (Form Actions) Keine (statische Ausgabe)
Client-State-Management Alpine.js (15KB) React state/context/Redux Framework Islands Manuelles JS
Backend-Sprache Python JavaScript/TypeScript JavaScript/TypeScript JavaScript
i18n-Ansatz Serverseitig (Middleware) next-intl oder ähnliches Paket @astrojs/i18n Manuell
Lighthouse Performance 100 (gemessen) 70-90 typisch4 95-100 typisch 95-100 typisch
Am besten geeignet für Content-Seiten, CRUD, Dashboards Komplexe SPA, große Teams Content-Seiten, Marketing Statische Blogs, Dokumentation

Astro und 11ty sind die nächsten Konkurrenten für Content-Seiten. Beide produzieren exzellente statische Ausgaben, erfordern aber einen Build-Schritt und eine JavaScript-Toolchain. Der FastAPI+HTMX-Stack tauscht Static-Site-Performance gegen serverseitige Interaktivität (Kategoriefilterung, Formularverarbeitung, Echtzeit-Suche), ohne einen Build-Schritt hinzuzufügen. Wenn Ihre Seite rein statisch ist und keine Server-Interaktionen hat, sind Astro oder 11ty möglicherweise die bessere Wahl.


Architekturübersicht

Ablauf einer Anfrage

Jede Anfrage durchläuft einen einzigen Pfad durch vier Schichten:

Browser                FastAPI                Jinja2              HTMX/Alpine
  |                      |                     |                     |
  |--- GET /about ------>|                     |                     |
  |                      |-- render template ->|                     |
  |                      |                     |-- base.html ------->|
  |                      |                     |   + about.html      |
  |                      |<-- full HTML -------|                     |
  |<--- HTML response ---|                     |                     |
  |                                                                  |
  |--- hx-get /search ------------------------------------------------>|
  |                      |<-- HTMX request ----|                     |
  |                      |-- render partial -->|                     |
  |                      |                     |-- _results.html     |
  |                      |<-- HTML fragment ---|                     |
  |<--- HTML fragment ---|                     |                     |
  |--- DOM swap -------------------------------------------------------->|

Vollständige Seitenaufrufe liefern komplette HTML-Dokumente zurück (Basis-Template + Seiten-Template). HTMX-Anfragen liefern HTML-Fragmente (Partials). Der Server entscheidet anhand des Anfragetyps, was gerendert wird. Alpine.js verwaltet clientseitigen Zustand, der den Server nie erreicht.

Komponentenrollen

Komponente Rolle Geltungsbereich
FastAPI Routing, Geschäftslogik, Datenzugriff, Validierung Server
Jinja2 Template-Rendering, Vererbung, Makros Server
HTMX Servergesteuerte Interaktivität (Formulare, Paginierung, Suche) Client ↔ Server
Alpine.js Rein clientseitiger Zustand (Dropdowns, Modals, Toggles) Nur Client
Bootstrap 5 Grid-System, Utility-Klassen, responsives Layout Client (CSS)
Plain CSS Benutzerdefinierte Properties, Komponentenstile, Design-Tokens Client (CSS)
Pydantic Anfrage-/Antwortvalidierung, Einstellungen Server

Projektstruktur

app/
├── main.py              # FastAPI app, middleware, templates
├── config.py            # Pydantic settings management
├── routes/
│   ├── pages.py         # Page routes (HTML responses)
│   └── api.py           # API routes (JSON/HTML fragment responses)
├── content.py           # Markdown loading, blog post parsing
├── security/
│   ├── headers.py       # CSP, HSTS, security headers middleware
│   ├── csrf.py          # HMAC-signed CSRF tokens
│   ├── rate_limit.py    # 3-tier rate limiting
│   └── logging.py       # Security event logging
├── i18n/
│   ├── config.py        # Supported locales, mappings
│   ├── middleware.py     # URL-based locale detection
│   ├── jinja.py         # Translation functions for templates
│   └── d1_client.py     # Cloudflare D1 translation storage
├── cache_assets.py      # Content-hash asset versioning
└── templates/
    ├── base.html         # Base layout with Alpine.js state
    ├── components/       # Reusable partials (_language_switcher.html, etc.)
    └── pages/            # Page templates (home.html, about.html, etc.)

content/
├── blog/                # Markdown blog posts with YAML frontmatter
└── guides/              # Multi-section guide markdown

static/
├── css/                 # Plain CSS (no preprocessors)
├── js/                  # Vanilla JavaScript (no bundlers)
│   └── vendor/          # Self-hosted HTMX, Alpine.js
└── images/              # Optimized images with WebP srcset

Die Struktur folgt einem einzigen Prinzip: Jedes Verzeichnis enthält genau eine Art von Dateien. Routen befinden sich in routes/. Templates in templates/. Statische Assets in static/. Kein Build-Schritt transformiert das eine in das andere.

Vergleich mit der SPA-Architektur

In einem React + Next.js-Projekt sähe die entsprechende Struktur folgendermaßen aus:

src/
├── components/       # React components (JSX)
├── pages/            # Route handlers (also JSX)
├── api/              # API routes (also in pages/)
├── hooks/            # Custom React hooks
├── context/          # React context providers
├── lib/              # Utility functions
├── styles/           # CSS modules or Tailwind config
└── types/            # TypeScript type definitions

# Plus build configuration
next.config.js
tsconfig.json
postcss.config.js
tailwind.config.js
eslint.config.js
package.json
package-lock.json
node_modules/         # 187+ MB of dependencies

Die SPA-Architektur erfordert eine Build-Zeit-Koordination zwischen diesen Verzeichnissen. TypeScript kompiliert .tsx zu JavaScript. PostCSS verarbeitet Tailwind-Direktiven zu CSS. Webpack (oder Turbopack) bündelt die Ausgabe in Chunks. Jeder Schritt kann unabhängig fehlschlagen.

Die No-Build-Architektur erfordert keinerlei Koordination. Das Template referenziert eine CSS-Datei. Die CSS-Datei existiert in static/css/. Der Browser lädt sie direkt. Benennen Sie eine Datei um, bricht die Template-Referenz zur Laufzeit — nicht zur Build-Zeit. Damit verlagern sich Fehler von der Kompilierzeit zur Laufzeit, was ein echtes Abwägungsproblem darstellt. Für einen einzelnen Entwickler, der während der Entwicklung uvicorn --reload ausführt, erscheinen Laufzeitfehler sofort im Browser. Für ein großes Team hingegen verhindert die Kompilierzeitprüfung durch TypeScript eine ganze Kategorie von Fehlern, die Laufzeitfehler nicht abfangen können.


FastAPI-Patterns

Anwendungseinrichtung

Die Anwendung wird in main.py mit expliziter Middleware-Reihenfolge initialisiert:

from fastapi import FastAPI
from fastapi.staticfiles import StaticFiles
from fastapi.templating import Jinja2Templates
from starlette.middleware.gzip import GZipMiddleware

app = FastAPI(
    title="Blake Crosley",
    docs_url=None,     # Disable docs in production
    redoc_url=None,
    openapi_url=None,  # Prevent /openapi.json exposure
)

# Middleware order matters: last added = first executed
app.add_middleware(SecurityHeadersMiddleware)
app.add_middleware(GZipMiddleware, minimum_size=500)
app.add_middleware(LocaleMiddleware)
app.add_middleware(RateLimitMiddleware)
app.add_middleware(SecurityLogMiddleware, site_name="blakecrosley.com")

# Static files
app.mount("/static", StaticFiles(directory=STATIC_DIR), name="static")

# Templates
templates = Jinja2Templates(directory=TEMPLATES_DIR)

Drei Designentscheidungen sind hier wichtig. Erstens deaktivieren docs_url=None und openapi_url=None die automatischen API-Dokumentationsendpunkte. Eine öffentlich zugängliche Content-Website muss /docs oder /openapi.json nicht im Internet verfügbar machen.8 Zweitens ist die Middleware-Reihenfolge entscheidend: Security Logging wird zuerst ausgeführt (zuletzt hinzugefügt) und erfasst deshalb jede Anfrage, einschließlich solcher, die durch Rate Limiting abgelehnt werden. Drittens komprimiert GZipMiddleware alle Antworten über 500 Byte, wodurch sich die HTML-Übertragungsgröße typischerweise um 70-80 % reduziert.

Routing

Routes teilen sich in zwei Kategorien auf: Page Routes geben vollständige HTML-Dokumente zurück, und API-Routes geben JSON- oder HTML-Fragmente zurück.

# routes/pages.py — full HTML responses
from fastapi import APIRouter, Request

router = APIRouter()

@router.get("/about")
async def about(request: Request):
    templates = request.app.state.templates
    return templates.TemplateResponse("pages/about.html", {
        "request": request,
        "page_title": "About — Blake Crosley",
        "page_description": "Designer, developer, dad.",
    })
# routes/api.py — JSON or HTML fragment responses
@router.get("/api/quiz/{quiz_id}/step")
async def quiz_step(request: Request, quiz_id: str, answers: str = ""):
    # Parse answers, compute next question or result
    question = get_next_question(quiz_id, answers)
    templates = request.app.state.templates
    return templates.TemplateResponse("components/_quiz_step.html", {
        "request": request,
        "question": question,
        "answers": answers,
        "step": len(answers.split(",")) if answers else 0,
    })

Die Unterscheidung ist für HTMX wichtig. Vollständige Page Routes geben Dokumente zurück, die base.html erweitern. API-Routes geben HTML-Fragmente zurück, die HTMX in bestehende DOM-Elemente austauscht. Dieselbe Jinja2-Template-Engine rendert beides: keine separate API-Schicht.

Dependency Injection

Das Depends()-System von FastAPI sorgt für eine saubere Trennung zwischen Route Handlers und gemeinsam genutzter Logik:

from fastapi import Depends, Request

def get_templates(request: Request):
    """Get templates from app state."""
    return request.app.state.templates

def get_current_locale(request: Request) -> str:
    """Get locale from middleware-set request state."""
    return getattr(request.state, "locale", "en")

@router.get("/blog/{slug}")
async def blog_post(
    request: Request,
    slug: str,
    templates=Depends(get_templates),
    locale: str = Depends(get_current_locale),
):
    post = load_post_by_slug(slug)
    if not post:
        raise HTTPException(404, "Post not found")
    return templates.TemplateResponse("pages/blog/post.html", {
        "request": request,
        "post": post,
        "locale": locale,
    })

Dependencies lassen sich zusammensetzen. Eine get_db-Dependency kann von get_current_locale abhängen, die wiederum von der Anfrage abhängt. FastAPI löst die Kette automatisch auf.

Pydantic Settings

Die Konfiguration nutzt Pydantics BaseSettings mit Vorrang für Umgebungsvariablen:

from pydantic_settings import BaseSettings

class Settings(BaseSettings):
    D1_WORKER_URL: str = ""
    D1_AUTH_SECRET: str = ""
    CLOUDFLARE_ACCOUNT_ID: str = ""
    CLOUDFLARE_API_TOKEN: str = ""
    ANALYTICS_PASSKEY: str = ""

    class Config:
        env_file = ".env"
        env_file_encoding = "utf-8"

settings = Settings()

Umgebungsvariablen überschreiben Werte aus der .env-Datei. In Produktion (Railway) werden Secrets als Umgebungsvariablen gesetzt. Lokal liefert eine .env-Datei Standardwerte. Die Klasse Settings validiert Typen beim Start: Ein fehlendes Pflichtfeld schlägt früh fehl, statt erst zur Laufzeit.

Async-Patterns

FastAPI-Routes sind standardmäßig async. Bei I/O-gebundenen Operationen (Datenbankabfragen, HTTP-Anfragen, Dateilesevorgänge) verhindert async, dass der Event Loop blockiert wird:

@asynccontextmanager
async def lifespan(app: FastAPI):
    # Load translations into memory cache at startup
    async with httpx.AsyncClient() as client:
        for locale in SUPPORTED_LOCALES:
            resp = await client.post(f"{D1_URL}/query", ...)
            TRANSLATIONS[locale] = resp.json()["results"]
    yield
    # Cleanup on shutdown (if needed)

app = FastAPI(lifespan=lifespan)

Lifespan ist jetzt der einzige Startup-/Shutdown-Pfad. Starlette erreichte im März 2026 seine erste stabile Version 1.0 (1.3.1 Stand 12. Juni) und entfernte die lange als deprecated markierten Hooks on_event, on_startup und on_shutdown: lifespan (oben) ist der einzige Mechanismus, und @app.route() / @app.websocket_route() wurden durch Route / WebSocketRoute in der routes-Liste ersetzt. FastAPI 0.137.0 (14. Juni 2026) pinnt Starlette auf die 1.x-Reihe und überarbeitet die eigenen Router-Interna: router.routes ist keine flache Liste von APIRoute-Objekten mehr, sondern ein Baum aus Zwischenknoten. Behandeln Sie es deshalb als internes Detail, nicht als etwas, über das Sie iterieren sollten. Der Vorteil: Routes, die einem Router nach include_router() hinzugefügt werden, werden nun live berücksichtigt, und ein Sub-Router kann eingebunden werden, bevor seine Routes definiert sind.24 Nichts davon ändert die Patterns in diesem Guide: Er verwendet durchgehend lifespan und Standard-Route-Deklarationen. Wenn Sie allerdings Tooling pflegen, das router.routes durchläuft, oder noch alte @app.on_event-Handler ausführen, sind 0.137.0 / Starlette 1.0 Breaking Changes. FastAPI 0.137.2 (18. Juni 2026) ergänzt anschließend iter_route_contexts(), den unterstützten Weg, Routes aufzulisten, nachdem router.routes nun intern ist. FastAPI 0.138.0 (20. Juni 2026) fügt dann app.frontend("/", directory="dist") / router.frontend(...) hinzu, um ein gebautes statisches Frontend bereitzustellen: nützlich, wenn Sie einen separaten SPA-Build ausliefern, aber unabhängig vom No-Build-Ansatz dieses Guides mit serverseitigem Rendering (es mountet ein dist/-Verzeichnis, statt HTML auf dem Server zu rendern).25

CPU-gebundene Operationen (Markdown-Rendering, CSS-Extraktion) können synchrone Funktionen verwenden. FastAPI führt sie automatisch in einem Thread Pool aus, wenn der Route Handler nicht als async deklariert ist:

# Sync function — FastAPI runs it in a thread pool
@router.get("/blog/{slug}")
def blog_post(slug: str):
    post = load_post_by_slug(slug)  # CPU-bound Markdown parsing
    return templates.TemplateResponse(...)

Die Regel: Wenn die Funktion auf I/O wartet, machen Sie sie async. Wenn sie CPU-Arbeit erledigt, lassen Sie sie synchron. Mischen Sie await nicht mit blockierenden Aufrufen in derselben Funktion.9


Jinja2-Templates

Template-Vererbung

Das Vererbungssystem von Jinja2 ersetzt Reacts Komponentenkomposition durch ein einfacheres Modell. Ein einziges Basis-Template definiert das Seitengerüst. Kind-Templates füllen benannte Blöcke:

<!-- base.html — the skeleton -->
<!DOCTYPE html>
<html lang="{{ lang_attr() }}">
<head>
  <meta charset="UTF-8">
  <meta name="viewport" content="width=device-width, initial-scale=1.0">
  <title>{{ page_title | default("Blake Crosley") }}</title>
  <meta name="description" content="{{ page_description | default('...') }}">

  <!-- CSS — single file, no preprocessor -->
  <link rel="stylesheet" href="{{ asset('css/styles.css') }}">

  <!-- JSON-LD structured data -->
  <script type="application/ld+json">
  { "@context": "https://schema.org", "@graph": [...] }
  </script>

  {% block head %}{% endblock %}
</head>
<body>
  <header class="header">...</header>

  <main id="main" role="main">
    {% block content %}{% endblock %}
  </main>

  <footer class="footer">...</footer>

  <!-- Scripts deferred for performance -->
  <script defer src="{{ asset('js/vendor/alpine.min.js') }}"></script>
  <script defer src="{{ asset('js/vendor/htmx.min.js') }}"></script>
  <script defer src="{{ asset('js/main.js') }}"></script>

  {% block scripts %}{% endblock %}
</body>
</html>
<!-- pages/about.html — fills the blocks -->
{% extends "base.html" %}

{% block head %}
<script type="application/ld+json">
{ "@type": "AboutPage", "name": "About Blake Crosley", ... }
</script>
{% endblock %}

{% block content %}
<section class="hero">
  <h1>About</h1>
  <p>Designer, developer, dad.</p>
</section>
{% endblock %}

Die {% extends %}-Direktive stellt eine Eltern-Kind-Beziehung her. Das Kind-Template definiert nur die Blöcke, die es überschreiben muss. Alles andere — der <head>, der Header, der Footer, die Script-Tags — stammt aus dem Basis-Template. Komposition erfolgt hier durch Subtraktion statt durch Konstruktion.

Das asset()-Global

Statische Assets verwenden Content-Hash-Versionierung für Cache-Busting:

# cache_assets.py
def build_asset_map(static_dir: Path) -> dict[str, str]:
    """Compute MD5 hashes of all static files at startup."""
    asset_map = {}
    for filepath in static_dir.rglob("*"):
        if filepath.is_file():
            rel_path = str(filepath.relative_to(static_dir))
            content_hash = hashlib.md5(filepath.read_bytes()).hexdigest()[:10]
            asset_map[rel_path] = content_hash
    return asset_map

def make_asset_url(asset_map: dict, path: str) -> str:
    """Generate versioned URL: /static/css/styles.css?v=a3f8b2c1d0"""
    clean_path = path.lstrip("/")
    version = asset_map.get(clean_path, "0")
    return f"/static/{clean_path}?v={version}"

Im Template rendert {{ asset('css/styles.css') }} zu /static/css/styles.css?v=a3f8b2c1d0. Der Hash ändert sich bei Dateiänderungen und invalidiert damit den CDN-Cache. Webpacks [contenthash]-Dateinamenstrategie wird so durch 30 Zeilen Python ersetzt, die beim Start berechnet werden.

Include für wiederverwendbare Partials

Komponenten, die sich über mehrere Seiten hinweg wiederholen, nutzen {% include %}:

<!-- base.html -->
{% include "components/_language_switcher.html" %}
<!-- components/_language_switcher.html -->
{%- set current = current_locale() -%}
{%- set locales = all_locales() -%}

<div class="language-switcher"
     x-data="{ open: false }"
     @click.away="open = false">
  <button @click="open = !open" :aria-expanded="open">
    {{ current_locale_native() }}
  </button>
  <ul class="language-switcher-menu"
      :class="{ 'is-open': open }"
      x-cloak>
    {% for locale in locales %}
    <li>
      <a href="{{ locale_url(request.url.path, locale.code) }}"
         hreflang="{{ locale.code }}">
        {{ locale.native }}
      </a>
    </li>
    {% endfor %}
  </ul>
</div>

Der Unterstrich-Präfix (_language_switcher.html) ist eine Konvention für Partials — Template-Fragmente, die nicht eigenständig gerendert werden sollen. Diese Komponente nutzt sowohl Alpine.js (für den Dropdown-Toggle) als auch Jinja2 (für die Sprachliste). Die Grenze ist klar gezogen: Alpine.js verwaltet den Öffnen/Schließen-Zustand, Jinja2 die Daten.

Macros für wiederverwendbare Komponenten

Macros sind die Funktionen von Jinja2 — wiederverwendbare Template-Blöcke mit Parametern:

<!-- components/_macros.html -->
{% macro card(title, description, href, badge=None) %}
<article class="card">
  <a href="{{ href }}" class="card__link">
    {% if badge %}
    <span class="card__badge">{{ badge }}</span>
    {% endif %}
    <h3 class="card__title">{{ title }}</h3>
    {% if description %}
    <p class="card__description">{{ description }}</p>
    {% endif %}
  </a>
</article>
{% endmacro %}

{% macro optimized_image(image_config, loading="lazy") %}
{% if image_config.get("svg") %}
  <img src="{{ image_config.svg }}"
       width="{{ image_config.width }}"
       height="{{ image_config.height }}"
       alt="{{ image_config.alt }}">
{% else %}
  <picture>
    <source type="image/webp"
            srcset="{{ image_config.webp_srcset }}"
            sizes="(max-width: 768px) 100vw, 50vw">
    <img src="{{ image_config.fallback }}"
         width="{{ image_config.width }}"
         height="{{ image_config.height }}"
         alt="{{ image_config.alt }}"
         loading="{{ loading }}">
  </picture>
{% endif %}
{% endmacro %}

Macros werden in Seiten-Templates importiert und verwendet:

{% from "components/_macros.html" import card, optimized_image %}

<section class="projects">
  {% for project in projects %}
    {{ card(
      title=project.title,
      description=project.description,
      href=project.link,
      badge="New" if project.is_new else None
    ) }}
  {% endfor %}
</section>

Macros ersetzen React-Komponenten für Präsentationsmuster. Sie akzeptieren Parameter, unterstützen Standardwerte und lassen sich mit anderen Macros kombinieren. Der entscheidende Unterschied: Macros werden einmalig auf dem Server gerendert und erzeugen statisches HTML. React-Komponenten hingegen werden auf dem Client gerendert und verwalten einen Zustand. Für die Darstellung von Inhalten sind Macros das richtige Werkzeug.

Template-Kontext und Globals

Jinja2-Globals sind Funktionen, die in jedem Template ohne explizite Übergabe verfügbar sind:

# In main.py — register globals
templates.env.globals["asset"] = lambda path: make_asset_url(_asset_map, path)
templates.env.globals["csrf_token"] = generate_csrf_token
templates.env.globals["analytics_script"] = analytics.tracking_script

Das asset()-Global erzeugt versionierte URLs. Das csrf_token()-Global generiert frische CSRF-Token. Das analytics_script()-Global fügt das Tracking-Snippet ein. Diese Funktionen sind in jedem Template aufrufbar, ohne dass der Route-Handler sie explizit übergeben muss.

Für i18n ist die Einrichtung aufwendiger — Übersetzungsfunktionen benötigen Zugriff auf die Locale des aktuellen Requests:

# i18n/jinja.py
def setup_i18n_jinja(env):
    """Register translation functions as Jinja2 globals."""
    env.globals["_"] = get_translation        # _('ui.nav.about')
    env.globals["locale_prefix"] = get_locale_prefix  # '/ja' or ''
    env.globals["current_locale"] = get_current_locale
    env.globals["all_locales"] = get_all_locales
    env.globals["alternate_urls"] = get_alternate_urls
    env.globals["lang_attr"] = get_lang_attr  # 'ja' for HTML lang
    env.globals["og_locale"] = get_og_locale  # 'ja_JP' for og:locale
    env.globals["jsonld_lang"] = get_jsonld_lang  # 'ja-JP' for JSON-LD

Jede Funktion liest die Locale aus der Kontextvariable des Requests, die von der Locale-Middleware gesetzt wird. Im Template liefert der Aufruf {{ _('ui.nav.about') }} den übersetzten String für die Locale des aktuellen Requests — ganz ohne expliziten Locale-Parameter.

Bedingte Blöcke

Das Blocksystem von Jinja2 unterstützt bedingte Überschreibungen:

<!-- base.html -->
{% block head %}{% endblock %}

<!-- pages/blog/post.html -->
{% block head %}
<script type="application/ld+json">
{
  "@type": "Article",
  "headline": "{{ post.meta.title }}",
  "author": { "@id": "https://blakecrosley.com/#person" },
  "datePublished": "{{ post.meta.date.isoformat() }}",
  "dateModified": "{{ post.meta.updated.isoformat() if post.meta.updated else post.meta.date.isoformat() }}"
}
</script>

{% if post.meta.scripts %}
{% for script in post.meta.scripts %}
<script defer src="{{ asset(script.lstrip('/static/')) }}"></script>
{% endfor %}
{% endif %}

{% if post.meta.styles %}
{% for style in post.meta.styles %}
<link rel="stylesheet" href="{{ asset(style.lstrip('/static/')) }}">
{% endfor %}
{% endif %}
{% endblock %}

Blogbeiträge deklarieren ihre Abhängigkeiten im YAML-Frontmatter (scripts: ["/static/js/boids.js"]). Das Template bindet diese bedingt ein. Seiten ohne zusätzliche Skripte oder Styles liefern auch keine aus — kein toter Code, keine ungenutzten Importe.

Benutzerdefinierte Filter

Jinja2-Filter transformieren Daten während des Renderings. Der sanitize-Filter verhindert XSS in nutzergenerierten Inhalten:

import nh3

ALLOWED_TAGS = {"a", "b", "blockquote", "br", "code", "em", "h1", "h2",
                "h3", "h4", "h5", "h6", "hr", "i", "img", "li", "ol",
                "p", "pre", "span", "strong", "table", "td", "th", "tr", "ul"}

def sanitize_html(value: str) -> str:
    """Sanitize HTML to prevent XSS attacks."""
    if not value:
        return ""
    return nh3.clean(
        value,
        tags=ALLOWED_TAGS,
        attributes={"a": {"href", "title"}, "img": {"src", "alt"}},
        link_rel="noopener noreferrer",
    )

templates.env.filters["sanitize"] = sanitize_html

Im Template: {{ user_content | sanitize }}. Die nh3-Bibliothek ist ein Rust-basierter HTML-Sanitizer — schnell und sicher. Sie entfernt sämtliche Tags und Attribute, die nicht auf der Erlaubnisliste stehen, und verhindert so Stored XSS selbst dann, wenn der Inhalt aus einer nicht vertrauenswürdigen Quelle stammt.10


HTMX im Detail

HTMX macht jedes HTML-Element fähig, HTTP-Anfragen zu senden und die Antwort in das DOM einzufügen. Die zentrale Erkenntnis ist architektonischer Natur: Serverseitig gerendertes HTML ist die API. Der Server liefert die endgültige Darstellung. Kein clientseitiges Rendering, keine JSON-Serialisierung, keine Hydration.

Kernattribute

Attribut Zweck Beispiel
hx-get GET-Anfrage senden hx-get="/search?q=term"
hx-post POST-Anfrage senden hx-post="/contact"
hx-target Ziel für die Antwort hx-target="#results"
hx-swap Art der Einfügung hx-swap="innerHTML" (Standard), outerHTML, beforeend
hx-trigger Auslöser für die Anfrage hx-trigger="click", keyup changed delay:300ms, load
hx-indicator Element, das während der Anfrage angezeigt wird hx-indicator="#spinner"
hx-push-url Browser-URL aktualisieren hx-push-url="true"
hx-replace-url URL ohne Verlaufseintrag ersetzen hx-replace-url="true"

Muster 1: Interaktives Quiz (mehrstufiger Serverstatus)

blakecrosley.com enthält ein interaktives Quiz, das Benutzer durch die Werkzeugauswahl führt. Der gesamte Quiz-Zustand liegt auf dem Server — kein clientseitiges State Management:

<!-- _quiz_container.html — initial load -->
<div hx-get="/api/quiz/claude-vs-codex/step?answers="
     hx-trigger="load"
     hx-swap="innerHTML"
     id="quiz-wrapper">
  <p>Loading quiz...</p>
</div>
<!-- _quiz_step.html — each question -->
<div class="quiz-step" id="quiz-container">
  <p>Question {{ step }} of {{ total }}</p>
  <h3>{{ question.question }}</h3>
  <div class="quiz-step__options">
    {% for opt in question.options %}
    <button class="quiz-step__btn"
            hx-get="/api/quiz/claude-vs-codex/step?answers={{ answers }},{{ opt.value }}"
            hx-target="#quiz-container"
            hx-swap="outerHTML">
      {{ opt.label }}
    </button>
    {% endfor %}
  </div>
</div>

Jeder Klick auf eine Schaltfläche sendet die gesammelten Antworten als Query-Parameter. Der Server berechnet die nächste Frage oder das Endergebnis anhand der Antworthistorie. Der Zustand akkumuliert sich in der URL — keine Cookies, keine Sessions, kein clientseitiges JavaScript. Das Quiz schreitet durch outerHTML-Swaps voran: Jede Antwort ersetzt das gesamte Quiz-Schritt-Element.

Muster 2: Paginierte Blog-Liste

Die Schreibseite nutzt HTMX für nahtlose Paginierung mit URL-Aktualisierung:

<!-- Pagination link -->
<a href="/writing?page=2&category=Engineering"
   hx-get="/writing?page=2&category=Engineering"
   hx-target="#writing-content"
   hx-swap="innerHTML"
   hx-replace-url="true"
   hx-indicator="#writing-loading"
   aria-label="Go to page 2">
  2
</a>

Vier Attribute arbeiten zusammen:

  1. hx-get sendet die Anfrage an dieselbe URL wie das href (Progressive Enhancement — funktioniert auch ohne JavaScript)
  2. hx-target platziert die Antwort im #writing-content-Container
  3. hx-replace-url="true" aktualisiert die Browser-URL, ohne einen Verlaufseintrag hinzuzufügen
  4. hx-indicator zeigt einen Lade-Spinner während der Anfrage an

Der Server erkennt HTMX-Anfragen über den HX-Request-Header und liefert nur das Beitragslisten-Fragment statt der vollständigen Seite zurück. Deshalb fügt die Security-Headers-Middleware Vary: HX-Request hinzu — damit CDN-Caches die vollständige Seite und das Fragment getrennt speichern.11

Muster 3: Suche mit Debounce

<input type="search" name="q"
       hx-get="/api/search"
       hx-trigger="keyup changed delay:300ms"
       hx-target="#results"
       hx-indicator="#search-spinner" />
<div id="results"></div>

Das hx-trigger-Attribut kombiniert drei Modifikatoren:

  • keyup löst beim Loslassen einer Taste aus
  • changed löst nur aus, wenn sich der Wert tatsächlich geändert hat (verhindert doppelte Anfragen durch Modifikatortasten)
  • delay:300ms entprellt — wartet 300 ms nach dem letzten Keyup, bevor ausgelöst wird

Der Server liefert ein gerendertes HTML-Fragment zurück:

@router.get("/api/search")
async def search(request: Request, q: str = ""):
    results = search_content(q)
    return templates.TemplateResponse("components/_search_results.html", {
        "request": request,
        "results": results,
        "query": q,
    })

Kein clientseitiger Zustand. Keine Debounce-Bibliothek. Kein useEffect. Das Template rendert die Ergebnisse, HTMX fügt sie ein, und der Server bleibt die einzige Quelle der Wahrheit.

Muster 4: Out-of-Band (OOB) Swaps

Manchmal muss eine einzelne Serveraktion mehrere DOM-Elemente aktualisieren. Der Out-of-Band-Swap-Mechanismus von HTMX bewältigt dies ohne clientseitige Orchestrierung:

<!-- Server returns multiple elements in one response -->
<!-- Primary target: swapped normally via hx-target -->
<div id="cart-items">
  <ul>
    <li>Widget A — $29.99</li>
    <li>Widget B — $14.99</li>
  </ul>
</div>

<!-- OOB target: swapped independently via hx-swap-oob -->
<span id="cart-count" hx-swap-oob="true">2 items</span>
<span id="cart-total" hx-swap-oob="true">$44.98</span>

Das Attribut hx-swap-oob="true" weist HTMX an, das Element anhand seiner id überall im DOM zu finden und zu ersetzen — unabhängig vom hx-target. Dies ersetzt Reacts „State nach oben heben”-Muster: Der Server berechnet den gesamten abgeleiteten Zustand und sendet das fertige HTML für jedes Element in einer einzigen Antwort.

Ein Kontaktformular veranschaulicht dies gut: Das Absenden des Formulars könnte den Formularkörper durch eine Erfolgsmeldung ersetzen und gleichzeitig ein Benachrichtigungs-Badge über einen OOB-Swap aktualisieren:

HTMX kann Standard-Navigationslinks „boosten”, sodass sie AJAX statt vollständiger Seitenladevorgänge verwenden:

<nav hx-boost="true">
  <a href="/about">About</a>
  <a href="/writing">Writing</a>
  <a href="/guides">Guides</a>
</nav>

Mit hx-boost="true" ruft ein Klick auf einen Link die Seite per AJAX ab, tauscht den <body>-Inhalt aus und aktualisiert die URL — ohne vollständigen Seitenneulade. Die Browserhistorie funktioniert normal (Vor-/Zurück-Tasten). Falls JavaScript fehlschlägt, funktionieren die Links als reguläre Navigation.

Der Vorteil liegt in der wahrgenommenen Performance: Geboostete Navigation fühlt sich sofort an, da der Browser weder CSS neu parsen, noch Skripte neu auswerten oder das Layout neu rendern muss. Nur der <body>-Inhalt ändert sich. Boosted Links eignen sich besonders für Hauptnavigationselemente und lassen Seitenwechsel wie eine Single-Page Application wirken — ohne die SPA-Architektur.

Muster 6: HTMX Request-Header

HTMX sendet benutzerdefinierte Header mit jeder Anfrage:

Header Wert Anwendungsfall
HX-Request true HTMX-Anfragen serverseitig erkennen
HX-Target Element-ID Zielelement der Antwort identifizieren
HX-Trigger Element-ID Auslösendes Element identifizieren
HX-Current-URL Vollständige URL Aktuelle Seite des Benutzers ermitteln

Der Server kann HX-Request nutzen, um unterschiedliche Antworten zurückzugeben:

@router.get("/writing")
async def writing(request: Request, page: int = 1, category: str = None):
    posts = load_all_posts(page=page, category=category)
    context = {"request": request, "posts": posts, "current_page": page}

    # HTMX request: return only the post list fragment
    if request.headers.get("HX-Request"):
        return templates.TemplateResponse(
            "pages/writing/_post_list.html", context
        )

    # Normal request: return the full page
    return templates.TemplateResponse("pages/writing/index.html", context)

Dieses Dual-Response-Muster ist zentral für die Architektur. Ein vollständiger Seitenladevorgang liefert das komplette Dokument (Basis-Template + Seiteninhalt). Eine HTMX-Navigation liefert nur den geänderten Inhalt. Der Server entscheidet — nicht der Client.

Muster 7: Progressive Enhancement

Jeder HTMX-Link auf blakecrosley.com enthält ein Standard-href-Attribut:

<a href="/writing?page=2"
   hx-get="/writing?page=2"
   hx-target="#writing-content"
   hx-swap="innerHTML">
  Next Page
</a>

Falls JavaScript nicht geladen werden kann, funktioniert das href als normaler Link. Wird HTMX geladen, fängt es den Klick ab und führt einen AJAX-Swap durch. Das ist Progressive Enhancement: Die Seite funktioniert ohne JavaScript, und HTMX verbessert das Erlebnis, wenn verfügbar.

Muster 8: Ladezustände

<button hx-post="/api/contact"
        hx-target="#form-result"
        hx-indicator="#submit-spinner">
  <span id="submit-spinner" class="htmx-indicator">Sending...</span>
  <span>Send Message</span>
</button>

HTMX fügt dem auslösenden Element während Anfragen die Klasse htmx-request hinzu. Das hx-indicator-Attribut verweist auf ein Element, das während der Anfrage sichtbar wird. Gestylt wird es mit CSS:

.htmx-indicator {
  display: none;
}
.htmx-request .htmx-indicator,
.htmx-request.htmx-indicator {
  display: inline;
}

Kein Ladezustand-Management. Kein useState(false). Kein setLoading(true). CSS steuert die Sichtbarkeit, HTMX übernimmt den Klassen-Toggle.


Alpine.js-Muster

Alpine.js füllt die Lücke, die HTMX offenlässt: clientseitiger Zustand, der niemals den Server berühren muss. Wenn ein Benutzer auf ein Dropdown klickt und es sich öffnet, existiert dieser Zustand ausschließlich im Browser. Alpine.js verwaltet ihn mit HTML-Attributen.

Die Grenzregel

Die Grenze zwischen HTMX und Alpine.js ist klar definiert:

Zustandstyp Werkzeug Beispiel
Benötigt Serverdaten HTMX Suchergebnisse, Formularvalidierung, Paginierung
Existiert nur im Browser Alpine.js Dropdown öffnen/schließen, mobiles Menü umschalten, Modal-Sichtbarkeit
Kombiniert beides Beide Sprachumschalter (Alpine.js-Toggle, HTMX-artige Navigation)

Mobile Navigation

Das Basis-Template umschließt den gesamten Header mit einer Alpine.js-Komponente:

<div x-data="{ navOpen: false, langOpen: false }"
     @keydown.escape.window="navOpen = false; langOpen = false">

  <!-- Mobile hamburger button -->
  <button @click="navOpen = !navOpen; langOpen = false"
          :aria-expanded="navOpen"
          :class="navOpen ? 'nav__toggle is-open' : 'nav__toggle'"
          aria-label="Toggle navigation">
    <span class="nav__toggle-icon">
      <span class="nav__toggle-bar"></span>
      <span class="nav__toggle-bar"></span>
      <span class="nav__toggle-bar"></span>
    </span>
  </button>

  <!-- Mobile menu panel -->
  <div class="mobile-menu" x-show="navOpen" x-cloak>
    <nav class="mobile-menu__nav">
      <a href="/about" @click="navOpen = false">About</a>
      <a href="/#work" @click="navOpen = false">Work</a>
      <a href="/writing" @click="navOpen = false">Writing</a>
    </nav>
  </div>
</div>

Wichtige Alpine.js-Muster:

  • x-data deklariert den Komponentenbereich und den Anfangszustand
  • x-show steuert die Sichtbarkeit basierend auf dem Zustand (nutzt CSS display: none)
  • x-cloak verbirgt das Element, bis Alpine.js initialisiert ist (verhindert ein Aufblitzen ungestylter Inhalte)
  • @click bindet Klick-Handler mit Ausdrücken
  • :aria-expanded (Kurzform für x-bind:aria-expanded) setzt Attribute dynamisch
  • @keydown.escape.window lauscht global auf die Escape-Taste, um Panels zu schließen

Der Sprachumschalter verwendet Alpine.js für den Toggle-Zustand mit @click.away zum Schließen bei Klick außerhalb:

<div x-data="{ open: false }"
     @click.away="open = false"
     @keydown.escape.window="open = false">

  <button @click="open = !open"
          :aria-expanded="open"
          aria-haspopup="listbox">
    English
    <svg :class="{ 'rotated': open }">...</svg>
  </button>

  <ul :class="{ 'is-open': open }"
      :aria-hidden="!open"
      role="listbox"
      x-cloak>
    <li role="option">
      <a href="/ja/about">日本語</a>
    </li>
    <!-- more languages -->
  </ul>
</div>

Der @click.away-Modifier schließt das Dropdown bei Klick außerhalb. Alpine.js erledigt dies mit einem einzigen Attribut — kein Event-Listener-Registrierung, kein Aufräumen, kein Ref-Management.

Wann Alpine.js und wann Vanilla JavaScript

Alpine.js eignet sich, wenn:

  • Der Zustand auf ein einzelnes DOM-Element beschränkt ist (Dropdown, Modal, Toggle)
  • Interaktionen binär oder einfach sind (öffnen/schließen, ein-/ausblenden, umschalten)
  • Mehrere Elemente auf dieselbe Zustandsänderung reagieren müssen
  • Barrierefreiheits-Attribute mit der Sichtbarkeit synchron bleiben müssen

Vanilla JavaScript eignet sich, wenn:

  • Die Interaktion komplexe Berechnungen erfordert (Visualisierungen, Simulationen)
  • Die Komponente eine eigene Render-Schleife hat (Canvas, Animation)
  • Performance entscheidend ist (Alpine.js erzeugt Overhead pro x-data-Komponente)
  • Die Logik 20–30 Zeilen Alpine.js-Ausdrücke übersteigt

blakecrosley.com verwendet Alpine.js für Navigation, Sprachumschaltung und Inhalts-Toggles. Die 20 interaktiven Blog-Komponenten (Boids-Simulation, Hamming-Code-Visualisierer usw.) nutzen Vanilla JavaScript, da sie Canvas-Rendering und komplexe Zustandsautomaten erfordern.


End-to-End-Beispiel: Kategoriefilterung auf /writing

Dieser Abschnitt verfolgt ein reales Feature aus der Produktions-Codebasis durch jede Schicht: Route, Template, HTMX-Interaktion, Sicherheit, Caching und gerendertes Ergebnis. Das Feature: Kategorie-Tabs auf der Writing-Seite, die Blogbeiträge ohne vollständiges Neuladen der Seite filtern.

Die Route (app/routes/pages.py:508)

async def writing_listing(request: Request, page: int = 1, category: str | None = None):
    """Writing page — blog posts and external publications."""
    templates = get_templates(request)
    markdown_posts = load_all_posts(published_only=True)
    all_posts = CUSTOM_BLOG_POSTS + markdown_posts

    # Filter by category if specified
    if category and category in CATEGORY_MAP:
        display_name = CATEGORY_MAP[category]
        all_posts = [
            p for p in all_posts
            if _get_post_category(p).lower() == display_name.lower()
        ]

    # Pagination
    total_pages = max(1, (len(all_posts) + POSTS_PER_PAGE - 1) // POSTS_PER_PAGE)
    page = max(1, min(page, total_pages))
    paginated = all_posts[(page - 1) * POSTS_PER_PAGE : page * POSTS_PER_PAGE]

    template_context = {
        "request": request,
        "posts": paginated,
        "categories": categories,
        "current_category": category,
        "current_page": page,
        "total_pages": total_pages,
        # ... SEO: canonical, prev/next URLs
    }

    # HTMX partial: return just the post list fragment
    if request.headers.get("HX-Request"):
        return templates.TemplateResponse(
            "pages/writing/_post_list.html",
            template_context,
        )

    # Full page for direct navigation
    return templates.TemplateResponse(
        "pages/writing/index.html",
        template_context,
    )

Die HX-Request-Header-Prüfung ist das zentrale Muster: dieselbe Route, dieselben Daten, unterschiedliches Template. HTMX erhält ein Fragment. Browser erhalten die vollständige Seite.

Die Kategorie-Tabs (HTMX)

<!-- Category filter tabs -->
<nav class="writing-categories">
  <a href="/writing"
     hx-get="/writing"
     hx-target="#post-list"
     hx-push-url="true"
     class="category-tab {% if not current_category %}active{% endif %}">
    All ({{ total_posts }})
  </a>
  {% for cat in categories %}
  <a href="/writing?category={{ cat.slug }}"
     hx-get="/writing?category={{ cat.slug }}"
     hx-target="#post-list"
     hx-push-url="true"
     class="category-tab {% if current_category == cat.slug %}active{% endif %}">
    {{ cat.name }} ({{ cat.count }})
  </a>
  {% endfor %}
</nav>

<div id="post-list">
  {% include "pages/writing/_post_list.html" %}
</div>

Jeder Tab hat sowohl href (funktioniert ohne JavaScript) als auch hx-get (tauscht nur die Beitragsliste aus). hx-push-url aktualisiert die Browser-URL, sodass die gefilterte Ansicht teilbar und als Lesezeichen speicherbar ist.

Das Partial (pages/writing/_post_list.html)

Das Partial rendert identisch, ob es beim Seitenladen eingebunden oder von HTMX ausgetauscht wird:

{% for post in posts %}
<article class="post-card">
  <a href="{{ locale_prefix() }}/blog/{{ post.meta.slug }}">
    <h3>{{ post.meta.title }}</h3>
    <p>{{ post.meta.description }}</p>
    <time>{{ post.meta.date }}</time> · {{ post.reading_time }}m
  </a>
</article>
{% endfor %}

Kein spezielles HTMX-Markup im Partial. Keine clientseitige Rendering-Logik. Dasselbe HTML funktioniert für das initiale Laden der Seite und jeden nachfolgenden Filter.

Sicherheit

Kategoriewerte werden vor dem Filtern gegen CATEGORY_MAP (ein serverseitiges Dictionary) validiert. Ungültige Kategorien werden ignoriert, nicht zurückgegeben. Keine Benutzereingaben werden in SQL oder HTML interpoliert. Der CSP-Header blockiert Inline-Skripte.

Caching

Kategorie-Antworten sind dynamisch (kein CDN-Cache). Statische Assets (CSS, HTMX, Alpine.js) sind jedoch inhaltlich gehasht und nach dem ersten Laden unbegrenzt gecacht. Nachfolgende Kategoriewechsel übertragen nur das HTML-Partial (~3–5 KB) — kein CSS, kein JS, keine Bilder werden erneut geladen.

Was dieses Beispiel zeigt

Ein Feature, echter Produktionscode, null Build-Tools. Der Server filtert und rendert HTML. HTMX tauscht die Beitragsliste aus. Alpine.js ist nicht beteiligt (kein clientseitiger Zustand nötig). Die URL wird für Teilbarkeit aktualisiert. Progressive Enhancement: Die Tabs funktionieren als gewöhnliche Links ohne JavaScript. Eigener JavaScript-Code für dieses Feature insgesamt: null Zeilen.


Optionale Erweiterungen

Die folgenden Abschnitte behandeln Muster, die den Kern-Stack ergänzen, aber auf blakecrosley.com nicht eingesetzt werden. Sie sind enthalten, weil sie die häufigsten Ergänzungen darstellen, die Teams bei der Einführung dieser Architektur vornehmen.


Bootstrap 5 ohne Sass

Hinweis: blakecrosley.com verwendet reines CSS mit Custom Properties — kein Bootstrap. Dieser Abschnitt behandelt Bootstrap 5 als Option für Teams, die ein Utility-Framework ohne Build-Schritt einsetzen möchten. Das kompilierte CSS von Bootstrap lässt sich über ein CDN laden oder in Ihr Stylesheet einbinden. Die folgenden Muster sind generisch und funktionieren zusammen mit dem in vorherigen Abschnitten beschriebenen HTMX- und Alpine.js-Ansatz.

Bootstrap 5 hat die jQuery-Abhängigkeit entfernt und unterstützt die eigenständige Verwendung von CSS. Weder Sass noch PostCSS oder andere Build-Tools sind erforderlich, um Bootstraps Grid-System und Utility-Klassen zu nutzen.

CDN-freies Self-Hosting

blakecrosley.com hostet alle Vendor-Bibliotheken selbst:

<!-- base.html — no CDN, no external requests -->
<script defer src="{{ asset('js/vendor/alpine.min.js') }}"></script>
<script defer src="{{ asset('js/vendor/htmx.min.js') }}"></script>

Self-Hosting eliminiert externe Abhängigkeiten, verhindert, dass CDN-Ausfälle die Seite beeinträchtigen, und ermöglicht unveränderliches Caching mit Content-Hash-URLs. Laden Sie das kompilierte CSS von Bootstrap herunter (nicht die Sass-Quelldateien) und legen Sie es unter static/css/vendor/ ab.

Grid-System

Bootstraps Grid funktioniert mit einfachen HTML-Klassen:

<div class="container">
  <div class="row">
    <div class="col-12 col-md-8">
      <article>Main content</article>
    </div>
    <div class="col-12 col-md-4">
      <aside>Sidebar</aside>
    </div>
  </div>
</div>

Keine Sass-Mixins. Kein @include make-col(). Das kompilierte CSS enthält bereits die responsiven Grid-Klassen. Für benutzerdefinierte Breakpoints über Bootstraps Standardwerte hinaus schreiben Sie einfache CSS-Media-Queries.

Einfache CSS-Überschreibungen

Überschreiben Sie Bootstraps Standardwerte mit CSS Custom Properties und Standard-Selektoren:

/* Custom design tokens — no Sass, no Tailwind */
:root {
  --color-bg-dark:        #000000;
  --color-text-primary:   #ffffff;
  --color-text-secondary: rgba(255, 255, 255, 0.65);
  --color-text-tertiary:  rgba(255, 255, 255, 0.40);
  --spacing-sm:           1rem;
  --spacing-md:           1.5rem;
  --spacing-lg:           2rem;
  --gutter:               48px;
  --font-size-lg:         1.25rem;
}

/* Responsive override — the browser reads this at runtime */
@media (max-width: 768px) {
  :root {
    --gutter: var(--spacing-md);  /* 48px → 24px on mobile */
  }
}

/* Override Bootstrap's default body styles */
body {
  background: var(--color-bg-dark);
  color: var(--color-text-primary);
  font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", system-ui, sans-serif;
}

CSS Custom Properties kaskadieren durch das DOM, werden von übergeordneten Elementen vererbt und reagieren zur Laufzeit auf Media Queries. Sass-Variablen hingegen werden zu statischen Werten kompiliert und verschwinden. Dieser Unterschied ist entscheidend für Theming: Eine einzige Änderung einer Custom Property kann jeden abgeleiteten Wert aktualisieren — ganz ohne Neukompilierung.12

Utility-Klassen vs. Komponenten-CSS

Verwenden Sie Bootstrap-Utility-Klassen für einmalige Abstände und Layouts. Für wiederkehrende Muster setzen Sie auf Komponenten-CSS:

<!-- Bootstrap utility for one-off spacing -->
<div class="mt-4 mb-3 px-2">One-off layout</div>

<!-- Component class for repeated patterns -->
<article class="writing__item">
  <h3 class="writing__item-title">Post Title</h3>
  <p class="writing__item-description">Description</p>
</article>
/* Component CSS — BEM naming, reusable */
.writing__item {
  padding: var(--spacing-md);
  border-bottom: 1px solid rgba(255, 255, 255, 0.1);
  transition: background 0.15s ease;
}
.writing__item:hover {
  background: rgba(255, 255, 255, 0.03);
}
.writing__item-title {
  font-size: var(--font-size-lg);
  margin-bottom: 0.5rem;
}

Das Prinzip: Bootstrap-Utilities für Layout-Mechaniken (Margin, Padding, Flexbox). Benutzerdefiniertes CSS für die visuelle Identität (Farben, Typografie, Animationen). Mischen Sie niemals Utility-Klassen und Komponenten-Styling für denselben Zweck.


Internationalisierung und Lokalisierung

blakecrosley.com stellt Inhalte in 10 Sprachen bereit: Englisch, Japanisch, Koreanisch, vereinfachtes Chinesisch, traditionelles Chinesisch, Deutsch, Französisch, Spanisch, Polnisch und Portugiesisch (Brasilianisch).

URL-basiertes Locale-Routing

Das Locale befindet sich im URL-Pfad: /about (Englisch), /ja/about (Japanisch), /zh-Hans/about (vereinfachtes Chinesisch). Englisch ist die Standardsprache und hat kein Präfix.

# i18n/config.py
SUPPORTED_LOCALES = [
    "en", "zh-Hans", "zh-Hant", "fr", "de", "ja", "ko", "pl", "pt-BR", "es"
]
DEFAULT_LOCALE = "en"

Die Locale-Middleware extrahiert das Locale aus dem URL-Pfad:

# i18n/middleware.py
class LocaleMiddleware(BaseHTTPMiddleware):
    async def dispatch(self, request, call_next):
        path = request.url.path
        # Check if path starts with a supported locale
        for locale in SUPPORTED_LOCALES:
            if path.startswith(f"/{locale}/") or path == f"/{locale}":
                request.state.locale = locale
                # Strip locale prefix for route matching
                request.scope["path"] = path[len(f"/{locale}"):]
                break
        else:
            request.state.locale = DEFAULT_LOCALE

        response = await call_next(request)
        return response

Die Middleware entfernt das Locale-Präfix vor dem Route-Matching. Dadurch benötigen Route-Handler keine locale-spezifischen Pfade — /about verarbeitet sowohl Englisch (/about) als auch Japanisch (/ja/about), da die Middleware den Pfad normalisiert.

Übersetzungsfunktionen in Templates

Jinja2-Globals stellen Übersetzungsfunktionen bereit:

<!-- Template usage -->
<h3>{{ _('ui.footer.navigate') | default('Navigate') }}</h3>
<a href="{{ locale_prefix() }}/about">
  {{ _('ui.nav.about') | default('About') }}
</a>

Die Funktion _() sucht einen Übersetzungsschlüssel im Speicher-Cache. Der Filter | default() liefert den englischen Fallback, falls die Übersetzung fehlt. Die Funktion locale_prefix() gibt das URL-Präfix für das aktuelle Locale zurück ("" für Englisch, "/ja" für Japanisch).

Hreflang-Tags

Jede Seite enthält Hreflang-Tags für alle unterstützten Locales:

<!-- Generated in base.html -->
{% for alt in alternate_urls(request.url.path) %}
<link rel="alternate" hreflang="{{ alt.hreflang }}" href="{{ alt.url }}">
{% endfor %}

Das erzeugt:

<link rel="alternate" hreflang="en" href="https://blakecrosley.com/about">
<link rel="alternate" hreflang="ja" href="https://blakecrosley.com/ja/about">
<link rel="alternate" hreflang="zh-Hans" href="https://blakecrosley.com/zh-Hans/about">
<!-- ... all 10 locales -->
<link rel="alternate" hreflang="x-default" href="https://blakecrosley.com/about">

Suchmaschinen verwenden Hreflang, um in den Suchergebnissen die korrekte Sprachversion auszuliefern. Der x-default-Eintrag verweist auf die englische Version als Fallback.13

Übersetzungsspeicherung und Speicher-Cache

Übersetzungen werden in Cloudflare D1 (SQLite an der Edge) gespeichert und über den lifespan-Handler in einen In-Memory-Cache geladen:

@asynccontextmanager
async def lifespan(app: FastAPI):
    # Load translations into memory at startup
    for locale in SUPPORTED_LOCALES:
        data = await fetch_translations(locale)
        TRANSLATIONS[locale] = data
    yield

app = FastAPI(lifespan=lifespan)

Der Speicher-Cache vermeidet Datenbankabfragen bei jedem Seitenaufruf. Übersetzungsaktualisierungen erfordern eine Cache-Aktualisierung (ausgelöst über einen Admin-Endpunkt oder ein Deployment). Diese Architektur opfert Aktualität zugunsten der Performance — Übersetzungen ändern sich selten, Seitenaufrufe hingegen erfolgen bei jeder Anfrage.

Gesundheitsüberwachung

blakecrosley.com enthält einen i18n-Health-Check-Endpunkt, der die Übersetzungsabdeckung pro Locale überwacht:

@app.get("/health/i18n")
async def health_i18n():
    cache = get_translation_cache()
    result = {
        "status": "healthy",
        "cache_loaded": cache.is_loaded,
        "locales": {},
        "alerts": [],
    }

    # Check coverage for each locale
    for locale in SUPPORTED_LOCALES:
        coverage = await calculate_coverage(locale, en_count)
        result["locales"][locale] = {"coverage": round(coverage, 2)}

        if coverage < 99.5:
            result["alerts"].append(
                f"{locale}: {coverage:.1f}% coverage (threshold: 99.5%)"
            )
            result["status"] = "warning"

    return result

Der Schwellenwert von 99,5 % Abdeckung erkennt fehlende Übersetzungen, bevor Benutzer auf unübersetzte Zeichenketten stoßen. Der Health-Endpunkt ist in das Monitoring von Railway integriert und warnt bei sinkender Abdeckung — beispielsweise nach dem Hinzufügen neuer UI-Zeichenketten, die noch nicht übersetzt wurden.

Locale-abhängiges Content-Rendering

Blogbeiträge und Guides unterstützen locale-spezifische Übersetzungen von Metadaten und Inhalten:

# In route handler
translated = get_blog_translation(post.meta.slug, locale)
return templates.TemplateResponse("pages/blog/post.html", {
    "request": request,
    "post": post,
    "translated_title": translated.title if translated else post.meta.title,
    "translated_description": translated.description if translated else post.meta.description,
})
<!-- In template -->
<h1>{{ translated_title }}</h1>
<p class="post__description">{{ translated_description }}</p>
<!-- Body content falls back to English if translation unavailable -->
{{ post.html | sanitize | safe }}

Das Muster ist konsistent: zuerst den übersetzten Inhalt verwenden, bei Fehlen auf Englisch zurückfallen. So ist eine teilweise Übersetzung möglich — ein japanischer Benutzer sieht übersetzte Titel und Beschreibungen, auch wenn der vollständige Artikeltext auf Englisch bleibt. Der Jinja2-Filter | default() kodiert dieses Muster in einer einzigen Pipe:

{{ translated.title if translated else post.meta.title }}

Locale-Datenübersetzung

Statische Inhalte wie Projektbeschreibungen und Navigationsbezeichnungen werden über Hilfsfunktionen übersetzt, die dieselbe Datenstruktur beibehalten und locale-spezifische Zeichenketten einfügen:

# i18n/data.py
def translate_projects(projects: list, locale: str) -> list:
    """Return projects with translated titles and descriptions."""
    if locale == "en":
        return projects
    translated = []
    for project in projects:
        t = get_translation(f"project.{project['slug']}.title", locale)
        d = get_translation(f"project.{project['slug']}.description", locale)
        translated.append({
            **project,
            "title": t or project["title"],
            "description": d or project["description"],
        })
    return translated

Dieser Ansatz hält die Übersetzungsschicht von der Datenschicht getrennt. Routen übergeben unabhängig vom Locale dieselbe projects-Liste. Die Übersetzungsfunktionen umhüllen die Daten transparent.

Sitemap mit Hreflang-Alternativen

Die dynamische Sitemap enthält alle Seiten in allen Locales mit Querverweisen:

@app.get("/sitemap.xml")
async def sitemap():
    for page in static_pages:
        for locale in SUPPORTED_LOCALES:
            # Each URL entry includes alternates for all locales
            locale_path = f"/{locale}{path}" if locale != "en" else path
            xml_parts.append(f"<loc>{base_url}{locale_path}</loc>")
            # Add xhtml:link alternates
            for alt_locale in SUPPORTED_LOCALES:
                alt_path = f"/{alt_locale}{path}" if alt_locale != "en" else path
                hreflang = LOCALE_TO_HREFLANG[alt_locale]
                xml_parts.append(
                    f'<xhtml:link rel="alternate" hreflang="{hreflang}" '
                    f'href="{base_url}{alt_path}"/>'
                )

Das ergibt 10 URL-Einträge pro Seite (einen pro Locale), jeweils mit 11 alternativen Links (10 Locales + x-default). Bei einer Website mit 50 Seiten enthält die Sitemap 500 URL-Einträge mit 5.500 Hreflang-Links. Die Sitemap wird dynamisch generiert und für eine Stunde zwischengespeichert.


Datenbankmuster

Hinweis: blakecrosley.com verwendet Cloudflare D1 (serverless SQLite) über HTTP für alle persistenten Daten, nicht SQLAlchemy. Dieser Abschnitt behandelt das standardmäßige async-Muster von SQLAlchemy für FastAPI-Projekte, die eine relationale Datenbank benötigen — das häufigste Produktionssetup für diesen Stack.

SQLAlchemy 2.0 Async

Für Anwendungen, die eine relationale Datenbank benötigen, lässt sich die async-Unterstützung von SQLAlchemy 2.0 sauber in FastAPI integrieren:

from sqlalchemy.ext.asyncio import create_async_engine, AsyncSession
from sqlalchemy.orm import sessionmaker, DeclarativeBase

engine = create_async_engine("sqlite+aiosqlite:///./data.db")
async_session = sessionmaker(engine, class_=AsyncSession, expire_on_commit=False)

class Base(DeclarativeBase):
    pass

Installationshinweis (SQLAlchemy 2.0.50+): Seit 2.0.50 wird die greenlet-Abhängigkeit des async-Stacks nicht mehr standardmäßig installiert. Verwenden Sie das asyncio-Extra, damit sie mitinstalliert wird; andernfalls schlägt das erste await gegen die Engine mit einem Missing-greenlet-Fehler fehl:23

pip install "sqlalchemy[asyncio]" aiosqlite

SQLAlchemy 2.0.50 erfordert außerdem Python 3.10+ (3.7–3.9 wurden eingestellt) und fügt Wheels für Free-Threaded (3.13t) hinzu.23

Dependency Injection für Datenbanksitzungen

from fastapi import Depends
from sqlalchemy.ext.asyncio import AsyncSession

async def get_db() -> AsyncGenerator[AsyncSession, None]:
    async with async_session() as session:
        try:
            yield session
            await session.commit()
        except Exception:
            await session.rollback()
            raise

@router.get("/users/{user_id}")
async def get_user(request: Request, user_id: int, db: AsyncSession = Depends(get_db)):
    result = await db.execute(select(User).where(User.id == user_id))
    user = result.scalar_one_or_none()
    if not user:
        raise HTTPException(404, "User not found")
    return templates.TemplateResponse("pages/user.html", {
        "request": request, "user": user
    })

Die get_db-Dependency verwaltet den Sitzungslebenszyklus: Sie öffnet eine Sitzung, übergibt sie per Yield an den Route Handler, committet bei Erfolg und führt bei einer Exception ein Rollback aus. Jede Datenbankoperation verwendet parametrisierte Abfragen — niemals String-Interpolation.

Pydantic-Integration

Pydantic-Modelle validieren Eingaben an der API-Grenze und serialisieren Ausgaben für Templates:

from pydantic import BaseModel, EmailStr

class ContactForm(BaseModel):
    name: str
    email: EmailStr
    message: str

@router.post("/contact")
async def submit_contact(request: Request, form: ContactForm):
    # form.name, form.email, form.message are validated
    await send_email(form)
    return templates.TemplateResponse("components/_contact_success.html", {
        "request": request
    })

Pydantic validiert Typen, Formate (E-Mail, URL) und Constraints (min./max. Länge), bevor der Route Handler ausgeführt wird. Ungültige Eingaben geben automatisch eine 422-Antwort zurück. Das ersetzt clientseitige Formularvalidierungsbibliotheken — der Server validiert, und HTMX tauscht entweder die Erfolgsmeldung oder das Fehlerfeedback ein.

Migrationen mit Alembic

Alembic verwaltet Änderungen am Datenbankschema:

# Generate a migration from model changes
alembic revision --autogenerate -m "add user preferences table"

# Apply migrations
alembic upgrade head

# Roll back one migration
alembic downgrade -1

Die Autogenerate-Funktion vergleicht SQLAlchemy-Modelle mit dem aktuellen Datenbankschema und generiert Migrationsskripte. Diese Skripte sind versionierte Python-Dateien, die im Repository liegen:

# alembic/versions/001_add_user_preferences.py
def upgrade():
    op.create_table(
        "user_preferences",
        sa.Column("id", sa.Integer, primary_key=True),
        sa.Column("user_id", sa.Integer, sa.ForeignKey("users.id")),
        sa.Column("locale", sa.String(10), default="en"),
        sa.Column("theme", sa.String(20), default="dark"),
    )

def downgrade():
    op.drop_table("user_preferences")

Migrationen laufen während des Deployments (bevor die Anwendung startet). Dadurch ist sichergestellt, dass das Datenbankschema zum Anwendungscode passt. Bei blakecrosley.com liegen die meisten Daten in Cloudflare D1 (Zugriff über HTTP), daher gelten Alembic-Migrationen für die lokale SQLite- oder PostgreSQL-Datenbank, die für Sitzungsdaten und Analytics verwendet wird.

Das Cloudflare D1-Muster

blakecrosley.com verwendet Cloudflare D1 als Remote-Datenbank, auf die über einen Cloudflare Worker-Proxy zugegriffen wird:

class D1Client:
    """HTTP client for Cloudflare D1 via Worker proxy."""

    def __init__(self, worker_url: str, auth_secret: str):
        self.worker_url = worker_url
        self.auth_secret = auth_secret

    async def fetch_all(self, sql: str, params: list = None) -> list[dict]:
        async with httpx.AsyncClient() as client:
            response = await client.post(
                f"{self.worker_url}/query",
                json={"sql": sql, "params": params or []},
                headers={"Authorization": f"Bearer {self.auth_secret}"},
            )
            return response.json()["results"]

Dieses Muster eignet sich für Anwendungen, die eine Datenbank benötigen, aber keinen Datenbankserver verwalten möchten. D1 ist SQLite am Edge von Cloudflare und wird über HTTP angesprochen. Der Worker-Proxy übernimmt Authentifizierung und Rate Limiting. Der Trade-off ist Latenz: Jede Abfrage ist ein HTTP-Request (~50-100ms) statt einer lokalen Datenbankverbindung (~1-5ms). Der In-Memory-Cache beim Start mindert das bei leseintensiven Workloads wie Übersetzungen.


Sicherheit

Middleware für Security Headers

blakecrosley.com implementiert gehärtete Security Headers über eine eigene Middleware:

class SecurityHeadersMiddleware(BaseHTTPMiddleware):
    CSP_DIRECTIVES = {
        "default-src": "'self'",
        "script-src": "'self' 'unsafe-inline' 'unsafe-eval'",
        "style-src": "'self' 'unsafe-inline'",
        "img-src": "'self' data: https:",
        "connect-src": "'self'",
        "frame-ancestors": "'self'",
        "base-uri": "'self'",
        "form-action": "'self'",
        "upgrade-insecure-requests": "",
    }

    async def dispatch(self, request, call_next):
        response = await call_next(request)
        response.headers["X-Content-Type-Options"] = "nosniff"
        response.headers["X-Frame-Options"] = "SAMEORIGIN"
        response.headers["Referrer-Policy"] = "strict-origin-when-cross-origin"
        response.headers["Strict-Transport-Security"] = (
            "max-age=31536000; includeSubDomains"
        )
        response.headers["Cross-Origin-Opener-Policy"] = "same-origin"
        response.headers["Content-Security-Policy"] = self.csp
        response.headers["Permissions-Policy"] = self.PERMISSIONS_POLICY
        return response

Die CSP enthält 'unsafe-inline' und 'unsafe-eval', weil Alpine.js sie für die Auswertung von Ausdrücken benötigt. Die Alternative ist der CSP-kompatible Build von Alpine.js, der Einschränkungen hat.14 Alle anderen Funktionen sind abgeriegelt: frame-ancestors verhindert Clickjacking, form-action beschränkt Formularübermittlungen auf denselben Ursprung, und upgrade-insecure-requests erzwingt HTTPS.

CDN-Cache-Sicherheit mit HTMX

Die Middleware für Security Headers fügt Vary: HX-Request zu HTMX-Antworten hinzu:

if request.headers.get("HX-Request"):
    existing_vary = response.headers.get("Vary", "")
    if "HX-Request" not in existing_vary:
        parts = [v.strip() for v in existing_vary.split(",") if v.strip()]
        parts.append("HX-Request")
        response.headers["Vary"] = ", ".join(parts)

Ohne diesen Header könnte ein CDN eine HTMX-Fragmentantwort cachen und sie bei einer Nicht-HTMX-Anfrage als vollständige Seite ausliefern (oder umgekehrt). Der Vary-Header weist das CDN an, getrennte Cache-Einträge basierend auf dem Wert des HX-Request-Headers zu speichern.11

CSRF-Schutz

HTMX-Formulare verwenden zustandslose, HMAC-signierte CSRF-Token:

# csrf.py
def generate_csrf_token() -> str:
    """Token format: timestamp:random:HMAC-SHA256-signature"""
    timestamp = str(int(time.time()))
    random_value = secrets.token_hex(16)
    payload = f"{timestamp}:{random_value}"
    signature = hmac.new(
        CSRF_SECRET.encode(), payload.encode(), hashlib.sha256
    ).hexdigest()
    return f"{payload}:{signature}"

def validate_csrf_token(token: str) -> bool:
    """Verify signature and check expiration (1 hour)."""
    timestamp, random_value, signature = token.split(":")
    if int(time.time()) - int(timestamp) > 3600:
        return False
    expected = hmac.new(
        CSRF_SECRET.encode(),
        f"{timestamp}:{random_value}".encode(),
        hashlib.sha256
    ).hexdigest()
    return hmac.compare_digest(expected, signature)

Das Token wird im Template über ein Jinja2-Global generiert und in HTMX-Formularanfragen eingebunden:

<form hx-post="/contact" hx-target="#form-result">
  <input type="hidden" name="csrf_token" value="{{ csrf_token() }}">
  <!-- form fields -->
</form>

Zustandslose Token machen serverseitige Session-Speicherung überflüssig. Die HMAC-Signatur stellt sicher, dass das Token vom Server generiert wurde. Der Zeitstempel verhindert Replay-Angriffe. hmac.compare_digest verhindert Timing-Angriffe.15

HTML-Sanitization

Benutzergenerierte Inhalte laufen vor dem Rendering durch nh3:

templates.env.filters["sanitize"] = sanitize_html
# In templates: {{ content | sanitize }}

Die Bibliothek nh3 entfernt Tags und Attribute, die nicht in der Allowlist stehen. Links erhalten automatisch rel="noopener noreferrer". Diese Abwehr ist unabhängig von CSP: Sie verhindert gespeichertes XSS auf der Rendering-Ebene, während CSP eingeschleuste Skripte auf Browser-Ebene verhindert. Defense in depth.

Eingabevalidierung

Pydantic-Modelle validieren alle Eingaben an der API-Grenze:

from pydantic import BaseModel, Field, EmailStr

class ContactRequest(BaseModel):
    name: str = Field(..., min_length=1, max_length=100)
    email: EmailStr
    message: str = Field(..., min_length=10, max_length=5000)

FastAPI gibt bei ungültigen Eingaben automatisch 422 Unprocessable Entity zurück. Zusammen mit parametrisierten Datenbankabfragen (SQLAlchemy interpoliert niemals Strings) verhindert dies SQL Injection und stellt Typsicherheit an den Grenzen sicher.


Performance

Lighthouse 100/100/100/100

blakecrosley.com erreicht 100 Punkte in allen vier Lighthouse-Kategorien: Performance, Accessibility, Best Practices und SEO. Überprüfen Sie das mit PageSpeed Insights.2

Die wichtigsten Optimierungen:

Ladestrategie für CSS

blakecrosley.com lädt CSS mit einem einzigen <link>-Tag und inhaltsgehashten URLs für unveränderliches Caching:

<link rel="stylesheet" href="{{ asset('css/styles.css') }}">

Der asset()-Helper hängt einen Content Hash (?v=a3b2c1d4) an, sodass der Browser die Datei unbegrenzt cached, bis sich der Inhalt ändert. Keine Extraktion von kritischem CSS, kein Print-Media-Trick, kein JavaScript-basiertes Laden. Die CSS-Datei ist gzip-komprimiert etwa 8 KB groß und damit klein genug, dass der Single-Request-Ansatz 100 Punkte bei Lighthouse Performance erreicht, ohne Optimierungsgymnastik.

GZip-Komprimierung

app.add_middleware(GZipMiddleware, minimum_size=500)

Antworten über 500 Byte werden komprimiert. HTML lässt sich um 70-80% komprimieren, wodurch ein 15-KB-Dokument auf 3-4 KB schrumpft.

Unveränderliches Caching statischer Assets

# In security headers middleware
if request.url.path.startswith("/static/"):
    if os.environ.get("RAILWAY_ENVIRONMENT"):
        response.headers["Cache-Control"] = "public, max-age=31536000, immutable"

Statische Assets mit inhaltsgehashten URLs (?v=a3f8b2c1d0) werden ein Jahr lang mit immutable gecached. Der Hash ändert sich, wenn sich die Datei ändert, und zwingt Browser und CDNs dadurch, die neue Version abzurufen.

Verzögertes Laden von Skripten

<script defer src="{{ asset('js/vendor/alpine.min.js') }}"></script>
<script defer src="{{ asset('js/vendor/htmx.min.js') }}"></script>
<script defer src="{{ asset('js/main.js') }}"></script>

Das Attribut defer lädt Skripte parallel zum Parsen von HTML herunter, führt sie aber erst aus, nachdem das Dokument geparst wurde. Dadurch wird Render-Blocking verhindert, ohne die Komplexität von async-Laden und Verwaltung der Ausführungsreihenfolge.

Bildoptimierung

Bilder verwenden WebP mit responsivem srcset und expliziten Abmessungen:

OPTIMIZED_IMAGES = {
    "vision-sprint": {
        "webp_srcset": (
            "/static/images/optimized/vision-sprint-400w.webp 400w, "
            "/static/images/optimized/vision-sprint-800w.webp 800w, "
            "/static/images/optimized/vision-sprint-1200w.webp 1200w"
        ),
        "fallback": "/static/images/optimized/vision-sprint-fallback.jpg",
        "width": 1200,
        "height": 1045,
    },
}
<picture>
  <source type="image/webp"
          srcset="{{ image.webp_srcset }}"
          sizes="(max-width: 768px) 100vw, 50vw">
  <img src="{{ image.fallback }}"
       width="{{ image.width }}"
       height="{{ image.height }}"
       alt="{{ image.alt }}"
       loading="lazy">
</picture>

Explizite Attribute width und height verhindern Cumulative Layout Shift (CLS). Das Attribut loading="lazy" verschiebt Offscreen-Bilder. WebP liefert bei gleichwertiger Qualität 25-35% kleinere Dateien als JPEG.16

Early Hints

# In main.py
app.state.preload_links = [
    f'<{make_asset_url(_asset_map, "css/styles.css")}>; rel=preload; as=style',
]

# In security headers middleware
if "text/html" in content_type:
    preload_links = getattr(request.app.state, "preload_links", [])
    if preload_links:
        response.headers["Link"] = ", ".join(preload_links)

Der Link-Header mit rel=preload weist Cloudflare an, eine 103-Early-Hints-Antwort zu senden, sodass der Browser bereits mit dem Abrufen von CSS beginnen kann, bevor der Server die HTML-Antwort fertig generiert hat.17

Minimales JavaScript

Der gesamte JavaScript-Footprint:

Bibliothek Größe (minifiziert + gzip-komprimiert)
HTMX ~16 KB
Alpine.js ~15 KB
Seitenspezifisches JS 4-8 KB
Gesamt 35-39 KB

Eine typische React-Anwendung liefert 100-300 KB Framework-JavaScript aus, noch bevor Anwendungscode hinzukommt.18 Der No-Build-Ansatz liefert weniger JavaScript aus, weil weniger JavaScript ausgeliefert werden muss.


Deployment

Railway

blakecrosley.com wird per git push auf Railway bereitgestellt:

# railway.toml
[build]
builder = "nixpacks"

[deploy]
startCommand = "uvicorn app.main:app --host 0.0.0.0 --port ${PORT:-8000}"
healthcheckPath = "/health"
healthcheckTimeout = 300
restartPolicyType = "ON_FAILURE"
restartPolicyMaxRetries = 10

Der Nixpacks-Builder von Railway erkennt das Python-Projekt anhand von requirements.txt, installiert Abhängigkeiten und führt den Startbefehl aus. Keine Dockerfile erforderlich. Der Health-Check-Endpunkt stellt sicher, dass die Anwendung reagiert, bevor sie Traffic erhält:

@app.get("/health")
async def health():
    return {"status": "healthy"}

Die Deploy-Pipeline

git push origin main
  → Railway detects push
  → Nixpacks installs Python + requirements.txt (cached)
  → uvicorn starts
  → Health check passes
  → Traffic routes to new deployment
  → ~40 seconds total

Kein npm install. Kein npm run build. Keine webpack-Kompilierung. Keine TypeScript-Kompilierung. Der einzige Installationsschritt ist pip install -r requirements.txt, der zwischen Deployments zwischengespeichert wird.

Procfile

web: uvicorn app.main:app --host 0.0.0.0 --port ${PORT:-8000}

Das Procfile bietet eine Heroku-kompatible Alternative. Railway unterstützt sowohl railway.toml als auch Procfile. Die Syntax ${PORT:-8000} verwendet den von der Plattform bereitgestellten Port oder fällt für die lokale Entwicklung auf 8000 zurück.

Uvicorn-Produktionskonfiguration

Für Deployments mit höherem Traffic verwenden Sie mehrere Worker:

uvicorn app.main:app \
  --host 0.0.0.0 \
  --port ${PORT:-8000} \
  --workers 4 \
  --loop uvloop \
  --http httptools
  • --workers 4 führt vier Worker-Prozesse aus (Faustregel: 2 * CPU-Kerne + 1)
  • --loop uvloop verwendet den schnelleren uvloop-Event-Loop (Drop-in-Ersatz für asyncio)
  • --http httptools verwendet den schnelleren httptools-HTTP-Parser

Für die Entwicklung überwacht --reload Dateiänderungen:

uvicorn app.main:app --reload --port 8000

Docker-Alternative

Für Plattformen, die Docker erfordern:

FROM python:3.11-slim

WORKDIR /app
COPY requirements.txt .
RUN pip install --no-cache-dir -r requirements.txt

COPY . .

EXPOSE 8000
CMD ["uvicorn", "app.main:app", "--host", "0.0.0.0", "--port", "8000"]

Das schlanke Basis-Image hält den Container klein. --no-cache-dir verhindert, dass pip heruntergeladene Pakete im Image-Layer speichert.

Cloudflare CDN

blakecrosley.com nutzt Cloudflare für CDN-Caching, DNS und Workers:

# Cache headers for HTML pages (set in security middleware)
response.headers["Cache-Control"] = (
    "public, max-age=300, s-maxage=3600, "
    "stale-while-revalidate=86400"
)
  • max-age=300 — Browser cachen 5 Minuten lang
  • s-maxage=3600 — CDN cached 1 Stunde lang
  • stale-while-revalidate=86400 — veraltete Inhalte 24 Stunden lang ausliefern, während im Hintergrund revalidiert wird

Statische Assets erhalten max-age=31536000, immutable, weil Content-Hash-URLs Aktualität garantieren.


Entscheidungsframework

Brauchen Sie Build Tools?

Beantworten Sie vier Fragen:

1. Teilen sich mehr als fünf Entwickler JavaScript-Schnittstellen? Falls ja, verhindert die Compile-Time-Typprüfung von TypeScript Integrationsfehler, die Runtime-Tests zu spät finden. Fügen Sie einen Build-Schritt hinzu.

2. Verwaltet Ihre Anwendung komplexen clientseitigen Zustand? Wenn Drag-and-drop, Echtzeitkollaboration oder Offline-first-Daten Kernfunktionen sind (nicht nur Nice-to-haves), rechtfertigt ein Framework wie React oder Svelte seine Komplexität. Fügen Sie einen Build-Schritt hinzu.

3. Nutzen mehrere Produkte eine gemeinsame Komponentenbibliothek? Falls ja, braucht diese Bibliothek npm-Packaging, semantische Versionierung und Tree Shaking. Fügen Sie einen Build-Schritt hinzu.

4. Sind Sie von npm-Ökosystembibliotheken abhängig, die einen Bundler voraussetzen? Wenn Radix, Framer Motion, TanStack Query oder ähnliche Bibliotheken zentral für das Produkt sind, ist eine Build-Pipeline Pflicht.

Wenn alle vier Antworten „nein“ lauten, ist der No-Build-Ansatz tragfähig. Wenn eine Antwort „ja“ lautet, lösen Build Tools ein echtes Problem. Der Fehler besteht darin, Build Tools hinzuzufügen, obwohl alle vier Antworten „nein“ lauten — Sie lösen Probleme, die Sie nicht haben, und schaffen gleichzeitig Overhead beim Abhängigkeitsmanagement.1

Stack-Vergleich

Kategorie No-Build (dieser Guide) React + Build Tools
Am besten geeignet für Content-Websites, Portfolios, interne Tools, CRUD-Apps SaaS-Produkte, komplexe SPAs, Designsystem-Konsumenten
Teamgröße 1-5 Entwickler 5-50+ Entwickler
State Management Server (HTMX) + Client (Alpine.js) Client (React state, Redux, Zustand)
Typsicherheit Runtime (Pydantic serverseitig) Compile-Time (TypeScript)
Komponentenwiederverwendung Jinja2 includes + macros npm-Pakete, gemeinsame Bibliotheken
SEO Standardmäßig servergerendert Erfordert SSR/SSG-Konfiguration
Performance-Untergrenze Hoch (minimales JS, servergerendert) Variiert (Framework-Overhead)
Komplexitätsobergrenze Niedriger (kein Offline, kein umfangreicher Client-Zustand) Höher (jede Client-Interaktion möglich)
Abhängigkeiten 17 Python-Pakete 300+ npm-Pakete
Build-Zeit 0 Sekunden 15-60 Sekunden

Wann HTMX falsch ist

HTMX ersetzt Client-Zustand durch Server-Roundtrips. Das funktioniert, bis Latenz relevant wird:

  • Drag-and-drop-Schnittstellen — 200 ms Server-Roundtrip pro Drag-Event sind inakzeptabel
  • Echtzeitkollaboration — WebSocket-gesteuerter Zustand erfordert clientseitige Konfliktauflösung
  • Offline-first-Anwendungen — kein Server bedeutet kein HTMX
  • Komplexe Animationen, die an Zustand gekoppelt sind — Framer Motion und React Spring setzen ein React-Reconciliation-Modell voraus
  • Canvas/WebGL-Anwendungen — die Rendering-Schleife ist von Natur aus clientseitig

Für diese Anwendungsfälle ist ein clientseitiges Framework das richtige Werkzeug. Der No-Build-Ansatz versucht nicht, sie zu ersetzen.


Schnellreferenzkarte

FastAPI

# Development
source venv/bin/activate
uvicorn app.main:app --reload --port 8000

# Production
uvicorn app.main:app --host 0.0.0.0 --port ${PORT:-8000}

# Testing
python -m pytest -v --cov=app

# Database migrations
alembic upgrade head
alembic revision --autogenerate -m "description"

HTMX-Attribute

hx-get="/url"                     <!-- GET request -->
hx-post="/url"                    <!-- POST request -->
hx-target="#element"              <!-- Where to put response -->
hx-swap="innerHTML"               <!-- How to insert (innerHTML, outerHTML, beforeend) -->
hx-trigger="click"                <!-- What triggers request -->
hx-trigger="keyup changed delay:300ms"  <!-- Debounced input -->
hx-trigger="load"                 <!-- Fire on element load -->
hx-indicator="#spinner"           <!-- Show during request -->
hx-push-url="true"                <!-- Update browser URL -->
hx-replace-url="true"             <!-- Replace URL (no history) -->

Alpine.js-Attribute

x-data="{ open: false }"         <!-- Component scope + state -->
x-show="open"                    <!-- Toggle visibility -->
x-cloak                          <!-- Hide until Alpine inits -->
@click="open = !open"            <!-- Event handler -->
@click.away="open = false"       <!-- Outside click -->
@keydown.escape="open = false"   <!-- Keyboard event -->
:class="{ 'active': open }"      <!-- Dynamic class -->
:aria-expanded="open"            <!-- Dynamic attribute -->
x-text="count"                   <!-- Dynamic text content -->
x-init="fetchData()"             <!-- Run on init -->

CSS Custom Properties

:root {
  --color-bg:     #000000;
  --color-text:   #ffffff;
  --spacing-sm:   1rem;
  --spacing-md:   1.5rem;
  --font-size-lg: 1.25rem;
}
@media (max-width: 768px) {
  :root { --gutter: 24px; }
}

Security Headers

Content-Security-Policy: default-src 'self'; script-src 'self' 'unsafe-eval'
Strict-Transport-Security: max-age=31536000; includeSubDomains
X-Content-Type-Options: nosniff
X-Frame-Options: SAMEORIGIN
Referrer-Policy: strict-origin-when-cross-origin
Cross-Origin-Opener-Policy: same-origin
Permissions-Policy: camera=(), microphone=(), geolocation=()

Projekt-Setup-Checkliste

[ ] FastAPI app with Jinja2Templates
[ ] Security headers middleware (CSP, HSTS, X-Frame-Options)
[ ] CSRF token generation and validation
[ ] GZip middleware (minimum_size=500)
[ ] Content-hash asset versioning (cache busting)
[ ] HTMX self-hosted in /static/js/vendor/
[ ] Alpine.js self-hosted in /static/js/vendor/
[ ] CSS custom properties for design tokens
[ ] Health check endpoint (/health)
[ ] Error handlers (404, 500)
[ ] robots.txt, sitemap.xml, llms.txt
[ ] JSON-LD structured data in base template
[ ] Hreflang tags for i18n (if multi-language)
[ ] HTML sanitization filter (nh3)
[ ] Rate limiting middleware
[ ] Deferred script loading

FAQ

Ist HTMX produktionsreif für echte Webanwendungen?

Ja. HTMX ist seit 2020 stabil und wird in mehreren Branchen produktiv eingesetzt. Carson Gross, der Entwickler, pflegt Abwärtskompatibilität als zentrales Designprinzip — in der HTMX-Dokumentation steht, dass die Library bestehende Anwendungen innerhalb einer Hauptversion nicht beschädigen wird.19 Die Library ist minifiziert und gzip-komprimiert etwa 16 KB groß, hat keine Abhängigkeiten und folgt semantischer Versionierung. blakecrosley.com nutzt HTMX seit drei Jahren in Produktion, ohne einen einzigen HTMX-bezogenen Fehler.20

Kann ich TypeScript ohne Build-Schritt verwenden?

Teilweise. TypeScript-Dateien können mit tsc --noEmit typgeprüft werden, ohne Ausgabedateien zu erzeugen; damit dient die Prüfung zur Compile-Zeit als Linter. Browser können .ts-Dateien allerdings nicht direkt ausführen, daher ist zum Ausliefern von TypeScript weiterhin ein Build-Schritt erforderlich. Die Alternative sind JSDoc-Typannotationen in einfachen .js-Dateien, die TypeScript ohne Kompilierung prüfen kann. So erhalten Sie Typsicherheit während der Entwicklung und liefern trotzdem standardmäßiges JavaScript aus.

Wie ist dieser Ansatz im Vergleich zu Astro oder 11ty?

Astro und 11ty sind Static-Site-Generatoren, die einfaches HTML mit minimalem clientseitigem JavaScript erzeugen, aber einen Build-Schritt benötigen (Node.js, npm install, Build-Befehl). Der No-Build-Ansatz entfernt diesen Schritt — der Server rendert HTML bei jeder Anfrage. Der Kompromiss: Astro/11ty erzeugen schnellere statische Seiten (keine Serverberechnung), während FastAPI + HTMX dynamische Inhalte nativ verarbeitet (benutzerspezifische Daten, Formularübermittlungen, Echtzeitaktualisierungen), ohne eine separate API-Schicht.

Was ist mit serverseitigem Rendering (SSR) mit React?

Next.js SSR und der Ansatz mit FastAPI + HTMX verfolgen dasselbe Ziel: servergerendertes HTML an den Browser senden. Der Unterschied liegt darin, was nach dem ersten Rendering passiert. Next.js hydratisiert die Seite mit React und liefert dabei die Framework-Runtime sowie den Komponenten-Code an den Client aus. FastAPI + HTMX hydratisiert nicht — das HTML ist die endgültige Ausgabe. HTMX verarbeitet anschließende Interaktionen, indem neue HTML-Fragmente vom Server angefordert werden. Das Ergebnis: FastAPI + HTMX liefert insgesamt ungefähr 35-40 KB JavaScript aus, gegenüber 100-300 KB bei einer Next.js-Anwendung.18

Wie gehe ich mit Formularvalidierung in diesem Stack um?

Serverseitig. Pydantic validiert die Eingabe, wenn das Formular übermittelt wird. Schlägt die Validierung fehl, gibt der Server das Formular mit Fehlermeldungen zurück. HTMX tauscht die Antwort in den DOM ein:

<form hx-post="/contact" hx-target="#form-container" hx-swap="outerHTML">
  <input type="email" name="email" required>
  <button type="submit">Send</button>
</form>
@router.post("/contact")
async def contact(request: Request, email: str = Form(...)):
    if not validate_email(email):
        return templates.TemplateResponse("components/_contact_form.html", {
            "request": request,
            "error": "Please enter a valid email address",
            "email": email,  # Preserve input
        })
    await send_email(email)
    return templates.TemplateResponse("components/_contact_success.html", {
        "request": request
    })

Der Server validiert, der Server rendert Fehlerzustände, und HTMX tauscht das Ergebnis ein. Eine clientseitige Validierungs-Library ist nicht nötig. Das HTML-Attribut required bietet als erste Verteidigungslinie eine grundlegende Validierung auf Browser-Ebene.

Kann ich Echtzeitfunktionen (WebSockets) hinzufügen?

Ja. FastAPI hat integrierte Unterstützung für WebSocket:

from fastapi import WebSocket

@app.websocket("/ws/notifications")
async def websocket_endpoint(websocket: WebSocket):
    await websocket.accept()
    while True:
        data = await get_notification()
        await websocket.send_text(render_notification_html(data))

HTMX hat eine WebSocket-Erweiterung (hx-ws), die Elemente mit WebSocket-Endpunkten verbindet:

<!-- HTMX 2.x WebSocket extension syntax -->
<div hx-ext="ws" ws-connect="/ws/notifications">
  <div id="notifications" ws-send></div>
</div>

Hinweis: HTMX 1.x verwendete die Syntax hx-ws="connect:...". In HTMX 2.x wurde die WebSocket-Unterstützung in eine separate Erweiterung (htmx-ext-ws) verschoben, mit den oben gezeigten Attributen ws-connect und ws-send. Wenn Sie HTMX 1.x verwenden, funktioniert die alte hx-ws-Syntax weiterhin.

HTMX 4.0-Beta-Track: htmx 4.0.0-beta4 ist jetzt unter dem npm-Tag next und in der 4.0-Dokumentation verfügbar, während der Schnellstart auf htmx.org und der npm-Tag latest weiterhin bei 2.0.10 bleiben. Dieser Leitfaden zielt weiterhin auf HTMX 2.x, die bis zur Stabilisierung von 4.0 die empfohlene Version für Produktionsarbeit bleibt; die Migration von 2.x auf 4.x ist ein Generationssprung, kein 2.x-Point-Release. Das Versionierungsmuster von big-skies-software überspringt ungerade Hauptversionen, daher ist 4.0 der nächste Schritt nach 2.x.2122

Beobachtenswert aus der 4.0-Dokumentation. Zwei Ergänzungen fallen für Sicherheits- und Architekturprüfungen vor 4.0 GA besonders auf: Die neue Erweiterung hx-live führt DOM-reaktive Ausdrücke ein, die neu ausgewertet werden, wenn sich referenzierter Zustand ändert, und die neue Erweiterung hx-nonce sperrt die Verarbeitung von htmx-Attributen hinter CSP-Nonces. Der 4.0-Migrationsleitfaden verschiebt außerdem mehrere Konfigurationskonzepte, stellt bestimmtes Ereignis-/Verlaufsverhalten wieder her oder ändert es und entfernt einige JavaScript-Hilfsfunktionen aus dem Kern. Behandeln Sie 4.0 als Migrationsprojekt, nicht als austauschbaren 2.x-Patch.21

Nachrichten vom Server werden mit denselben Targeting- und Swap-Mechaniken wie HTTP-Antworten in den DOM eingetauscht. Der Server sendet HTML-Fragmente über den WebSocket, und HTMX fügt sie ein.

Wie handhabt dieser Stack SEO?

Servergerendertes HTML ist von Natur aus SEO-freundlich, weil Crawler den vollständigen Seiteninhalt erhalten, ohne JavaScript auszuführen. blakecrosley.com ergänzt mehrere SEO-Schichten:

  • JSON-LD-strukturierte Daten in <head> für jede Seite (Person-, Article-, WebSite-, FAQPage-Schemas)
  • Dynamische Sitemap mit hreflang-Alternativen für alle 10 Locales
  • RSS-Feed unter /blog/feed.xml
  • llms.txt im Root-Verzeichnis für Auffindbarkeit durch AI-Crawler
  • Kanonische URLs und Open Graph-Tags im Basistemplate
  • Semantisches HTML: <article>, <section>, <main>, korrekte Überschriftenhierarchie

Keine SSR-Konfiguration nötig. Kein getStaticProps. Kein ISR. Das HTML wird bei jeder Anfrage gerendert — das ist das Standardverhalten, keine Optimierung.

Wie steil ist die Lernkurve im Vergleich zu React?

Für Python-Entwickler ist die Lernkurve deutlich flacher. Sie kennen die Sprache bereits. Die Route-Handler von FastAPI geben Template-Antworten zurück — dasselbe mentale Modell wie bei Flask- oder Django-Views. HTMX ergänzt eine Handvoll HTML-Attribute (hx-get, hx-target, hx-swap). Alpine.js ergänzt einige weitere (x-data, x-show, @click). Es gibt kein JSX, kein virtuelles DOM, kein Hook-System, keine State-Management-Library und keine Build-Tool-Konfiguration, die Sie lernen müssten.

Die HTMX-Dokumentation passt auf eine einzelne lange Seite. Die Alpine.js-Dokumentation passt auf einige wenige Seiten. Die React-Dokumentation umfasst Hunderte Seiten zu Hooks, Context, Refs, Effects, Suspense, Server Components und Streaming SSR.

Für JavaScript-/React-Entwickler ist die Umstellung eher konzeptionell als syntaktisch. Die zentrale Erkenntnis lautet: Der Server besitzt den Zustand, und der Server rendert das HTML. Clientseitiges State Management wird zu serverseitigem Route Handling. Clientseitiges Data Fetching wird zu HTMX-Attributen auf HTML-Elementen. Die Syntax ist einfacher — das mentale Modell erfordert, die SPA-Annahme zu verlernen, dass der Client das Rendering besitzt.


Änderungsprotokoll

Datum Änderung
2026-06-22 FastAPI 0.138.0 + 0.137.2. 0.138.0 (20. Juni) ergänzt app.frontend("/", directory="dist") / router.frontend(...), um ein gebautes statisches Frontend bereitzustellen (SPA-Ausgabe dist/) — das steht quer zur No-Build-These dieses Guides mit serverseitigem Rendering und wird als Kontrast im Abschnitt zu Async Patterns erwähnt. 0.137.2 (18. Juni) ergänzt iter_route_contexts() als unterstützten Weg, Routen aufzuzählen, nachdem router.routes inzwischen intern ist (seit 0.137.0). Beides sind Funktionserweiterungen ohne Breaking Changes; Starlette (1.3.1), Pydantic (2.13.4), HTMX (2.0.10), Alpine.js (3.15.12), Bootstrap (5.3.8), SQLAlchemy (2.0.51) bleiben alle unverändert.
2026-06-16 FastAPI 0.137.0/0.137.1 + Starlette 1.0→1.3.1. FastAPI 0.137.0 (14. Juni) überarbeitet die Router-Interna: router.routes ist jetzt ein interner Baum und keine flache Liste von APIRoute mehr (Breaking Change für alles, was darüber iteriert), ermöglicht zugleich aber Routen, die nach include_router() hinzugefügt werden, sowie neue Hooks APIRouter.matches()/.handle(); 0.137.1 (15. Juni) behebt das Typing von APIRoute und prefixlose Router mit leerem Pfad. Starlette hat seine erste stabile Version 1.0 (22. März) veröffentlicht und ist jetzt bei 1.3.1 (12. Juni); dabei wurden die veralteten Hooks on_event/on_startup/on_shutdown und die Decorators @app.route()/@app.websocket_route() entfernt — lifespan und Route/WebSocketRoute sind die einzigen Wege; FastAPI 0.137.0 pinnt Starlette 1.3.1. Im Abschnitt zu Async Patterns wurde ein Hinweis zu Lifespan/Router ergänzt. SQLAlchemy 2.0.51 (15. Juni) enthält nur Bugfixes.
2026-06-08 SQLAlchemy 2.0.50 mit Änderung bei der Async-Installation. Ab SQLAlchemy 2.0.50 wird die greenlet-Abhängigkeit des Async-Stacks nicht mehr standardmäßig installiert — installieren Sie das Extra sqlalchemy[asyncio] (andernfalls schlägt das erste await gegen die Engine mit einem Missing-Greenlet-Fehler fehl). 2.0.50 erfordert außerdem Python 3.10+ (3.7–3.9 entfallen) und ergänzt Wheels für Free-Threaded 3.13t. Im Abschnitt zu SQLAlchemy 2.0 Async wurde ein Installationshinweis ergänzt. Keine Änderung am Haupttext für den Rest des Stacks: Die neueste Version von FastAPI ist weiterhin 0.136.3 (2026-05-23, kein Juni-Release), htmx stable bleibt 2.0.10 (4.0.0-beta4 „The Fetchening“ ist in der Beta mit einem stabilen Ziel grob Anfang 2027, noch keine Produktionsempfehlung), Alpine.js 3.15.12, Bootstrap 5.3.x unverändert. Produktionsempfehlung unverändert: HTMX 2.x verwenden, bis 4.0 stabil ist.23
2026-05-24 Wartungsprüfung: Die lokale Inhaltsinventur zeigt weiterhin 210 Blogbeiträge, 11 Kern-Guides, 48 Designstudien und 10 unterstützte Locales einschließlich Englisch. Die neueste Version von FastAPI ist 0.136.3 (2026-05-23); das einzige in den Release Notes genannte app-seitige Refactoring ist eine strengere Behandlung von Underscore-Headern, wenn convert_underscores=True gesetzt ist, und 0.136.2 validiert Server-Sent-Event-Felder, um defekte Event-Daten zu vermeiden. htmx stable bleibt 2.0.10, während npm next und die 4.0-Dokumentation jetzt auf 4.0.0-beta4 verweisen; die neueste Version von SQLAlchemy 2.0 ist 2.0.50; die neueste Version von Pydantic bleibt 2.13.4. Die Produktionsempfehlung bleibt unverändert: HTMX 2.x verwenden, bis 4.0 stabil ist.122
2026-05-18 Aktualisierung der Site-Inventur: Die lokale Inhaltsinventur zeigt jetzt 210 Blogbeiträge, 11 Kern-Guides, 48 Designstudien und 10 unterstützte Locales einschließlich Englisch. Die neueste Version von FastAPI bleibt 0.136.1; htmx stable bleibt 2.0.10 mit npm next auf 4.0.0-beta3; die neueste npm-Version von Alpine.js bleibt 3.15.12. Die Produktionsempfehlung bleibt unverändert: HTMX 2.x verwenden, bis 4.0 stabil ist.12021
2026-05-15 Wartungsprüfung: Die neueste Version von FastAPI bleibt 0.136.1; diese lokale Site-Umgebung importiert FastAPI 0.128.0 und Starlette 0.50.0; htmx stable bleibt 2.0.10 und npm next ist jetzt 4.0.0-beta3; die neueste npm-Version von Alpine.js ist 3.15.12; die neueste Version von Bootstrap ist 5.3.8; die neueste Version von SQLAlchemy 2.0 ist 2.0.49; die neueste Version von Pydantic ist 2.13.4. Produktionsempfehlung unverändert: HTMX 2.x verwenden, bis 4.0 stabil ist.2021
2026-05-09 Tracking von htmx 4.0.0-beta3 (8. Mai 2026): htmx 4.0.0-beta3 ist über den npm-Tag next und in der 4.0-Dokumentation verfügbar, während npm latest bei 2.0.10 bleibt. Punkte, die vor GA beachtet werden sollten: neue Erweiterung hx-live (DOM-reaktive Ausdrücke), neue Erweiterung hx-nonce (CSP-Nonce-Schutz für htmx-Attribute) und Änderungen im Migrationsleitfaden an Konfiguration, Verlauf, Events und zentralen JavaScript-Hilfsfunktionen. Produktionsempfehlung unverändert: htmx 2.x bleibt der neueste npm-Tag und die empfohlene Version, bis 4.0 GA erreicht.21
2026-05-07 Wartungsprüfung: Die neueste Version von FastAPI bleibt 0.136.1; htmx stable ist 2.0.10 und v4 bleibt Beta mit Ziel Sommer 2026; die neueste npm-Version von Alpine.js ist 3.15.12; die neueste Version von Bootstrap ist 5.3.8; die neueste Version von SQLAlchemy 2.0 ist 2.0.49; die neueste Version von Pydantic ist 2.13.4. Site-lokale Metriken wurden auf 182 Blogbeiträge, 11 Guides, zehn unterstützte Locales und 17 Python-Requirements aktualisiert. Migrationshinweis unverändert: Verwenden Sie HTMX 2.x für die Produktion, bis 4.0 stabil ist.20
2026-04-25 FastAPI 0.136.1 (23. April 2026): Bereinigung von Pydantic-v2-Deprecations (keine Verhaltensänderungen für App-Code). HTMX 4.0-Zeitplan erfasst: htmx 4.0.0-beta1 (6. April) und 4.0.0-beta2 (14. April) wurden veröffentlicht. Migrationshinweis unverändert — htmx 2.x bleibt auf dem neuesten npm-Tag, bis 4.0 stabil ist; Sicherheitsfixes laufen weiter, kein Upgrade-Druck. Wichtige 4.0-Änderungen, die Sie jetzt bei der Planung berücksichtigen sollten: (1) fetch() ersetzt XMLHttpRequest als zentrale Ajax-Infrastruktur, (2) Attributvererbung wird standardmäßig explizit, (3) History-Support stellt für wiederhergestellte Inhalte eine Netzwerkanfrage (kein lokaler DOM-Snapshot). FastAPI 0.135.4 (16. April) entfernte den April-Fools-Decorator @app.vibe(), der in 0.135.3 gelandet war.
2026-04-16 Bewusstsein für HTMX 4.0-beta ergänzt (Forward-Reference). Unterstützung von FastAPI 0.136.0 für Free-Threaded-Builds von Python 3.14t vermerkt. Funktionen von Pydantic 2.13.x (Default Factories für private Attribute mit Zugriff auf validierte Modelldaten, Namespace pydantic.v1 auf 1.10.26 mit 3.14-Unterstützung). Fixes in Alpine.js 3.15.11: Modifier x-anchor.noflip, Warnung bei mehreren Root-Elementen in x-for, Fix für $refs-Morph-Regression.
2026-03-24 Erstveröffentlichung

Quellen


Dieser Guide behandelt das vollständige System, mit dem blakecrosley.com erstellt wird. The No-Build Manifesto liefert das philosophische Argument. Der Beitrag Lighthouse Perfect Score dokumentiert den Weg der Performance-Optimierung. Der Beitrag Vibe Coding vs. Engineering untersucht, wo KI-gestützte Entwicklung in diesen Workflow passt.


  1. Produktionsmetriken von blakecrosley.com mit Stand vom 18. Mai 2026. Die Website umfasst 210 Blogbeiträge, interaktive JavaScript-Komponenten, 11 zentrale Guides, 48 Designstudien, Englisch plus 9 übersetzte Locales, minimale Python-Abhängigkeiten und keine Build-Tools. Verifiziert anhand des lokalen Content-Inventars, app/i18n/config.py und requirements.txt

  2. Google PageSpeed Insights (pagespeed.web.dev) führt Lighthouse-Audits für jede öffentliche URL aus. blakecrosley.com erreicht mit Stand März 2026 Werte von 100/100/100/100 (Performance, Barrierefreiheit, Best Practices, SEO). Die Ergebnisse sind öffentlich überprüfbar. Den vollständigen Optimierungsweg finden Sie in Von 76 auf 100: Einen perfekten Lighthouse-Score erreichen

  3. Ein frisches npx create-next-app@latest (Next.js 15, getestet im Februar 2026) installiert 311 Pakete in node_modules/ mit insgesamt 187 MB. Produktionsprojekte mit zusätzlichen Abhängigkeiten liegen tendenziell höher. Einzelne Projekte unterscheiden sich. Quelle: Tests des Autors, dokumentiert in The No-Build Manifesto

  4. Die Next.js-Performance-Dokumentation von Vercel empfiehlt bestimmte Optimierungen (Bildoptimierung, Font-Laden, Code-Splitting), um Scores über 90 zu erreichen. Siehe nextjs.org/docs/app/building-your-application/optimizing. Der Bereich 70-90 spiegelt Standardeinstellungen wider, bevor diese Optimierungen angewendet werden. 

  5. Vollständige Abhängigkeitsliste aus requirements.txt von blakecrosley.com mit Stand Mai 2026 verifiziert. Die Datei enthält derzeit 17 Python-Requirement-Einträge und keine Build-Tools, Compiler oder Bundler. 

  6. Basierend auf der Erfahrung des Autors mit der Wartung von Next.js-Projekten (2021-2024) erzeugt das JavaScript-Ökosystem bei aktiven Projekten 15-25 Dependabot-PRs pro Monat, meist für Aktualisierungen transitiver Abhängigkeiten, die der Entwickler nie direkt importiert hat. 

  7. Tim Berners-Lee formulierte Rückwärtskompatibilität als Webdesign-Prinzip: „a browser should be backwards-compatible.“ Eine Seite von 1996 wird in Chrome 2026 gerendert. Siehe w3.org/DesignIssues/Principles

  8. OWASP empfiehlt, API-Dokumentationsendpunkte in der Produktion zu deaktivieren, um die Angriffsfläche zu reduzieren. Der Endpunkt /openapi.json legt alle Routendefinitionen, Parameter und Response-Modelle offen. 

  9. FastAPI-Dokumentation zu async- vs. sync-Handlern: fastapi.tiangolo.com/async/. Wenn Sie in async-Funktionen await mit blockierenden Aufrufen mischen, wird der Event-Loop ausgehungert. 

  10. nh3 ist ein Rust-basierter HTML-Sanitizer und der Nachfolger der Bleach-Bibliothek. Er wird vom PyO3-Projekt gepflegt und bietet Allowlist-basierte HTML-Sanitization. Siehe github.com/messense/nh3

  11. Der Vary-Header ist in RFC 9110 Abschnitt 12.5.5 definiert. Er weist Caches an, separate Responses auf Basis der angegebenen Request-Header-Werte zu speichern. Ohne Vary: HX-Request könnte ein CDN ein HTMX-Fragment als vollständige Seitenantwort ausliefern. Siehe httpwg.org/specs/rfc9110.html#field.vary

  12. CSS Custom Properties (CSS Variables) werden von mehr als 97 % der globalen Browser unterstützt. Sie kaskadieren, werden vererbt und reagieren zur Laufzeit auf Media Queries — Fähigkeiten, die Präprozessor-Variablen fehlen. Quelle: caniuse.com/css-variables

  13. Googles hreflang-Dokumentation: developers.google.com/search/docs/specialty/international/localized-versions. Der Wert x-default kennzeichnet die Fallback-Seite für Benutzer, deren Sprache nicht in der hreflang-Liste enthalten ist. 

  14. Alpine.js erfordert 'unsafe-eval' in der Content Security Policy für seine Engine zur Ausdrucksauswertung. Der CSP-kompatible Build (@alpinejs/csp) vermeidet diese Anforderung, hat aber Einschränkungen. Siehe alpinejs.dev/advanced/csp

  15. HMAC-basierte CSRF-Tokens folgen dem Muster „Signed Double-Submit Cookie“, das im OWASP CSRF Prevention Cheat Sheet beschrieben wird. hmac.compare_digest nutzt einen Constant-Time-Vergleich, um Timing-Seitenkanalangriffe zu verhindern. Siehe cheatsheetseries.owasp.org/cheatsheets/Cross-Site_Request_Forgery_Prevention_Cheat_Sheet.html

  16. WebP bietet bei gleichwertiger visueller Qualität 25-35 % kleinere Dateien als JPEG. Googles WebP-Studie: developers.google.com/speed/webp/docs/webp_study

  17. 103 Early Hints ermöglicht es dem Server (oder CDN), eine vorläufige Response mit Preload-Hinweisen zu senden, bevor die endgültige Response bereit ist. Cloudflare unterstützt Early Hints für Link-Header mit rel=preload. Siehe developer.chrome.com/blog/early-hints

  18. React 18 + ReactDOM wiegt ungefähr 42 KB minified + gzipped. Mit Router, State-Management-Bibliothek und Build-Framework-Runtime liefern typische React-Anwendungen 100-300 KB Framework-JavaScript aus. Quelle: bundlephobia.com/package/[email protected]

  19. Die Versionierungsrichtlinie und das Rückwärtskompatibilitätsversprechen von HTMX sind unter htmx.org/migration-guide-htmx-1/ dokumentiert. Carson Gross hat das Prinzip der Rückwärtskompatibilität in Hypermedia Systems (2023) von Gross, Stepinski und Cotter formuliert: hypermedia.systems

  20. Wartungsprüfung vom 15. Mai 2026. FastAPI PyPI und Release Notes listen 0.136.1; die lokale Importverifizierung ergab FastAPI 0.128.0 und Starlette 0.50.0 für diese Website-Umgebung; htmx.org listet 2.0.10 im Quick Start; npm view htmx.org version dist-tags gab latest=2.0.10 und next=4.0.0-beta3 zurück; npm view alpinejs version und npm view @alpinejs/csp version gaben 3.15.12 zurück; der offizielle Bootstrap-Blog und npm-Paketmetadaten listen 5.3.8; SQLAlchemy PyPI und Dokumentation listen 2.0.49; Pydantic PyPI listet 2.13.4. 

  21. Die Paketmetadaten von htmx 4.0.0-beta3 nannten eine Veröffentlichung am 8. Mai 2026, und npm next zeigte auf 4.0.0-beta3; npm latest blieb bei 2.0.10. Die 4.0-Dokumentation unter four.htmx.org zeigte [email protected], der 4.0 Extensions Index listete hx-live und hx-nonce, und der 4.0 Migration Guide dokumentierte Migrationsänderungen, die vor dem Umstieg von Produktions-Apps von 2.x geprüft werden sollten. Für das Tracking der neuesten Linie durch 22 abgelöst. 

  22. Wartungsprüfung vom 24. Mai 2026. Lokale Inventarbefehle gaben 210 Markdown-Blogbeiträge, 11 Guide-Dateien auf oberster Ebene und 48 Designstudien-Dateien zurück. FastAPI Release Notes listen 0.136.3 vom 2026-05-23 mit strengerer Underscore-Header-Behandlung, wenn convert_underscores=True; 0.136.2 validiert Server-Sent-Event-Felder. python3 -m pip index versions fastapi gab als neueste Version 0.136.3 zurück; python3 -m pip index versions sqlalchemy gab als neueste Version 2.0.50 zurück; python3 -m pip index versions pydantic gab als neueste Version 2.13.4 zurück. npm view htmx.org dist-tags version time.modified --json gab latest=2.0.10, next=4.0.0-beta4 und time.modified=2026-05-22T15:56:21.948Z zurück; die Installationsdokumentation von four.htmx.org zeigt [email protected]

  23. SQLAlchemy 2.0.50 Changelog und Release-Blog, veröffentlicht am 2026-05-24. Die asyncio-greenlet-Abhängigkeit wird nicht mehr standardmäßig installiert; das Installationsziel sqlalchemy[asyncio] ist jetzt erforderlich, um sie einzubeziehen. 2.0.50 entfernt außerdem Python 3.7/3.8/3.9 (jetzt 3.10+), ergänzt Free-Threaded-Python-Wheels und fügt einen over(..., exclude=...)-Window-Frame-Parameter hinzu. Als neueste Version auf PyPI mit Stand 2026-06-08 verifiziert. htmx 4.0.0-beta4 („The Fetchening“, 2026-05-22) bleibt Beta mit einem stabilen Ziel Anfang 2027; FastAPI 0.136.3 (2026-05-23), Alpine.js 3.15.12 und Bootstrap 5.3.x sind in diesem Zeitraum unverändert. 

  24. FastAPI Release Notes: 0.137.0 (2026-06-14) refaktoriert Router-Interna, sodass router.routes keine flache Liste von APIRoute-Objekten mehr ist, sondern ein Baum aus Zwischenobjekten (als intern behandeln); außerdem ermöglicht es das Hinzufügen von Routen nach include_router(), einschließlich eines Sub-Routers, bevor dessen Routen definiert sind, vermeidet das Kopieren von Routen und fügt APIRouter.matches()/.handle() hinzu; pinnt Starlette 1.3.1. 0.137.1 (2026-06-15) behebt APIRoute-Typing und einen leeren Pfad in einem Router ohne Präfix. Starlette Release Notes: 1.0.0 (2026-03-22), die erste stabile Version seit rund 8 Jahren, entfernte on_startup/on_shutdown/on_event() und die Decorators @app.route()/@app.websocket_route() (verwenden Sie lifespan und Route/WebSocketRoute); neueste Version ist 1.3.1 (2026-06-12). SQLAlchemy 2.0.51 (Changelog, 2026-06-15) enthält nur Bugfixes ohne Auswirkungen auf async oder Installation. Verifiziert über PyPI und offizielle Release Notes am 2026-06-16. 

  25. FastAPI Release Notes: 0.138.0 (2026-06-20) fügt app.frontend("/", directory="dist") und router.frontend("/", directory="dist") zum Ausliefern eines gebauten statischen Frontends hinzu (PR #15800; Frontend-Dokumentation) — eine statische dist/-SPA-Serving-Funktion, kein serverseitig gerendertes Muster; keine Breaking Change. 0.137.2 (2026-06-18) fügt iter_route_contexts() für fortgeschrittene Nutzung hinzu, bei der zuvor router.routes durchlaufen wurde (seit 0.137.0 intern); keine Breaking Change. Keine neuere Version als 0.138.0 mit Stand 2026-06-22. Starlette 1.3.1, Pydantic 2.13.4, Uvicorn 0.49.0, SQLAlchemy 2.0.51, HTMX 2.0.10, Alpine.js 3.15.12, Bootstrap 5.3.8 alle unverändert. Verifiziert über PyPI und offizielle Release Notes am 2026-06-22. 

NORMAL fastapi-htmx.md EOF