# joerg-lohrer.de Persönliche Webseite. Nach einer Transition von einer Hugo-basierten, statischen Seite läuft `joerg-lohrer.de` jetzt als SvelteKit-SPA, die Blog-Posts live aus signierten Nostr-Events (NIP-23, `kind:30023`) rendert. ## Aktueller Stand - **`https://joerg-lohrer.de/`** — SvelteKit-SPA, seit 2026-04-18 live. Seit 2026-04-21 **multilingual** (Deutsch + Englisch via NIP-32 `l`-Tag und NIP-33-`a`-Tag-Verlinkung). - **`https://staging.joerg-lohrer.de/`** — Staging (gleicher Build, ein Schritt vor Prod). - **`https://svelte.joerg-lohrer.de/`** — Entwicklungs-Deploy-Target der Pipeline. - **`https://spa.joerg-lohrer.de/`** — Vanilla-HTML-Mini-Spike (Proof of Concept, historisch). Detailliert in [`docs/STATUS.md`](docs/STATUS.md). ## Wie die Seite funktioniert 1. **Inhalte** liegen als Markdown in `content/posts///index.md` (z. B. `content/posts/de//` oder `content/posts/en//`) mit strukturierten Bild-Metadaten im Frontmatter (Alt-Text, Lizenz, Autor:innen). Übersetzungen eines Posts werden über bidirektionale `a:`-Tags im Frontmatter verlinkt — Details in [`docs/superpowers/specs/2026-04-21-multilingual-posts-design.md`](docs/superpowers/specs/2026-04-21-multilingual-posts-design.md). 2. **Publish-Pipeline** (`publish/`, Deno) lädt Bilder auf Blossom-Server (content-addressed) und publiziert signierte `kind:30023`-Events via NIP-46-Bunker (Amber) auf 5 Relays — inkl. NIP-32 `l`-Tag (Sprache) und NIP-33 `a`-Tag (Verlinkung zu anderssprachigen Varianten). 3. **SvelteKit-SPA** (`app/`) lädt diese Events zur Laufzeit und rendert Post-Liste + Detailseiten. UI-Chrome via `svelte-i18n` (DE/EN), Browser- Locale als Default, Listen nach aktivem Locale gefiltert. Keine Server-Komponente, Static-Hosting reicht. 4. **CI**: GitHub Actions triggert die Publish-Pipeline bei Push auf `main` (via Forgejo→GitHub Push-Mirror). Identität und Assets: - **Pubkey:** `npub1f7jar3qnu269uyx5p0e4v24hqxjnxysxudvujza2ur5ehltvdeqsly2fx9` - **NIP-05:** `joerglohrer@joerg-lohrer.de` (statisches `.well-known/nostr.json`) - **Blossom-Server:** `blossom.edufeed.org`, `blossom.primal.net` - **Relays:** `relay.damus.io`, `nos.lol`, `relay.primal.net`, `relay.tchncs.de`, `relay.edufeed.org` ## Navigation - 📍 **Stand und Live-URLs:** [`docs/STATUS.md`](docs/STATUS.md) - 🔜 **Wie es weitergeht:** [`docs/HANDOFF.md`](docs/HANDOFF.md) - 🤖 **Claude-Einstieg:** [`CLAUDE.md`](CLAUDE.md) (Agent-Konventionen, Deploy-Falle, Commit-Stil) - 📐 **SPA-Spec:** [`docs/superpowers/specs/2026-04-15-nostr-page-design.md`](docs/superpowers/specs/2026-04-15-nostr-page-design.md) - 📐 **Publish-Pipeline-Spec:** [`docs/superpowers/specs/2026-04-15-publish-pipeline-design.md`](docs/superpowers/specs/2026-04-15-publish-pipeline-design.md) - 📐 **Bild-Metadaten-Konvention:** [`docs/superpowers/specs/2026-04-16-image-metadata-convention.md`](docs/superpowers/specs/2026-04-16-image-metadata-convention.md) - 📐 **Multilinguale Posts:** [`docs/superpowers/specs/2026-04-21-multilingual-posts-design.md`](docs/superpowers/specs/2026-04-21-multilingual-posts-design.md) - 🛠 **SvelteKit-SPA-Plan:** [`docs/superpowers/plans/2026-04-15-spa-sveltekit.md`](docs/superpowers/plans/2026-04-15-spa-sveltekit.md) (35 Tasks, erledigt) - 🛠 **Publish-Pipeline-Plan:** [`docs/superpowers/plans/2026-04-16-publish-pipeline.md`](docs/superpowers/plans/2026-04-16-publish-pipeline.md) (24 Tasks, erledigt) - 🛠 **Multilingual 1/3 — Pipeline:** [`docs/superpowers/plans/2026-04-21-multilingual-posts-pipeline.md`](docs/superpowers/plans/2026-04-21-multilingual-posts-pipeline.md) (10 Tasks, erledigt) - 🛠 **Multilingual 2/3 — SPA-Auflösung:** [`docs/superpowers/plans/2026-04-21-multilingual-posts-spa.md`](docs/superpowers/plans/2026-04-21-multilingual-posts-spa.md) (8 Tasks, erledigt) - 🛠 **Multilingual 3/3 — UI-i18n:** [`docs/superpowers/plans/2026-04-21-multilingual-posts-i18n.md`](docs/superpowers/plans/2026-04-21-multilingual-posts-i18n.md) (11 Tasks, erledigt) - 🤖 **Claude-Workflow-Skill:** [`.claude/skills/joerglohrerde-workflow.md`](.claude/skills/joerglohrerde-workflow.md) ## Branches - **`main`** — kanonisch. Seit Cutover (2026-04-18) Produktions-Quelle. - **`spa`** — historischer SvelteKit-Arbeitszweig, inzwischen gemerged. - **`hugo-archive`** — eingefrorener Hugo-Zustand als Orphan-Branch. Rollback-Option über `git checkout hugo-archive && hugo build`. ## Repo-Struktur ``` content/posts/// Markdown-Posts pro Sprache (26× de, 1× en) content/impressum.md Statisches Impressum (wird von SPA geladen) app/ SvelteKit-SPA (Laufzeit-Renderer) src/lib/i18n/ UI-Lokalisierung (svelte-i18n + Messages) src/lib/nostr/ Relay-Loader, Translations-Resolving publish/ Deno-Publish-Pipeline (Blossom + Nostr) preview/spa-mini/ Vanilla-HTML-Mini-Spike (historische Referenz) scripts/deploy-svelte.sh FTPS-Deploy, Targets: svelte/staging/prod static/ Site-Assets (Favicons, Profilbild, .well-known/) docs/ Specs, Pläne, Status, Handoff, Wiki-Entwürfe .github/workflows/ GitHub-Actions CI (Publish-Pipeline-Trigger) .claude/ Claude-Code-Sessions (Transparenz) + Skills CLAUDE.md Einstiegspunkt für Claude-Sessions ``` ## Entwicklung ```sh # SPA lokal cd app && npm run dev # SPA testen cd app && npm run test:unit cd app && npm run test:e2e cd app && npm run check # Publish-Pipeline cd publish && deno task check # pre-flight cd publish && deno task publish --dry-run # Simulation cd publish && deno task publish # diff-modus echt cd publish && deno task publish --post # ein Post cd publish && deno task test # Tests # Deploy DEPLOY_TARGET=staging ./scripts/deploy-svelte.sh DEPLOY_TARGET=prod ./scripts/deploy-svelte.sh ``` ## Lizenz Inhalte: [CC0 1.0 Universal](https://creativecommons.org/publicdomain/zero/1.0/deed.de) (Namensnennung erwünscht, aber rechtlich nicht erforderlich), sofern nicht anders vermerkt. Drittinhalte sind beim jeweiligen Bild mit Autor:innen und Lizenz gekennzeichnet. Code: siehe [LICENSE](LICENSE).