Compare commits

8 Commits

Author SHA1 Message Date
denshooter
8397e5acf2 Merge dev into production
All checks were successful
CI / CD / test-build (push) Successful in 10m19s
CI / CD / deploy-dev (push) Has been skipped
CI / CD / deploy-production (push) Successful in 23s
2026-04-09 18:02:37 +02:00
denshooter
7b5fdbd611 refactor: remove snippets feature and n8n project detection
All checks were successful
CI / CD / test-build (push) Successful in 10m12s
CI / CD / deploy-dev (push) Successful in 1m22s
CI / CD / deploy-production (push) Has been skipped
- Remove snippets page, component, API route, Directus types, and setup script
- Remove snippets section from About.tsx (card, modal, state)
- Remove snippets link from 404 page, simplify layout
- Remove n8n Docker event and callback handler workflows (auto project detection)
- Remove Gitea runner setup and deploy scripts
2026-04-09 18:02:21 +02:00
denshooter
5bcaade558 Merge dev into production
All checks were successful
CI / CD / test-build (push) Successful in 10m14s
CI / CD / deploy-dev (push) Has been skipped
CI / CD / deploy-production (push) Successful in 23s
2026-04-09 17:23:29 +02:00
denshooter
8ff17c552b chore: update workflows, messages, and footer
All checks were successful
CI / CD / test-build (push) Successful in 10m19s
CI / CD / deploy-dev (push) Successful in 2m7s
CI / CD / deploy-production (push) Has been skipped
2026-04-09 17:22:56 +02:00
denshooter
a958008add fix: remove review truncation and show full reviews; fix telegram-cms workflow bugs
- ReadBooks.tsx: remove line-clamp-3, readMore button, and review modal
- Show full review text inline instead of truncated snippets
- Remove unused AnimatePresence, X import, selectedReview state
- Fix  typo in 6 handler nodes
- Fix Markdown/HTML mix (*text*</b> → <b>text</b>)
- Fix Switch condition syntax (.action → .action)
- Fix position collision (Review Info Handler)
- Hardcode Telegram bot token, fix response handling in Publish Handler
- Add AI-generated questions for .review flow (was .review HC_ID TEXT)
- New .answer command for submitting review answers
- Create/Refine Review: POST new translations if missing instead of skipping
- Remove all substring truncations from Telegram messages
2026-04-09 17:22:23 +02:00
denshooter
aee811309b fix: scroll to top on locale switch and remove dashes from hero text
All checks were successful
CI / CD / test-build (push) Successful in 10m15s
CI / CD / deploy-dev (push) Has been skipped
CI / CD / deploy-production (push) Successful in 1m53s
- HeaderClient: track locale prop changes with useRef and call
  window.scrollTo on switch to reliably reset scroll position
- messages/en.json + de.json: replace em dash with comma and remove
  hyphens from Self-Hoster/Full-Stack in hero description

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-03-08 14:37:56 +01:00
denshooter
48a29cd872 fix: pass locale explicitly to Hero and force-dynamic on locale-sensitive API routes
All checks were successful
CI / CD / test-build (push) Successful in 10m11s
CI / CD / deploy-dev (push) Has been skipped
CI / CD / deploy-production (push) Successful in 1m28s
- Hero.tsx: pass locale prop directly to getTranslations instead of
  relying on setRequestLocale async storage, which can be lost during
  Next.js RSC streaming
- book-reviews route: replace revalidate=300 with force-dynamic to
  prevent cached English responses being served to German locale requests
- content/page route: add runtime=nodejs and force-dynamic (was missing
  both, violating CLAUDE.md API route conventions)

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-03-08 13:43:26 +01:00
denshooter
c95fc3101b chore: merge branch 'dev' into 'production' (Release: Design Overhaul & Admin Redesign)
All checks were successful
CI / CD / test-build (push) Successful in 10m9s
CI / CD / deploy-dev (push) Has been skipped
CI / CD / deploy-production (push) Successful in 24s
2026-03-08 13:18:26 +01:00
36 changed files with 838 additions and 7156 deletions

View File

@@ -1,541 +0,0 @@
# 🚀 Telegram CMS - Complete Deployment Guide
**Für andere PCs / Fresh Install**
---
## 📋 Was du bekommst
Ein vollständiges Telegram-Bot-System zur Verwaltung deines DK0 Portfolios:
### ✨ Features
- **Dashboard** (`/start`) - Übersicht mit Draft-Zählern und Quick Actions
- **Listen** (`/list projects|books`) - Paginierte Listen mit Action-Buttons
- **Suche** (`/search <term>`) - Durchsucht Projekte & Bücher
- **Statistiken** (`/stats`) - Analytics Dashboard (Views, Kategorien, Ratings)
- **Vorschau** (`/preview<ID>`) - Zeigt EN + DE Übersetzungen
- **Publish** (`/publish<ID>`) - Veröffentlicht Items (auto-detect: Project/Book)
- **Delete** (`/delete<ID>`) - Löscht Items permanent
- **Delete Review** (`/deletereview<ID>`) - Löscht nur Review-Text
- **AI Review** (`.review <HC_ID> <RATING> <TEXT>`) - Generiert EN+DE Reviews via Gemini
### 🤖 Automatisierungen
- **Docker Events** - Erkennt neue Deployments, fragt ob AI Beschreibung generieren soll
- **Book Reviews** - AI generiert DE+EN Reviews aus deinem Input
- **Status API** - Spotify, Discord, WakaTime Integration (bereits vorhanden)
---
## 📦 Workflows zum Importieren
### 1. **ULTIMATE Telegram CMS** ⭐ (HAUPT-WORKFLOW)
**Datei:** `n8n-workflows/ULTIMATE-Telegram-CMS-COMPLETE.json`
**Beschreibung:**
- Zentraler Command Router für alle `/` Befehle
- Enthält alle Handler: Dashboard, List, Search, Stats, Preview, Publish, Delete, AI Reviews
- **Aktivieren:** Ja (Telegram Trigger)
**Credentials:**
- Telegram API: `DK0_Server` (ID: `ADurvy9EKUDzbDdq`)
- Directus Token: `RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB` (hardcoded in Nodes)
- OpenRouter API: `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`
---
### 2. **Docker Event Extended** (Optional, empfohlen)
**Datei:** `n8n-workflows/Docker Event (Extended).json`
**Beschreibung:**
- Reagiert auf Docker Webhooks (`https://n8n.dk0.dev/webhook/docker-event`)
- Erkennt eigene Projekte (`denshooter/dk0`) vs. CI/CD Container
- Holt letzten Commit + README von Gitea
- Fragt per Telegram-Button: Auto-generieren, Selbst beschreiben, Ignorieren
**Credentials:**
- Telegram API: `DK0_Server`
- Gitea Token: `gitea-token` (noch anzulegen!)
**Setup:**
1. Gitea Token erstellen: https://git.dk0.dev/user/settings/applications
- Name: `n8n-api`
- Permissions: ✅ `repo` (read)
2. In n8n: Credentials → New → HTTP Header Auth
- Name: `gitea-token`
- Header Name: `Authorization`
- Value: `token <YOUR_GITEA_TOKEN>`
---
### 3. **Docker Callback Handler** (Required if using Docker Events)
**Datei:** `n8n-workflows/Docker Event - Callback Handler.json`
**Beschreibung:**
- Verarbeitet Button-Klicks aus Docker Event Workflow
- Auto: Ruft AI (Gemini) mit Commit+README Context
- Manual: Fragt nach manueller Beschreibung
- Ignore: Bestätigt ignorieren
**Credentials:**
- Telegram API: `DK0_Server`
- OpenRouter API: (same as above)
---
### 4. **Book Review** (Legacy - kann ersetzt werden)
**Datei:** `n8n-workflows/Book Review.json`
**Status:** ⚠️ Wird von ULTIMATE CMS ersetzt (nutzt `.review` Command)
**Optional behalten falls:**
- Separate Webhook gewünscht
- Andere Trigger-Quelle (z.B. Hardcover API direkt)
---
### 5. **Reading / Finished Books** (Andere Features)
**Dateien:**
- `finishedBooks.json` - Hardcover finished books webhook
- `reading (1).json` - Currently reading books
**Status:** Optional, wenn du Hardcover Integration nutzt
---
## 🛠️ Schritt-für-Schritt Installation
### **Schritt 1: n8n Credentials prüfen**
Öffne n8n → Settings → Credentials
**Benötigt:**
| Name | Type | ID | Notes |
|------|------|-----|-------|
| `DK0_Server` | Telegram API | `ADurvy9EKUDzbDdq` | Telegram Bot Token |
| `gitea-token` | HTTP Header Auth | neu erstellen | Für Commit-Daten |
| OpenRouter | (hardcoded) | - | In Code Nodes |
---
### **Schritt 2: Workflows importieren**
1. **ULTIMATE Telegram CMS:**
```
n8n → Workflows → Import from File
→ Wähle: n8n-workflows/ULTIMATE-Telegram-CMS-COMPLETE.json
→ ✅ Activate Workflow
```
2. **Docker Event Extended:**
```
→ Wähle: n8n-workflows/Docker Event (Extended).json
→ Credentials mappen: DK0_Server + gitea-token
→ ✅ Activate Workflow
```
3. **Docker Callback Handler:**
```
→ Wähle: n8n-workflows/Docker Event - Callback Handler.json
→ Credentials mappen: DK0_Server
→ ✅ Activate Workflow
```
---
### **Schritt 3: Gitea Token erstellen**
1. Gehe zu: https://git.dk0.dev/user/settings/applications
2. **Generate New Token**
- Token Name: `n8n-api`
- Select Scopes: ✅ `repo` (Repository Read)
3. Kopiere Token: `<YOUR_TOKEN_HIER>`
4. In n8n:
```
Credentials → New → HTTP Header Auth
Name: gitea-token
Header Name: Authorization
Value: token <YOUR_TOKEN_HIER>
```
---
### **Schritt 4: Test Commands**
Öffne Telegram → DK0_Server Bot:
```bash
/start
# Expected: Dashboard mit Quick Stats + Buttons
/list projects
# Expected: Liste aller Draft Projekte
/stats
# Expected: Analytics Dashboard
/search nextjs
# Expected: Suchergebnisse
.review 427565 5 Great book about AI!
# Expected: AI generiert EN+DE Review, sendet Vorschau
```
---
## 🔧 Konfiguration anpassen
### Telegram Chat ID ändern
Aktuell: `145931600` (dein Telegram Account)
**Ändern in:**
1. Öffne Workflow: `ULTIMATE-Telegram-CMS-COMPLETE`
2. Suche Node: `Telegram Trigger`
3. Additional Fields → Chat ID → `<NEUE_CHAT_ID>`
**Chat ID herausfinden:**
```bash
curl https://api.telegram.org/bot<BOT_TOKEN>/getUpdates
# Schick dem Bot eine Nachricht, dann findest du in "chat":{"id":123456}
```
---
### Directus API Token ändern
Aktuell: `RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB`
**Ändern in allen Code Nodes:**
```javascript
// Suche nach:
"Authorization": "Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB"
// Ersetze mit:
"Authorization": "Bearer <NEUER_TOKEN>"
```
**Betroffene Nodes:**
- Dashboard Handler
- List Handler
- Search Handler
- Stats Handler
- Preview Handler
- Publish Handler
- Delete Handler
- Delete Review Handler
- Create Review Handler
---
### OpenRouter AI Model ändern
Aktuell: `google/gemini-2.0-flash-exp:free`
**Alternativen:**
- `google/gemini-2.5-flash` (besser, aber kostenpflichtig)
- `openrouter/free` (fallback)
- `anthropic/claude-3.5-sonnet` (premium)
**Ändern in:**
- Node: `Create Review Handler` (ULTIMATE CMS)
- Node: `Generate AI Description` (Docker Callback)
```javascript
// Suche:
"model": "google/gemini-2.0-flash-exp:free"
// Ersetze mit:
"model": "google/gemini-2.5-flash"
```
---
## 📊 Command Reference
### Basic Commands
| Command | Beschreibung | Beispiel |
|---------|--------------|----------|
| `/start` | Dashboard anzeigen | `/start` |
| `/list projects` | Alle Draft-Projekte | `/list projects` |
| `/list books` | Alle Draft-Bücher | `/list books` |
| `/search <term>` | Suche in Projekten & Büchern | `/search nextjs` |
| `/stats` | Statistiken anzeigen | `/stats` |
### Item Management
| Command | Beschreibung | Beispiel |
|---------|--------------|----------|
| `/preview<ID>` | Vorschau (EN+DE) | `/preview42` |
| `/publish<ID>` | Veröffentlichen (auto-detect) | `/publish42` |
| `/delete<ID>` | Löschen (auto-detect) | `/delete42` |
| `/deletereview<ID>` | Nur Review-Text löschen | `/deletereview42` |
### AI Review Creation
```bash
.review <HARDCOVER_ID> <RATING> <YOUR_REVIEW_TEXT>
# Beispiel:
.review 427565 5 Great book about AI and the future of work!
# Generiert:
# - EN Review (erweitert deinen Text)
# - DE Review (übersetzt + erweitert)
# - Setzt Rating auf 5/5
# - Erstellt Draft in Directus
# - Sendet Vorschau mit /publish Button
```
---
## 🐛 Troubleshooting
### "Item not found"
**Ursache:** ID existiert nicht in Directus
**Fix:**
```bash
# Prüfe in Directus:
https://cms.dk0.dev/admin/content/projects
https://cms.dk0.dev/admin/content/book_reviews
```
---
### "Error loading dashboard"
**Ursache:** Directus API nicht erreichbar oder Token falsch
**Fix:**
```bash
# Test Directus API:
curl -H "Authorization: Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB" \
https://cms.dk0.dev/items/projects?limit=1
# Expected: JSON mit Projekt-Daten
# Falls 401: Token abgelaufen/falsch
```
---
### AI Review schlägt fehl
**Ursache:** OpenRouter API Problem oder Model nicht verfügbar
**Fix:**
```bash
# Test OpenRouter:
curl -X POST https://openrouter.ai/api/v1/chat/completions \
-H "Authorization: Bearer sk-or-v1-..." \
-H "Content-Type: application/json" \
-d '{"model":"google/gemini-2.0-flash-exp:free","messages":[{"role":"user","content":"test"}]}'
# Falls 402: Credits aufgebraucht
# → Wechsel zu kostenpflichtigem Model
# → Oder nutze "openrouter/free"
```
---
### Telegram antwortet nicht
**Ursache:** Workflow nicht aktiviert oder Webhook Problem
**Fix:**
1. n8n → Workflows → ULTIMATE Telegram CMS → ✅ Active
2. Check Executions:
```
n8n → Executions → Filter by Workflow
→ Suche nach Fehlern (red icon)
```
3. Test Webhook manuell:
```bash
curl -X POST https://n8n.dk0.dev/webhook-test/telegram-cms-webhook-001 \
-H "Content-Type: application/json" \
-d '{"message":{"text":"/start","chat":{"id":145931600}}}'
```
---
### Docker Event erkennt keine Container
**Ursache:** Webhook wird nicht getriggert
**Fix:**
**1. Prüfe Docker Event Source:**
```bash
# Auf Server (wo Docker läuft):
docker events --filter 'event=start' --format '{{json .}}'
# Expected: JSON output bei neuen Containern
```
**2. Test Webhook manuell:**
```bash
curl -X POST https://n8n.dk0.dev/webhook/docker-event \
-H "Content-Type: application/json" \
-d '{
"container":"portfolio-dev",
"image":"denshooter/portfolio:latest",
"timestamp":"2026-04-02T10:00:00Z"
}'
# Expected: Telegram Nachricht mit Buttons
```
**3. Setup Docker Event Forwarder:**
Auf Server erstellen: `/opt/docker-event-forwarder.sh`
```bash
#!/bin/bash
docker events --filter 'event=start' --format '{{json .}}' | while read event; do
container=$(echo "$event" | jq -r '.Actor.Attributes.name')
image=$(echo "$event" | jq -r '.Actor.Attributes.image')
timestamp=$(echo "$event" | jq -r '.time')
curl -X POST https://n8n.dk0.dev/webhook/docker-event \
-H "Content-Type: application/json" \
-d "{\"container\":\"$container\",\"image\":\"$image\",\"timestamp\":\"$timestamp\"}"
done
```
Systemd Service: `/etc/systemd/system/docker-event-forwarder.service`
```ini
[Unit]
Description=Docker Event Forwarder to n8n
After=docker.service
Requires=docker.service
[Service]
ExecStart=/opt/docker-event-forwarder.sh
Restart=always
User=root
[Install]
WantedBy=multi-user.target
```
Aktivieren:
```bash
chmod +x /opt/docker-event-forwarder.sh
systemctl daemon-reload
systemctl enable docker-event-forwarder
systemctl start docker-event-forwarder
```
---
## 📝 Environment Variables (Optional)
Falls du Tokens nicht hardcoden willst, nutze n8n Environment Variables:
**In `.env` (n8n Docker):**
```env
DIRECTUS_TOKEN=RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB
OPENROUTER_API_KEY=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
TELEGRAM_CHAT_ID=145931600
```
**In Workflows nutzen:**
```javascript
// Statt:
"Authorization": "Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB"
// Nutze:
"Authorization": `Bearer ${process.env.DIRECTUS_TOKEN}`
```
---
## 🔄 Backup & Updates
### Workflows exportieren
```bash
# In n8n:
Workflows → ULTIMATE Telegram CMS → ... → Download
# Speichern als:
n8n-workflows/ULTIMATE-Telegram-CMS-COMPLETE-v2.json
```
### Git Push
```bash
cd /pfad/zum/portfolio
git add n8n-workflows/
git commit -m "chore: update telegram cms workflows"
git push origin telegram-cms-deployment
```
---
## 🚀 Production Checklist
- [ ] Alle Workflows importiert
- [ ] Credentials gemappt (DK0_Server, gitea-token)
- [ ] Gitea Token erstellt & getestet
- [ ] `/start` Command funktioniert
- [ ] `/list projects` zeigt Daten
- [ ] `/stats` zeigt Statistiken
- [ ] AI Review generiert Text (`.review` Test)
- [ ] Docker Event Webhook getestet
- [ ] Inline Buttons funktionieren
- [ ] Error Handling in n8n Executions geprüft
- [ ] Workflows in Git committed
---
## 📚 Weitere Dokumentation
- **System Architecture:** `docs/TELEGRAM_CMS_SYSTEM.md`
- **Workflow Details:** `n8n-workflows/ULTIMATE-Telegram-CMS-COMPLETE-README.md`
- **Quick Reference:** `n8n-workflows/QUICK-REFERENCE.md`
- **Testing Checklist:** `n8n-workflows/TESTING-CHECKLIST.md`
---
## 🎯 Quick Start (TL;DR)
```bash
# 1. Clone Repo
git clone <REPO_URL>
cd portfolio
# 2. Import Workflows
# → n8n UI → Import → Select:
# - ULTIMATE-Telegram-CMS-COMPLETE.json
# - Docker Event (Extended).json
# - Docker Event - Callback Handler.json
# 3. Create Gitea Token
# → https://git.dk0.dev/user/settings/applications
# → Name: n8n-api, Scope: repo
# → Copy token → n8n Credentials → HTTP Header Auth
# 4. Activate Workflows
# → n8n → Workflows → ✅ Active (alle 3)
# 5. Test
# → Telegram: /start
```
**Done!** 🎉
---
**Version:** 1.0.0
**Last Updated:** 2026-04-02
**Author:** Dennis Konkol
**Status:** ✅ Production Ready

View File

@@ -1,294 +0,0 @@
"use client";
import React, { useState, useEffect, useCallback, useMemo } from "react";
import { motion, AnimatePresence } from "framer-motion";
import { Snippet } from "@/lib/directus";
import { X, Copy, Check, ChevronLeft, ChevronRight, Search } from "lucide-react";
// Color-coded language badges using the liquid design palette
const LANG_STYLES: Record<string, { bg: string; text: string; label: string }> = {
typescript: { bg: "bg-liquid-lavender/40", text: "text-purple-700 dark:text-purple-300", label: "TS" },
ts: { bg: "bg-liquid-lavender/40", text: "text-purple-700 dark:text-purple-300", label: "TS" },
javascript: { bg: "bg-liquid-amber/40", text: "text-amber-700 dark:text-amber-300", label: "JS" },
js: { bg: "bg-liquid-amber/40", text: "text-amber-700 dark:text-amber-300", label: "JS" },
python: { bg: "bg-liquid-sky/40", text: "text-sky-700 dark:text-sky-300", label: "PY" },
bash: { bg: "bg-liquid-mint/40", text: "text-emerald-700 dark:text-emerald-300", label: "SH" },
shell: { bg: "bg-liquid-mint/40", text: "text-emerald-700 dark:text-emerald-300", label: "SH" },
sh: { bg: "bg-liquid-mint/40", text: "text-emerald-700 dark:text-emerald-300", label: "SH" },
dockerfile: { bg: "bg-liquid-blue/40", text: "text-blue-700 dark:text-blue-300", label: "🐳" },
docker: { bg: "bg-liquid-blue/40", text: "text-blue-700 dark:text-blue-300", label: "🐳" },
css: { bg: "bg-liquid-pink/40", text: "text-pink-700 dark:text-pink-300", label: "CSS" },
scss: { bg: "bg-liquid-pink/40", text: "text-pink-700 dark:text-pink-300", label: "SCSS" },
go: { bg: "bg-liquid-teal/40", text: "text-teal-700 dark:text-teal-300", label: "GO" },
rust: { bg: "bg-liquid-peach/40", text: "text-orange-700 dark:text-orange-300", label: "RS" },
yaml: { bg: "bg-liquid-lime/40", text: "text-lime-700 dark:text-lime-300", label: "YAML" },
json: { bg: "bg-liquid-lime/40", text: "text-lime-700 dark:text-lime-300", label: "JSON" },
sql: { bg: "bg-liquid-coral/40", text: "text-red-700 dark:text-red-300", label: "SQL" },
nginx: { bg: "bg-liquid-teal/40", text: "text-teal-700 dark:text-teal-300", label: "NGINX" },
};
function getLangStyle(language: string) {
return LANG_STYLES[language?.toLowerCase()] ?? {
bg: "bg-liquid-purple/30",
text: "text-purple-700 dark:text-purple-300",
label: language?.toUpperCase() || "CODE",
};
}
function CodePreview({ code }: { code: string }) {
const lines = code.split("\n").slice(0, 4);
return (
<pre className="mt-4 bg-stone-950/80 rounded-xl p-3 text-[11px] font-mono text-stone-400 overflow-hidden leading-relaxed border border-stone-800/60 select-none">
{lines.map((line, i) => (
<div key={i} className="truncate">{line || " "}</div>
))}
{code.split("\n").length > 4 && (
<div className="text-stone-600 text-[10px] mt-1"></div>
)}
</pre>
);
}
export default function SnippetsClient({ initialSnippets }: { initialSnippets: Snippet[] }) {
const [selectedSnippet, setSelectedSnippet] = useState<Snippet | null>(null);
const [copied, setCopied] = useState(false);
const [activeCategory, setActiveCategory] = useState<string>("All");
const [search, setSearch] = useState("");
// Derived data
const categories = useMemo(() => {
const cats = Array.from(new Set(initialSnippets.map((s) => s.category))).sort();
return ["All", ...cats];
}, [initialSnippets]);
const filtered = useMemo(() => {
const q = search.toLowerCase();
return initialSnippets.filter((s) => {
const matchCat = activeCategory === "All" || s.category === activeCategory;
const matchSearch =
!q ||
s.title.toLowerCase().includes(q) ||
s.description.toLowerCase().includes(q) ||
s.category.toLowerCase().includes(q) ||
s.language.toLowerCase().includes(q);
return matchCat && matchSearch;
});
}, [initialSnippets, activeCategory, search]);
// Language badge for the currently open modal
const modalLang = useMemo(
() => (selectedSnippet ? getLangStyle(selectedSnippet.language) : null),
[selectedSnippet]
);
// Keyboard nav: ESC + arrows
const handleKeyDown = useCallback(
(e: KeyboardEvent) => {
if (!selectedSnippet) return;
if (e.key === "Escape") {
setSelectedSnippet(null);
} else if (e.key === "ArrowRight" || e.key === "ArrowDown") {
const idx = filtered.findIndex((s) => s.id === selectedSnippet.id);
if (idx < filtered.length - 1) setSelectedSnippet(filtered[idx + 1]);
} else if (e.key === "ArrowLeft" || e.key === "ArrowUp") {
const idx = filtered.findIndex((s) => s.id === selectedSnippet.id);
if (idx > 0) setSelectedSnippet(filtered[idx - 1]);
}
},
[selectedSnippet, filtered]
);
useEffect(() => {
window.addEventListener("keydown", handleKeyDown);
return () => window.removeEventListener("keydown", handleKeyDown);
}, [handleKeyDown]);
const copyToClipboard = useCallback((code: string) => {
navigator.clipboard.writeText(code);
setCopied(true);
setTimeout(() => setCopied(false), 2000);
}, []);
const currentIndex = selectedSnippet
? filtered.findIndex((s) => s.id === selectedSnippet.id)
: -1;
return (
<>
{/* ── Filter & Search bar ── */}
<div className="flex flex-col sm:flex-row gap-4 mb-10">
{/* Search */}
<div className="relative flex-1 max-w-sm">
<Search size={14} className="absolute left-4 top-1/2 -translate-y-1/2 text-stone-400 pointer-events-none" />
<input
type="text"
placeholder="Search snippets…"
value={search}
onChange={(e) => setSearch(e.target.value)}
className="w-full pl-10 pr-4 py-2.5 bg-white dark:bg-stone-900 border border-stone-200 dark:border-stone-800 rounded-2xl text-sm text-stone-900 dark:text-stone-100 placeholder:text-stone-400 focus:outline-none focus:border-liquid-purple transition-colors"
/>
</div>
{/* Category chips */}
<div className="flex flex-wrap gap-2">
{categories.map((cat) => (
<button
key={cat}
onClick={() => setActiveCategory(cat)}
className={`px-4 py-2 rounded-2xl text-[11px] font-black uppercase tracking-widest border transition-all ${
activeCategory === cat
? "bg-stone-900 dark:bg-stone-50 text-white dark:text-stone-900 border-stone-900 dark:border-stone-50 shadow-md"
: "bg-white dark:bg-stone-900 text-stone-500 dark:text-stone-400 border-stone-200 dark:border-stone-800 hover:border-liquid-purple hover:text-liquid-purple"
}`}
>
{cat}
</button>
))}
</div>
</div>
{/* ── Empty state ── */}
{filtered.length === 0 && (
<p className="text-center text-stone-400 py-24 text-sm">
No snippets found{search ? ` for "${search}"` : ""}.
</p>
)}
{/* ── Snippet Grid ── */}
<div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-6 md:gap-8">
{filtered.map((s, i) => {
const lang = getLangStyle(s.language);
return (
<motion.button
key={s.id}
initial={{ opacity: 0, y: 20 }}
animate={{ opacity: 1, y: 0 }}
transition={{ delay: i * 0.04 }}
onClick={() => setSelectedSnippet(s)}
className="text-left bg-white dark:bg-stone-900 rounded-[2.5rem] p-8 border border-stone-200/60 dark:border-stone-800/60 shadow-sm hover:shadow-xl hover:border-liquid-purple/40 transition-all group flex flex-col"
>
{/* Header row: category + language badge */}
<div className="flex items-center justify-between mb-5">
<span className="text-[10px] font-black uppercase tracking-widest text-stone-400 group-hover:text-liquid-purple transition-colors">
{s.category}
</span>
{s.language && (
<span className={`px-2.5 py-1 rounded-lg text-[10px] font-black uppercase tracking-wider ${lang.bg} ${lang.text}`}>
{lang.label}
</span>
)}
</div>
{/* Title */}
<h3 className="text-xl font-black text-stone-900 dark:text-white uppercase tracking-tighter mb-2 group-hover:text-liquid-purple transition-colors leading-tight">
{s.title}
</h3>
{/* Description */}
<p className="text-stone-500 dark:text-stone-400 text-sm line-clamp-2 leading-relaxed flex-1">
{s.description}
</p>
{/* Mini code preview */}
<CodePreview code={s.code} />
</motion.button>
);
})}
</div>
{/* ── Snippet Modal ── */}
<AnimatePresence>
{selectedSnippet && modalLang && (
<div className="fixed inset-0 z-[100] flex items-center justify-center p-4 md:p-8">
<motion.div
initial={{ opacity: 0 }}
animate={{ opacity: 1 }}
exit={{ opacity: 0 }}
onClick={() => setSelectedSnippet(null)}
className="absolute inset-0 bg-stone-950/60 backdrop-blur-md"
/>
<motion.div
key={selectedSnippet.id}
initial={{ opacity: 0, scale: 0.95, y: 16 }}
animate={{ opacity: 1, scale: 1, y: 0 }}
exit={{ opacity: 0, scale: 0.95, y: 16 }}
className="relative w-full max-w-3xl bg-white dark:bg-stone-900 rounded-[2.5rem] shadow-2xl border border-stone-200 dark:border-stone-800 overflow-hidden flex flex-col max-h-[90vh]"
>
<div className="p-8 md:p-10 overflow-y-auto">
{/* Modal header */}
<div className="flex justify-between items-start mb-6">
<div className="flex-1 min-w-0 pr-4">
<div className="flex items-center gap-3 mb-2 flex-wrap">
<p className="text-[10px] font-black uppercase tracking-[0.2em] text-liquid-purple">
{selectedSnippet.category}
</p>
{selectedSnippet.language && (
<span className={`px-2.5 py-0.5 rounded-lg text-[10px] font-black uppercase tracking-wider ${modalLang.bg} ${modalLang.text}`}>
{modalLang.label}
</span>
)}
</div>
<h3 className="text-2xl md:text-3xl font-black text-stone-900 dark:text-white uppercase tracking-tighter leading-tight">
{selectedSnippet.title}
</h3>
</div>
<button
onClick={() => setSelectedSnippet(null)}
className="p-3 bg-stone-50 dark:bg-stone-800 rounded-full hover:bg-stone-100 dark:hover:bg-stone-700 transition-colors shrink-0"
title="Close (Esc)"
>
<X size={20} />
</button>
</div>
<p className="text-stone-600 dark:text-stone-400 mb-8 leading-relaxed">
{selectedSnippet.description}
</p>
{/* Code block */}
<div className="relative">
<div className="absolute top-4 right-4 flex gap-2 z-10">
<button
onClick={() => copyToClipboard(selectedSnippet.code)}
className="p-2.5 bg-white/10 backdrop-blur-md hover:bg-white/20 rounded-lg border border-white/10 transition-all text-white"
title="Copy code"
>
{copied ? <Check size={16} className="text-emerald-400" /> : <Copy size={16} />}
</button>
</div>
<pre className="bg-stone-950 p-6 rounded-2xl overflow-x-auto text-sm font-mono text-stone-300 border border-stone-800 leading-relaxed">
<code>{selectedSnippet.code}</code>
</pre>
</div>
</div>
{/* Modal footer: navigation */}
<div className="px-8 py-5 bg-stone-50 dark:bg-stone-800/50 border-t border-stone-100 dark:border-stone-800 flex items-center justify-between">
<button
onClick={() => currentIndex > 0 && setSelectedSnippet(filtered[currentIndex - 1])}
disabled={currentIndex <= 0}
className="flex items-center gap-1.5 text-[10px] font-black uppercase tracking-[0.2em] text-stone-400 hover:text-stone-900 dark:hover:text-white transition-colors disabled:opacity-30 disabled:cursor-not-allowed"
title="Previous (←)"
>
<ChevronLeft size={14} /> Prev
</button>
<span className="text-[10px] font-black uppercase tracking-[0.2em] text-stone-300 dark:text-stone-600 tabular-nums">
{currentIndex + 1} / {filtered.length}
</span>
<button
onClick={() => currentIndex < filtered.length - 1 && setSelectedSnippet(filtered[currentIndex + 1])}
disabled={currentIndex >= filtered.length - 1}
className="flex items-center gap-1.5 text-[10px] font-black uppercase tracking-[0.2em] text-stone-400 hover:text-stone-900 dark:hover:text-white transition-colors disabled:opacity-30 disabled:cursor-not-allowed"
title="Next (→)"
>
Next <ChevronRight size={14} />
</button>
</div>
</motion.div>
</div>
)}
</AnimatePresence>
</>
);
}

View File

@@ -1,41 +0,0 @@
import React from "react";
import { getSnippets } from "@/lib/directus";
import { Terminal, ArrowLeft } from "lucide-react";
import Link from "next/link";
import SnippetsClient from "./SnippetsClient";
export default async function SnippetsPage({ params }: { params: Promise<{ locale: string }> }) {
const { locale } = await params;
const snippets = await getSnippets(100) || [];
return (
<main className="min-h-screen bg-[#fdfcf8] dark:bg-stone-950 py-24 px-6 transition-colors duration-500">
<div className="max-w-7xl mx-auto">
<Link
href={`/${locale}`}
className="inline-flex items-center gap-2 text-[10px] font-black uppercase tracking-[0.3em] text-stone-400 hover:text-stone-900 dark:hover:text-white transition-all mb-12 group"
>
<ArrowLeft size={14} className="group-hover:-translate-x-1 transition-transform" />
Back to Portfolio
</Link>
<header className="mb-20">
<div className="flex items-center gap-4 mb-6">
<div className="w-12 h-12 rounded-2xl bg-stone-900 dark:bg-stone-50 flex items-center justify-center text-white dark:text-stone-900">
<Terminal size={24} />
</div>
<h1 className="text-5xl md:text-8xl font-black tracking-tighter uppercase text-stone-900 dark:text-stone-50">
The Lab<span className="text-liquid-purple">.</span>
</h1>
</div>
<p className="text-xl md:text-2xl font-light text-stone-500 max-w-2xl leading-relaxed">
A collection of technical snippets, configurations, and mental notes from my daily building process.
</p>
</header>
<SnippetsClient initialSnippets={snippets} />
</div>
</main>
);
}

View File

@@ -1,21 +0,0 @@
import { NextRequest, NextResponse } from 'next/server';
import { getSnippets } from '@/lib/directus';
const CACHE_TTL = 300; // 5 minutes
export async function GET(request: NextRequest) {
try {
const { searchParams } = new URL(request.url);
const limit = parseInt(searchParams.get('limit') || '10');
const featured = searchParams.get('featured') === 'true' ? true : undefined;
const snippets = await getSnippets(limit, featured);
return NextResponse.json(
{ snippets: snippets || [] },
{ headers: { 'Cache-Control': `public, s-maxage=${CACHE_TTL}, stale-while-revalidate=${CACHE_TTL * 2}` } }
);
} catch (_error) {
return NextResponse.json({ error: 'Failed to fetch snippets' }, { status: 500 });
}
}

View File

@@ -7,13 +7,13 @@ import dynamic from "next/dynamic";
const RichTextClient = dynamic(() => import("./RichTextClient"), { ssr: false });
import CurrentlyReading from "./CurrentlyReading";
import ReadBooks from "./ReadBooks";
import { motion, AnimatePresence } from "framer-motion";
import { TechStackCategory, TechStackItem, Hobby, Snippet } from "@/lib/directus";
import { motion } from "framer-motion";
import { TechStackCategory, TechStackItem, Hobby } from "@/lib/directus";
import Link from "next/link";
import ActivityFeed from "./ActivityFeed";
import BentoChat from "./BentoChat";
import { Skeleton } from "./ui/Skeleton";
import { LucideIcon, X, Copy, Check } from "lucide-react";
import { LucideIcon } from "lucide-react";
const iconMap: Record<string, LucideIcon> = {
Globe, Server, Code, Wrench, Shield, Activity, Lightbulb, Gamepad2, BookOpen, Tv, Plane, Camera, Stars, Music, Terminal, Cpu
@@ -25,21 +25,17 @@ const About = () => {
const [cmsHtml, setCmsHtml] = useState<string | null>(null);
const [techStack, setTechStack] = useState<TechStackCategory[]>([]);
const [hobbies, setHobbies] = useState<Hobby[]>([]);
const [snippets, setSnippets] = useState<Snippet[]>([]);
const [selectedSnippet, setSelectedSnippet] = useState<Snippet | null>(null);
const [copied, setCopied] = useState(false);
const [_cmsMessages, setCmsMessages] = useState<Record<string, string>>({});
const [isLoading, setIsLoading] = useState(true);
useEffect(() => {
const fetchData = async () => {
try {
const [cmsRes, techRes, hobbiesRes, msgRes, snippetsRes] = await Promise.all([
const [cmsRes, techRes, hobbiesRes, msgRes] = await Promise.all([
fetch(`/api/content/page?key=home-about&locale=${locale}`),
fetch(`/api/tech-stack?locale=${locale}`),
fetch(`/api/hobbies?locale=${locale}`),
fetch(`/api/messages?locale=${locale}`),
fetch(`/api/snippets?limit=3&featured=true`)
fetch(`/api/messages?locale=${locale}`)
]);
const cmsData = await cmsRes.json();
@@ -53,9 +49,6 @@ const About = () => {
const msgData = await msgRes.json();
if (msgData?.messages) setCmsMessages(msgData.messages);
const snippetsData = await snippetsRes.json();
if (snippetsData?.snippets) setSnippets(snippetsData.snippets);
} catch (error) {
console.error("About data fetch failed:", error);
} finally {
@@ -65,12 +58,6 @@ const About = () => {
fetchData();
}, [locale]);
const copyToClipboard = (code: string) => {
navigator.clipboard.writeText(code);
setCopied(true);
setTimeout(() => setCopied(false), 2000);
};
return (
<section id="about" className="py-16 sm:py-24 md:py-32 px-4 sm:px-6 bg-[#fdfcf8] dark:bg-stone-950 transition-colors duration-500">
<div className="max-w-7xl mx-auto">
@@ -169,96 +156,61 @@ const About = () => {
</div>
</motion.div>
{/* 5. Library, Gear & Snippets */}
<div className="md:col-span-12 grid grid-cols-1 lg:grid-cols-12 gap-4 sm:gap-6 md:gap-8">
{/* Library - Larger Span */}
<motion.div
transition={{ delay: 0.4 }}
className="lg:col-span-7 bg-white dark:bg-stone-900 rounded-2xl sm:rounded-[2.5rem] md:rounded-[3rem] p-6 sm:p-8 md:p-10 border border-stone-200/60 dark:border-stone-800/60 shadow-sm flex flex-col group overflow-hidden relative min-h-[350px] sm:min-h-[400px] md:min-h-[500px]"
>
<div className="relative z-10 flex flex-col h-full">
<div className="flex justify-between items-center mb-6 sm:mb-8 md:mb-10">
<h3 className="text-xl sm:text-2xl font-black text-stone-900 dark:text-stone-50 flex items-center gap-2 sm:gap-3 uppercase tracking-tighter">
<BookOpen className="text-liquid-purple" size={24} /> Library
</h3>
<Link href={`/${locale}/books`} className="group/link flex items-center gap-2 text-stone-900 dark:text-stone-100 font-black border-b-2 border-stone-900 dark:border-stone-100 pb-1 hover:opacity-70 transition-all">
View All <ArrowRight size={14} className="group-hover/link:translate-x-1 transition-transform" />
</Link>
</div>
<CurrentlyReading />
<div className="mt-6 flex-1">
<ReadBooks />
{/* 5. Library */}
<motion.div
transition={{ delay: 0.4 }}
className="md:col-span-7 bg-white dark:bg-stone-900 rounded-2xl sm:rounded-[2.5rem] md:rounded-[3rem] p-6 sm:p-8 md:p-10 border border-stone-200/60 dark:border-stone-800/60 shadow-sm flex flex-col group overflow-hidden relative min-h-[350px] sm:min-h-[400px] md:min-h-[500px]"
>
<div className="relative z-10 flex flex-col h-full">
<div className="flex justify-between items-center mb-6 sm:mb-8 md:mb-10">
<h3 className="text-xl sm:text-2xl font-black text-stone-900 dark:text-stone-50 flex items-center gap-2 sm:gap-3 uppercase tracking-tighter">
<BookOpen className="text-liquid-purple" size={24} /> Library
</h3>
<Link href={`/${locale}/books`} className="group/link flex items-center gap-2 text-stone-900 dark:text-stone-100 font-black border-b-2 border-stone-900 dark:border-stone-100 pb-1 hover:opacity-70 transition-all">
View All <ArrowRight size={14} className="group-hover/link:translate-x-1 transition-transform" />
</Link>
</div>
<CurrentlyReading />
<div className="mt-6 flex-1">
<ReadBooks />
</div>
</div>
</motion.div>
{/* 6. My Gear */}
<motion.div
transition={{ delay: 0.5 }}
className="md:col-span-5 flex flex-col gap-4 sm:gap-6 md:gap-8"
>
<div className="flex-1 bg-stone-900 rounded-2xl sm:rounded-[2.5rem] md:rounded-[3rem] p-6 sm:p-8 md:p-10 border border-stone-800 shadow-2xl text-white relative overflow-hidden group">
<div className="relative z-10">
<h3 className="text-xl sm:text-2xl font-black mb-5 sm:mb-8 flex items-center gap-2 sm:gap-3 uppercase tracking-tighter text-white">
<Cpu className="text-liquid-mint" size={24} /> My Gear
</h3>
<div className="grid grid-cols-2 gap-4 sm:gap-6">
<div className="space-y-1">
<p className="text-[9px] font-black uppercase tracking-widest text-stone-400">Main</p>
<p className="text-sm font-bold text-stone-100">MacBook M4 Pro</p>
</div>
<div className="space-y-1">
<p className="text-[9px] font-black uppercase tracking-widest text-stone-400">PC</p>
<p className="text-sm font-bold text-stone-100">RTX 3080 / R7</p>
</div>
<div className="space-y-1">
<p className="text-[9px] font-black uppercase tracking-widest text-stone-400">Server</p>
<p className="text-sm font-bold text-stone-100">IONOS & RPi 4</p>
</div>
<div className="space-y-1">
<p className="text-[9px] font-black uppercase tracking-widest text-stone-400">OS</p>
<p className="text-sm font-bold text-stone-100">macOS / Linux</p>
</div>
</div>
</div>
</motion.div>
<div className="lg:col-span-5 flex flex-col gap-4 sm:gap-6 md:gap-8">
{/* My Gear (Uses) */}
<motion.div
transition={{ delay: 0.5 }}
className="bg-stone-900 rounded-2xl sm:rounded-[2.5rem] md:rounded-[3rem] p-6 sm:p-8 md:p-10 border border-stone-800 shadow-2xl text-white relative overflow-hidden group flex-1"
>
<div className="relative z-10">
<h3 className="text-xl sm:text-2xl font-black mb-5 sm:mb-8 flex items-center gap-2 sm:gap-3 uppercase tracking-tighter text-white">
<Cpu className="text-liquid-mint" size={24} /> My Gear
</h3>
<div className="grid grid-cols-2 gap-4 sm:gap-6">
<div className="space-y-1">
<p className="text-[9px] font-black uppercase tracking-widest text-stone-400">Main</p>
<p className="text-sm font-bold text-stone-100">MacBook M4 Pro</p>
</div>
<div className="space-y-1">
<p className="text-[9px] font-black uppercase tracking-widest text-stone-400">PC</p>
<p className="text-sm font-bold text-stone-100">RTX 3080 / R7</p>
</div>
<div className="space-y-1">
<p className="text-[9px] font-black uppercase tracking-widest text-stone-400">Server</p>
<p className="text-sm font-bold text-stone-100">IONOS & RPi 4</p>
</div>
<div className="space-y-1">
<p className="text-[9px] font-black uppercase tracking-widest text-stone-400">OS</p>
<p className="text-sm font-bold text-stone-100">macOS / Linux</p>
</div>
</div>
</div>
<div className="absolute bottom-0 right-0 w-32 h-32 bg-liquid-mint/10 blur-3xl rounded-full -mr-16 -mb-16" />
</motion.div>
<motion.div
transition={{ delay: 0.6 }}
className="bg-white dark:bg-stone-900 rounded-2xl sm:rounded-[2.5rem] md:rounded-[3rem] p-6 sm:p-8 md:p-10 border border-stone-200/60 dark:border-stone-800/60 shadow-sm flex flex-col justify-between group overflow-hidden relative flex-1"
>
<div className="relative z-10">
<h3 className="text-xl sm:text-2xl font-black text-stone-900 dark:text-stone-50 flex items-center gap-2 sm:gap-3 uppercase tracking-tighter mb-4 sm:mb-6">
<Terminal className="text-liquid-purple" size={24} /> Snippets
</h3>
<div className="space-y-3">
{isLoading ? (
Array.from({ length: 2 }).map((_, i) => <Skeleton key={i} className="h-12 rounded-xl" />)
) : snippets.length > 0 ? (
snippets.map((s) => (
<button
key={s.id}
onClick={() => setSelectedSnippet(s)}
className="w-full text-left p-3 bg-stone-50 dark:bg-stone-800 rounded-xl border border-stone-100 dark:border-stone-700 hover:border-liquid-purple transition-all group/s"
>
<p className="text-[9px] font-black uppercase tracking-widest text-stone-600 dark:text-stone-400 mb-0.5 group-hover/s:text-liquid-purple transition-colors">{s.category}</p>
<p className="text-xs font-bold text-stone-800 dark:text-stone-200">{s.title}</p>
</button>
))
) : (
<p className="text-xs text-stone-500 dark:text-stone-400 italic">No snippets yet.</p>
)}
</div>
</div>
<Link href={`/${locale}/snippets`} className="mt-6 group/btn inline-flex items-center gap-2 text-[10px] font-black uppercase tracking-widest text-stone-600 dark:text-stone-400 hover:text-stone-900 dark:hover:text-white transition-colors">
Enter the Lab <ArrowRight size={12} className="group-hover/btn:translate-x-1 transition-transform" />
</Link>
</motion.div>
<div className="absolute bottom-0 right-0 w-32 h-32 bg-liquid-mint/10 blur-3xl rounded-full -mr-16 -mb-16" />
</div>
</div>
</motion.div>
{/* 6. Hobbies */}
{/* 7. Hobbies */}
<motion.div
transition={{ delay: 0.5 }}
className="md:col-span-12"
@@ -293,71 +245,8 @@ const About = () => {
</div>
</div>
{/* Snippet Modal */}
<AnimatePresence>
{selectedSnippet && (
<div className="fixed inset-0 z-[100] flex items-center justify-center p-4 md:p-8">
<motion.div
initial={{ opacity: 0 }}
animate={{ opacity: 1 }}
exit={{ opacity: 0 }}
onClick={() => setSelectedSnippet(null)}
className="absolute inset-0 bg-stone-950/60 backdrop-blur-md"
/>
<motion.div
initial={{ opacity: 0, scale: 0.9, y: 20 }}
animate={{ opacity: 1, scale: 1, y: 0 }}
exit={{ opacity: 0, scale: 0.9, y: 20 }}
className="relative w-full max-w-3xl bg-white dark:bg-stone-900 rounded-2xl sm:rounded-[2.5rem] shadow-2xl border border-stone-200 dark:border-stone-800 overflow-hidden flex flex-col max-h-[90vh]"
>
<div className="p-5 sm:p-8 md:p-10 overflow-y-auto">
<div className="flex justify-between items-start mb-5 sm:mb-8">
<div>
<p className="text-[10px] font-black uppercase tracking-[0.2em] text-liquid-purple mb-1 sm:mb-2">{selectedSnippet.category}</p>
<h3 className="text-xl sm:text-2xl md:text-3xl font-black text-stone-900 dark:text-white uppercase tracking-tighter">{selectedSnippet.title}</h3>
</div>
<button
onClick={() => setSelectedSnippet(null)}
className="p-3 bg-stone-50 dark:bg-stone-800 rounded-full hover:bg-stone-100 dark:hover:bg-stone-700 transition-colors"
>
<X size={20} />
</button>
</div>
<p className="text-sm sm:text-base text-stone-600 dark:text-stone-400 mb-5 sm:mb-8 leading-relaxed">
{selectedSnippet.description}
</p>
<div className="relative group/code">
<div className="absolute top-3 right-3 sm:top-4 sm:right-4 flex gap-2">
<button
onClick={() => copyToClipboard(selectedSnippet.code)}
className="p-2 sm:p-2.5 bg-white/10 backdrop-blur-md hover:bg-white/20 rounded-lg border border-white/10 transition-all text-white"
title="Copy Code"
>
{copied ? <Check size={16} className="text-emerald-400" /> : <Copy size={16} />}
</button>
</div>
<pre className="bg-stone-950 p-4 sm:p-6 rounded-xl sm:rounded-2xl overflow-x-auto text-xs sm:text-sm font-mono text-stone-300 border border-stone-800 leading-relaxed">
<code>{selectedSnippet.code}</code>
</pre>
</div>
</div>
<div className="p-6 bg-stone-50 dark:bg-stone-800/50 border-t border-stone-100 dark:border-stone-800 text-center">
<button
onClick={() => setSelectedSnippet(null)}
className="text-[10px] font-black uppercase tracking-[0.2em] text-stone-600 dark:text-stone-400 hover:text-stone-900 dark:hover:text-white transition-colors"
>
Close Laboratory
</button>
</div>
</motion.div>
</div>
)}
</AnimatePresence>
</section>
);
};
export default About;
export default About;

View File

@@ -64,9 +64,14 @@ const Footer = () => {
{/* Bottom Bar */}
<div className="mt-10 sm:mt-16 md:mt-20 pt-6 sm:pt-8 border-t border-stone-100 dark:border-stone-900 flex flex-col md:flex-row justify-between items-center gap-4">
<p className="text-[10px] font-bold text-stone-600 dark:text-stone-400 uppercase tracking-widest">
Built with Next.js, Directus & Passion.
</p>
<div className="flex flex-col gap-1">
<p className="text-[10px] font-bold text-stone-600 dark:text-stone-400 uppercase tracking-widest">
Built with Next.js, Directus & Passion.
</p>
<p className="text-[10px] text-stone-400 dark:text-stone-600 tracking-wide">
{t("aiDisclaimer")}
</p>
</div>
<div className="flex items-center gap-2">
<div className="w-2 h-2 rounded-full bg-green-500 animate-pulse" />
<span className="text-[10px] font-bold text-stone-600 dark:text-stone-400 uppercase tracking-widest">Systems Online</span>

View File

@@ -1,7 +1,7 @@
"use client";
import { motion, AnimatePresence } from "framer-motion";
import { BookCheck, Star, ChevronDown, ChevronUp, X } from "lucide-react";
import { motion } from "framer-motion";
import { BookCheck, Star, ChevronDown, ChevronUp } from "lucide-react";
import { useEffect, useState } from "react";
import { useLocale, useTranslations } from "next-intl";
import Image from "next/image";
@@ -48,7 +48,6 @@ const ReadBooks = () => {
const [reviews, setReviews] = useState<BookReview[]>([]);
const [loading, setLoading] = useState(true);
const [expanded, setExpanded] = useState(false);
const [selectedReview, setSelectedReview] = useState<BookReview | null>(null);
const INITIAL_SHOW = 3;
@@ -199,17 +198,9 @@ const ReadBooks = () => {
{/* Review Text (Optional) */}
{review.review && (
<div>
<p className="text-sm text-stone-700 dark:text-stone-300 leading-relaxed line-clamp-3 italic">
&ldquo;{stripHtml(review.review)}&rdquo;
</p>
<button
onClick={() => setSelectedReview(review)}
className="text-xs text-liquid-mint dark:text-liquid-sky hover:underline mt-1 font-medium"
>
{t("readMore", { defaultValue: "Read full review" })}
</button>
</div>
<p className="text-sm text-stone-700 dark:text-stone-300 leading-relaxed italic">
&ldquo;{stripHtml(review.review)}&rdquo;
</p>
)}
{/* Finished Date */}
@@ -249,130 +240,7 @@ const ReadBooks = () => {
</motion.button>
)}
{/* Modal for full review */}
<AnimatePresence>
{selectedReview && (
<>
{/* Backdrop */}
<motion.div
initial={{ opacity: 0 }}
animate={{ opacity: 1 }}
exit={{ opacity: 0 }}
onClick={() => setSelectedReview(null)}
className="fixed inset-0 bg-black/70 backdrop-blur-md z-50"
/>
{/* Modal */}
<motion.div
initial={{ opacity: 0, scale: 0.95, y: 40 }}
animate={{ opacity: 1, scale: 1, y: 0 }}
exit={{ opacity: 0, scale: 0.95, y: 40 }}
transition={{ type: "spring", damping: 30, stiffness: 400 }}
className="fixed inset-x-4 bottom-4 top-20 sm:inset-auto sm:left-1/2 sm:top-1/2 sm:-translate-x-1/2 sm:-translate-y-1/2 sm:w-full sm:max-w-3xl sm:max-h-[85vh] z-50 bg-gradient-to-br from-white via-liquid-sky/5 to-liquid-mint/10 dark:from-stone-900 dark:via-stone-900 dark:to-stone-800 rounded-3xl shadow-2xl border-2 border-liquid-mint/30 dark:border-stone-700 overflow-hidden"
>
{/* Decorative blob */}
<div className="absolute -top-20 -right-20 w-64 h-64 bg-gradient-to-br from-liquid-mint/20 to-liquid-sky/20 dark:from-stone-700/30 dark:to-stone-600/30 rounded-full blur-3xl pointer-events-none" />
{/* Close button */}
<button
onClick={() => setSelectedReview(null)}
className="absolute top-4 right-4 p-2.5 rounded-full bg-white/80 dark:bg-stone-800/80 backdrop-blur-sm hover:bg-white dark:hover:bg-stone-700 transition-all duration-200 z-10 shadow-lg border border-stone-200 dark:border-stone-600"
aria-label="Close"
>
<X size={20} className="text-stone-600 dark:text-stone-300" />
</button>
{/* Content */}
<div className="relative h-full overflow-y-auto overscroll-contain p-6 sm:p-8 md:p-10">
<div className="flex flex-col sm:flex-row gap-6 mb-6">
{/* Book Cover */}
{selectedReview.book_image && (
<motion.div
initial={{ opacity: 0, x: -20 }}
animate={{ opacity: 1, x: 0 }}
transition={{ delay: 0.1 }}
className="flex-shrink-0 mx-auto sm:mx-0"
>
<div className="relative w-32 h-48 sm:w-36 sm:h-52 rounded-xl overflow-hidden shadow-2xl border-2 border-white/50 dark:border-stone-700">
<Image
src={selectedReview.book_image}
alt={selectedReview.book_title}
fill
className="object-cover"
sizes="(max-width: 640px) 128px, 144px"
priority
/>
<div className="absolute inset-0 bg-gradient-to-tr from-white/20 via-transparent to-white/10 pointer-events-none" />
</div>
</motion.div>
)}
{/* Book Info */}
<motion.div
initial={{ opacity: 0, y: 10 }}
animate={{ opacity: 1, y: 0 }}
transition={{ delay: 0.15 }}
className="flex-1 min-w-0"
>
<h2 className="text-2xl sm:text-3xl font-bold text-stone-900 dark:text-stone-100 mb-2 leading-tight">
{selectedReview.book_title}
</h2>
<p className="text-base sm:text-lg text-stone-600 dark:text-stone-400 mb-4">
{selectedReview.book_author}
</p>
{selectedReview.rating && selectedReview.rating > 0 && (
<div className="flex items-center gap-3 mb-4">
<div className="flex gap-1">
{[1, 2, 3, 4, 5].map((star) => (
<Star
key={star}
size={18}
className={
star <= selectedReview.rating!
? "text-amber-500 fill-amber-500"
: "text-stone-300 dark:text-stone-600"
}
/>
))}
</div>
<span className="text-base text-stone-600 dark:text-stone-400 font-semibold">
{selectedReview.rating}/5
</span>
</div>
)}
{selectedReview.finished_at && (
<p className="text-sm text-stone-500 dark:text-stone-400 flex items-center gap-2">
<BookCheck size={14} className="opacity-60" />
{t("finishedAt")}{" "}
{new Date(selectedReview.finished_at).toLocaleDateString(
locale === "de" ? "de-DE" : "en-US",
{ year: "numeric", month: "long", day: "numeric" }
)}
</p>
)}
</motion.div>
</div>
{/* Full Review */}
{selectedReview.review && (
<motion.div
initial={{ opacity: 0, y: 20 }}
animate={{ opacity: 1, y: 0 }}
transition={{ delay: 0.2 }}
className="bg-gradient-to-br from-liquid-mint/10 via-liquid-sky/5 to-transparent dark:from-stone-800/50 dark:via-stone-800/30 dark:to-transparent rounded-2xl p-6 border-l-4 border-liquid-mint dark:border-liquid-sky"
>
<p className="text-base sm:text-lg text-stone-700 dark:text-stone-300 leading-relaxed italic">
&ldquo;{stripHtml(selectedReview.review)}&rdquo;
</p>
</motion.div>
)}
</div>
</motion.div>
</>
)}
</AnimatePresence>
</div>
);
};

View File

@@ -1,7 +1,7 @@
"use client";
import { motion } from "framer-motion";
import { ArrowLeft, Search, Terminal } from "lucide-react";
import { ArrowLeft, Search } from "lucide-react";
import Link from "next/link";
import { useRouter } from "next/navigation";
import { useEffect, useState } from "react";
@@ -25,7 +25,7 @@ export default function NotFound() {
<motion.div
initial={{ opacity: 0, y: 30 }}
animate={{ opacity: 1, y: 0 }}
className="md:col-span-12 lg:col-span-8 bg-white dark:bg-stone-900 rounded-2xl sm:rounded-[2.5rem] md:rounded-[3rem] p-6 sm:p-8 md:p-16 border border-stone-200/60 dark:border-stone-800/60 shadow-sm flex flex-col justify-between min-h-[300px] sm:min-h-[400px]"
className="md:col-span-12 bg-white dark:bg-stone-900 rounded-2xl sm:rounded-[2.5rem] md:rounded-[3rem] p-6 sm:p-8 md:p-16 border border-stone-200/60 dark:border-stone-800/60 shadow-sm flex flex-col justify-between min-h-[300px] sm:min-h-[400px]"
>
<div>
<div className="flex items-center gap-3 mb-6 sm:mb-8 md:mb-12">
@@ -58,52 +58,29 @@ export default function NotFound() {
</div>
</motion.div>
{/* Sidebar Cards */}
<div className="md:col-span-12 lg:col-span-4 flex flex-col gap-4 sm:gap-6">
{/* Search/Explore Projects */}
<motion.div
initial={{ opacity: 0, x: 20 }}
animate={{ opacity: 1, x: 0 }}
transition={{ delay: 0.1 }}
className="bg-stone-900 rounded-2xl sm:rounded-[2.5rem] p-6 sm:p-8 md:p-10 border border-stone-800 shadow-2xl text-white relative overflow-hidden group flex-1 flex flex-col justify-between"
{/* Explore Work Card */}
<motion.div
initial={{ opacity: 0, x: 20 }}
animate={{ opacity: 1, x: 0 }}
transition={{ delay: 0.1 }}
className="md:col-span-12 bg-stone-900 rounded-2xl sm:rounded-[2.5rem] p-6 sm:p-8 md:p-10 border border-stone-800 shadow-2xl text-white relative overflow-hidden group flex flex-col justify-between min-h-[200px]"
>
<div className="relative z-10">
<Search className="text-liquid-mint mb-4 sm:mb-6" size={28} />
<h3 className="text-xl sm:text-2xl font-black uppercase tracking-tighter mb-1 sm:mb-2">Explore Work</h3>
<p className="text-stone-400 text-sm font-medium">Maybe what you need is in my project archive?</p>
</div>
<Link
href="/projects"
className="mt-5 sm:mt-8 inline-flex items-center gap-2 text-[10px] font-black uppercase tracking-widest text-liquid-mint group-hover:gap-4 transition-all"
>
<div className="relative z-10">
<Search className="text-liquid-mint mb-4 sm:mb-6" size={28} />
<h3 className="text-xl sm:text-2xl font-black uppercase tracking-tighter mb-1 sm:mb-2">Explore Work</h3>
<p className="text-stone-400 text-sm font-medium">Maybe what you need is in my project archive?</p>
</div>
<Link
href="/projects"
className="mt-5 sm:mt-8 inline-flex items-center gap-2 text-[10px] font-black uppercase tracking-widest text-liquid-mint group-hover:gap-4 transition-all"
>
View Projects <ArrowLeft className="rotate-180" size={14} />
</Link>
<div className="absolute top-0 right-0 w-32 h-32 bg-liquid-mint/5 blur-3xl rounded-full -mr-16 -mt-16" />
</motion.div>
{/* Visit the Lab */}
<motion.div
initial={{ opacity: 0, x: 20 }}
animate={{ opacity: 1, x: 0 }}
transition={{ delay: 0.2 }}
className="bg-white dark:bg-stone-900 rounded-2xl sm:rounded-[2.5rem] p-6 sm:p-8 md:p-10 border border-stone-200/60 dark:border-stone-800/60 shadow-sm flex-1 flex flex-col justify-between group"
>
<div className="relative z-10">
<Terminal className="text-liquid-purple mb-4 sm:mb-6" size={28} />
<h3 className="text-xl sm:text-2xl font-black uppercase tracking-tighter mb-1 sm:mb-2 text-stone-900 dark:text-stone-50">Technical</h3>
<p className="text-stone-500 dark:text-stone-400 text-sm font-medium">Check out my collection of code snippets and notes.</p>
</div>
<Link
href="/snippets"
className="mt-8 inline-flex items-center gap-2 text-[10px] font-black uppercase tracking-widest text-stone-400 hover:text-stone-900 dark:hover:text-white transition-colors"
>
Enter the Lab <ArrowLeft className="rotate-180" size={14} />
</Link>
</motion.div>
</div>
View Projects <ArrowLeft className="rotate-180" size={14} />
</Link>
<div className="absolute top-0 right-0 w-32 h-32 bg-liquid-mint/5 blur-3xl rounded-full -mr-16 -mt-16" />
</motion.div>
</div>
</div>
</main>
);
}
}

View File

@@ -1,154 +0,0 @@
# 🚀 TELEGRAM CMS - QUICK START GUIDE
## Installation (5 Minutes)
### Step 1: Import Main Workflow
1. Open n8n: https://n8n.dk0.dev
2. Click "Workflows" → "Import from File"
3. Select: `n8n-workflows/ULTIMATE-Telegram-CMS-COMPLETE.json`
4. Workflow should auto-activate
### Step 2: Verify Credentials
Check these credentials exist (should be auto-mapped):
- ✅ Telegram: `DK0_Server`
- ✅ Directus: Bearer token `RF2Qytq...`
- ✅ OpenRouter: Bearer token `sk-or-v1-...`
### Step 3: Test Commands
Open Telegram bot and type:
```
/start
```
You should see the dashboard! 🎉
---
## 📋 All Commands
| Command | Description | Example |
|---------|-------------|---------|
| `/start` | Main dashboard | `/start` |
| `/list projects` | Show draft projects | `/list projects` |
| `/list books` | Show pending reviews | `/list books` |
| `/search <term>` | Search everywhere | `/search nextjs` |
| `/stats` | Analytics dashboard | `/stats` |
| `/preview <ID>` | Preview item (EN+DE) | `/preview 42` |
| `/publish <ID>` | Publish to live site | `/publish 42` |
| `/delete <ID>` | Delete item | `/delete 42` |
| `/deletereview <ID>` | Delete book review | `/deletereview 3` |
| `.review <HC_ID> <RATING> <TEXT>` | Create book review | `.review427565 4 Great!` |
---
## 🔧 Companion Workflows (Auto-Import)
These workflows work together with the main CMS:
### 1. Docker Event Workflow
**File:** `Docker Event.json` (KEEP ACTIVE)
- Auto-detects new container deployments
- AI generates project descriptions
- Creates drafts in Directus
- Sends Telegram notification with buttons
### 2. Book Review Scheduler
**File:** `Book Review.json` (KEEP ACTIVE)
- Runs daily at 7 PM
- Checks for unreviewed books
- Sends AI-generated questions
- You reply with `.review` command
### 3. Finished Books Sync
**File:** `finishedBooks.json` (KEEP ACTIVE)
- Runs daily at 6 AM
- Syncs from Hardcover API
- Adds new books to Directus
### 4. Portfolio Status API
**File:** `portfolio-website.json` (KEEP ACTIVE)
- Real-time status endpoint
- Aggregates: Spotify + Discord + WakaTime
- Used by website for "Now" section
### 5. Currently Reading API
**File:** `reading (1).json` (KEEP ACTIVE)
- Webhook endpoint
- Fetches current books from Hardcover
- Returns formatted JSON
---
## 🎯 Typical Workflows
### Publishing a New Project:
1. Deploy Docker container
2. Get Telegram notification: "🚀 New Deploy: portfolio-dev"
3. Click "🤖 Auto-generieren" button
4. AI creates draft
5. Get notification: "Draft created (ID: 42)"
6. Type: `/preview 42` to check translations
7. Type: `/publish 42` to go live
### Adding a Book Review:
1. Finish reading book on Hardcover
2. Get Telegram prompt at 7 PM: "📚 Review this book?"
3. Reply: `.review427565 4 Great world-building but rushed ending`
4. AI generates EN + DE reviews
5. Get notification: "Review draft created (ID: 3)"
6. Type: `/publish 3` to publish
### Quick Search:
1. Type: `/search suricata`
2. See all projects/books mentioning "suricata"
3. Click action buttons to manage
---
## 🐛 Troubleshooting
### "Command not recognized"
- Check workflow is **Active** (toggle in n8n)
- Verify Telegram Trigger credential is set
### "Error fetching data"
- Check Directus is running: https://cms.dk0.dev
- Verify Bearer token in credentials
### "No button appears" (Docker workflow)
- Check `Docker Event - Callback Handler.json` is active
- Inline keyboard markup must be set correctly
### "AI generation fails"
- Check OpenRouter credit balance
- Model `openrouter/free` might be rate-limited, switch to `google/gemini-2.5-flash`
---
## 📊 Monitoring
Check n8n Executions:
- n8n → Left menu → "Executions"
- Filter by workflow name
- Red = Failed (click to see error details)
- Green = Success
---
## 🚀 Next Steps
1. **Test all commands** - Go through each one in Telegram
2. **Customize messages** - Edit text in Telegram nodes
3. **Add your own commands** - Extend the Switch node
4. **Set up monitoring** - Add error alerts to Slack/Discord
---
## 📞 Support
If something breaks:
1. Check n8n Execution logs
2. Verify API credentials
3. Test Directus API manually: `curl https://cms.dk0.dev/items/projects`
**Your system is now LIVE!** 🎉

View File

@@ -1,269 +0,0 @@
# 🚀 ULTIMATE TELEGRAM CMS SYSTEM - Implementation Plan
**Status:** Ready to implement
**Duration:** ~15 minutes
**Completion:** 8/8 workflows
---
## 🎯 System Overview
Your portfolio will be **fully manageable via Telegram** with these features:
### ✅ Commands (All work via Telegram Bot)
| Command | Function | Example |
|---------|----------|---------|
| `/start` | Main dashboard with quick action buttons | - |
| `/list projects` | Show all draft projects | `/list projects` |
| `/list books` | Show pending book reviews | `/list books` |
| `/search <term>` | Search projects & books | `/search nextjs` |
| `/stats` | Analytics dashboard (views, trends) | `/stats` |
| `/preview <ID>` | Show EN + DE translations before publish | `/preview 42` |
| `/publish <ID>` | Publish project or book (auto-detects type) | `/publish 42` |
| `/delete <ID>` | Delete project or book | `/delete 42` |
| `/deletereview <ID>` | Delete specific book review translation | `/deletereview 3` |
| `.review <HC_ID> <RATING> <TEXT>` | Create AI-powered book review | `.review427565 4 Great book!` |
---
## 📦 Workflow Architecture
```
┌─────────────────────────────────────────────────────────────┐
│ 🤖 ULTIMATE TELEGRAM CMS (Master Router) │
│ Handles: /start, /list, /search, /stats, /preview, etc. │
└─────────────────────────────────────────────────────────────┘
┌─────────────────────┼─────────────────────┐
│ │ │
┌────▼────┐ ┌────▼────┐ ┌────▼────┐
│ Docker │ │ Book │ │ Status │
│ Events │ │ Reviews │ │ API │
└─────────┘ └─────────┘ └─────────┘
Auto-creates AI prompts Spotify +
project drafts for reviews Discord +
WakaTime
```
---
## 🛠️ Implementation Steps
### **1. Command Router** ✅ (DONE)
- File: `ULTIMATE-Telegram-CMS.json`
- Central command parser
- Switch routes to 10 different actions
### **2. /start Dashboard**
```telegram
🏠 Portfolio CMS Dashboard
📊 Quick Stats:
├─ 3 Draft Projects
├─ 2 Pending Reviews
└─ Last updated: 2 hours ago
⚡ Quick Actions:
┌────────────────┬────────────────┐
│ 📋 List Drafts │ 🔍 Search │
└────────────────┴────────────────┘
┌────────────────┬────────────────┐
│ 📈 Stats │ 🔄 Sync Now │
└────────────────┴────────────────┘
```
### **3. /list Command**
```telegram
📋 Draft Projects (3):
1⃣ #42 Portfolio Website
Category: webdev
Created: 2 days ago
/preview42 · /publish42 · /delete42
2⃣ #38 Suricata IDS
Category: selfhosted
Created: 1 week ago
/preview38 · /publish38 · /delete38
───────────────────────────
/list books → See book reviews
```
### **4. /search Command**
```telegram
🔍 Search: "nextjs"
Found 2 results:
📦 Projects:
1. #42 - Portfolio Website (Next.js 15...)
📚 Books:
(none)
```
### **5. /stats Command**
```telegram
📈 Portfolio Stats (Last 30 Days)
🏆 Top Projects:
1. Portfolio Website - 1,240 views
2. Docker Setup - 820 views
3. Suricata IDS - 450 views
📚 Book Reviews:
├─ Total: 12 books
├─ This month: 3 reviews
└─ Avg rating: 4.2/5
⚡ Activity:
├─ Projects published: 5
├─ Drafts created: 8
└─ Reviews written: 3
```
### **6. /preview Command**
```telegram
👁️ Preview: Portfolio Website (#42)
🇬🇧 ENGLISH:
Title: Modern Portfolio with Next.js
Description: A responsive portfolio showcasing...
🇩🇪 DEUTSCH:
Title: Modernes Portfolio mit Next.js
Description: Ein responsives Portfolio das...
───────────────────────────
/publish42 · /delete42
```
### **7. Publish/Delete Logic**
- Auto-detects collection (projects vs book_reviews)
- Fetches item details from Directus
- Updates `status` field
- Sends confirmation with item title
### **8. AI Review Creator** ✅ (Already works!)
- `.review <HC_ID> <RATING> <TEXT>`
- Calls OpenRouter AI
- Generates EN + DE translations
- Creates draft in Directus
---
## 🔧 Technical Implementation
### **Workflow 1: ULTIMATE-Telegram-CMS.json**
**Nodes:**
1. Telegram Trigger (listens to messages)
2. Parse Command (regex matcher)
3. Switch Action (10 outputs)
4. Dashboard Node → Fetch stats from Directus
5. List Node → Query projects/books with pagination
6. Search Node → GraphQL search on Directus
7. Stats Node → Aggregate views/counts
8. Preview Node → Fetch translations
9. Publish Node → Update status field
10. Delete Node → Delete item + translations
### **Directus Collections Used:**
- `projects` (slug, title, category, status, technologies, translations)
- `book_reviews` (hardcover_id, rating, finished_at, translations)
- `tech_stack_categories` (name, technologies)
### **APIs Integrated:**
- ✅ Directus CMS (Bearer Token: `RF2Qytq...`)
- ✅ Hardcover.app (GraphQL)
- ✅ OpenRouter AI (Free models)
- ✅ Gitea (Self-hosted Git)
- ✅ Spotify, Discord Lanyard, Wakapi
---
## 🎨 Telegram UI Patterns
### **Inline Keyboards:**
```javascript
{
"replyMarkup": "inlineKeyboard",
"inlineKeyboard": {
"rows": [
{
"buttons": [
{ "text": "📋 List", "callbackData": "list_projects" },
{ "text": "🔍 Search", "callbackData": "search_prompt" }
]
}
]
}
}
```
### **Pagination:**
```javascript
{
"buttons": [
{ "text": "◀️ Prev", "callbackData": "list_page:1" },
{ "text": "Page 2/5", "callbackData": "noop" },
{ "text": "▶️ Next", "callbackData": "list_page:3" }
]
}
```
---
## 📊 Implementation Checklist
- [x] Command parser with 10 actions
- [ ] Dashboard (/start) with stats
- [ ] List command (projects/books)
- [ ] Search command (fuzzy matching)
- [ ] Stats dashboard (views, trends)
- [ ] Preview command (EN + DE)
- [ ] Unified publish logic (auto-detect collection)
- [ ] Unified delete logic with confirmation
- [ ] Error handling (try-catch all API calls)
- [ ] Logging (audit trail in Directus)
---
## 🚀 Deployment Steps
1. **Import workflow:** n8n → Import `ULTIMATE-Telegram-CMS.json`
2. **Set credentials:**
- Telegram Bot: `DK0_Server` (already exists)
- Directus Bearer: `RF2Qytq...` (already exists)
3. **Activate workflow:** Toggle ON
4. **Test commands:**
```
/start
/list projects
/stats
```
---
## 🎯 Future Enhancements
1. **Media Upload** - Send image → "For which project?" → Auto-upload
2. **Scheduled Publishing** - `/schedule <ID> <date>`
3. **Bulk Operations** - `/bulkpublish`, `/archive`
4. **Webhook Monitoring** - Alert if workflows fail
5. **Multi-language AI** - Switch between OpenRouter models
6. **Undo Command** - Revert last action
---
## 📝 Notes
- Chat ID: `145931600` (hardcoded, change if needed)
- Timezone: Europe/Berlin (hardcoded in some workflows)
- AI Model: `openrouter/free` (cheapest, decent quality)
- Rate Limit: None (add if needed)
---
**Ready to deploy?** Import `ULTIMATE-Telegram-CMS.json` into n8n and activate it!

View File

@@ -937,63 +937,6 @@ export async function getProjectBySlug(
}
}
// Snippets Types
export interface Snippet {
id: string;
title: string;
category: string;
code: string;
description: string;
language: string;
}
/**
* Get Snippets from Directus
*/
export async function getSnippets(limit = 10, featured?: boolean): Promise<Snippet[] | null> {
const filters = ['status: { _eq: "published" }'];
if (featured !== undefined) {
filters.push(`featured: { _eq: ${featured} }`);
}
const filterString = `filter: { _and: [{ ${filters.join(' }, { ')} }] }`;
const query = `
query {
snippets(
${filterString}
limit: ${limit}
) {
id
title
category
code
description
language
}
}
`;
try {
const result = await directusRequest(
'',
{ body: { query } }
);
interface SnippetsResult {
snippets: Snippet[];
}
const snippets = (result as SnippetsResult | null)?.snippets;
if (!snippets || snippets.length === 0) {
return null;
}
return snippets;
} catch (_error) {
console.error('Failed to fetch snippets:', _error);
return null;
}
}
// ─── Hardcover → Directus sync helpers ───────────────────────────────────────
export interface BookReviewCreate {

View File

@@ -157,6 +157,7 @@
"privacyPolicy": "Datenschutz",
"privacySettings": "Datenschutz-Einstellungen",
"privacySettingsTitle": "Datenschutz-Banner wieder anzeigen",
"builtWith": "Built with"
"builtWith": "Built with",
"aiDisclaimer": "Einige Inhalte dieser Seite können KI-generiert sein."
}
}

View File

@@ -160,7 +160,8 @@
"privacyPolicy": "Privacy policy",
"privacySettings": "Privacy settings",
"privacySettingsTitle": "Show privacy settings banner again",
"builtWith": "Built with"
"builtWith": "Built with",
"aiDisclaimer": "Some content on this site may be AI-assisted."
}
}

View File

@@ -1,260 +0,0 @@
{
"name": "Docker Event - Callback Handler",
"nodes": [
{
"parameters": {
"updates": ["callback_query"]
},
"type": "n8n-nodes-base.telegramTrigger",
"typeVersion": 1.2,
"position": [0, 0],
"id": "telegram-trigger",
"name": "Telegram Trigger"
},
{
"parameters": {
"jsCode": "const callback = $input.first().json;\nconst data = callback.callback_query?.data || '';\nconst chatId = callback.callback_query?.from?.id;\nconst messageId = callback.callback_query?.message?.message_id;\n\n// Parse: auto:slug, manual:slug, ignore:slug\nconst [action, slug] = data.split(':');\n\nreturn [{\n json: {\n action,\n slug,\n chatId,\n messageId,\n rawCallback: data\n }\n}];"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [220, 0],
"id": "parse-callback",
"name": "Parse Callback"
},
{
"parameters": {
"rules": {
"values": [
{
"conditions": {
"options": {
"caseSensitive": true,
"leftValue": ""
},
"conditions": [
{
"leftValue": "={{ $json.action }}",
"rightValue": "auto",
"operator": {
"type": "string",
"operation": "equals"
}
}
],
"combinator": "and"
},
"renameOutput": true,
"outputKey": "Auto"
},
{
"conditions": {
"options": {
"caseSensitive": true,
"leftValue": ""
},
"conditions": [
{
"leftValue": "={{ $json.action }}",
"rightValue": "manual",
"operator": {
"type": "string",
"operation": "equals"
}
}
],
"combinator": "and"
},
"renameOutput": true,
"outputKey": "Manual"
},
{
"conditions": {
"options": {
"caseSensitive": true,
"leftValue": ""
},
"conditions": [
{
"leftValue": "={{ $json.action }}",
"rightValue": "ignore",
"operator": {
"type": "string",
"operation": "equals"
}
}
],
"combinator": "and"
},
"renameOutput": true,
"outputKey": "Ignore"
}
]
},
"options": {}
},
"type": "n8n-nodes-base.switch",
"typeVersion": 3.2,
"position": [440, 0],
"id": "switch-action",
"name": "Switch Action"
},
{
"parameters": {
"url": "=https://cms.dk0.dev/items/projects?filter[slug][_eq]={{ $json.slug }}&limit=1",
"authentication": "predefinedCredentialType",
"nodeCredentialType": "httpBearerAuth",
"options": {}
},
"type": "n8n-nodes-base.httpRequest",
"typeVersion": 4.4,
"position": [660, -200],
"id": "get-project-data",
"name": "Get Project from CMS"
},
{
"parameters": {
"url": "=https://git.dk0.dev/api/v1/repos/denshooter/{{ $json.slug }}/commits?limit=3",
"authentication": "genericCredentialType",
"genericAuthType": "httpHeaderAuth",
"options": {}
},
"type": "n8n-nodes-base.httpRequest",
"typeVersion": 4.4,
"position": [880, -280],
"id": "get-commits-auto",
"name": "Get Commits"
},
{
"parameters": {
"url": "=https://git.dk0.dev/api/v1/repos/denshooter/{{ $json.slug }}/contents/README.md",
"authentication": "genericCredentialType",
"genericAuthType": "httpHeaderAuth",
"options": {}
},
"type": "n8n-nodes-base.httpRequest",
"typeVersion": 4.4,
"position": [880, -160],
"id": "get-readme-auto",
"name": "Get README"
},
{
"parameters": {
"model": "openrouter/free",
"options": {}
},
"type": "@n8n/n8n-nodes-langchain.lmChatOpenRouter",
"typeVersion": 1,
"position": [1320, -100],
"id": "openrouter-model-auto",
"name": "OpenRouter Chat Model"
},
{
"parameters": {
"promptType": "define",
"text": "=Du bist ein technischer Autor für das Portfolio von Dennis (dk0.dev).\n\nNeues eigenes Projekt deployed:\nRepo: {{ $('Parse Callback').item.json.slug }}\n\nREADME:\n{{ $('Get README').first().json.content ? Buffer.from($('Get README').first().json.content, 'base64').toString('utf8').substring(0, 1000) : 'Kein README' }}\n\nLetzte Commits:\n{{ $('Get Commits').first().json.map(c => '- ' + c.commit.message).join('\\n') }}\n\nErstelle eine Portfolio-Beschreibung:\n- Was macht das Projekt (Features, Zweck)\n- Tech-Stack und Architektur\n- Highlights aus den Commits\n- Warum ist es cool/interessant\n\nKategorie: webdev (wenn Web-App), automation (wenn Tool/Script), oder selfhosted\n\nAntworte NUR als JSON:\n{\n \"title_en\": \"Aussagekräftiger Titel\",\n \"title_de\": \"Aussagekräftiger Titel\",\n \"description_en\": \"4-6 Sätze\",\n \"description_de\": \"4-6 Sätze\",\n \"content_en\": \"2-3 Absätze Markdown mit technischen Details\",\n \"content_de\": \"2-3 Absätze Markdown mit technischen Details\",\n \"category\": \"webdev|automation|selfhosted\",\n \"technologies\": [\"Next.js\", \"Docker\", \"...\"]\n}"
},
"type": "@n8n/n8n-nodes-langchain.chainLlm",
"typeVersion": 1.9,
"position": [1100, -200],
"id": "ai-auto",
"name": "AI: Generate Description"
},
{
"parameters": {
"jsCode": "const raw = $input.first().json.text ?? \"\";\nconst match = raw.match(/\\{[\\s\\S]*\\}/);\nif (!match) throw new Error(\"No JSON found\");\nconst ai = JSON.parse(match[0]);\nreturn [{ json: ai }];"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [1320, -200],
"id": "parse-json-auto",
"name": "Parse JSON"
},
{
"parameters": {
"jsCode": "const ai = $input.first().json;\nconst ctx = $('Parse Callback').first().json;\n\nconst body = {\n slug: ctx.slug,\n status: \"draft\",\n featured: false,\n title: ai.title_en,\n category: ai.category,\n technologies: ai.technologies,\n tags: ai.technologies,\n date: new Date().toISOString().slice(0, 10),\n translations: {\n create: [\n {\n languages_code: \"en-US\",\n title: ai.title_en,\n description: ai.description_en,\n content: ai.content_en\n },\n {\n languages_code: \"de-DE\",\n title: ai.title_de,\n description: ai.description_de,\n content: ai.content_de\n }\n ]\n }\n};\n\nconst response = await this.helpers.httpRequest({\n method: \"POST\",\n url: \"https://cms.dk0.dev/items/projects\",\n headers: {\n \"Content-Type\": \"application/json\",\n \"Authorization\": \"Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB\"\n },\n body\n});\n\nreturn [{ json: response }];"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [1540, -200],
"id": "add-to-directus-auto",
"name": "Add to Directus"
},
{
"parameters": {
"chatId": "={{ $('Parse Callback').item.json.chatId }}",
"text": "={{ \n'✅ Projekt erstellt: ' + $json.data.title + '\\n\\n' +\n'📝 ' + $('Parse JSON').first().json.description_de.substring(0, 200) + '...\\n\\n' +\n'Status: Draft (ID: ' + $json.data.id + ')\\n\\n' +\n'/publishproject' + $json.data.id + ' — Veröffentlichen\\n' + \n'/deleteproject' + $json.data.id + ' — Löschen' \n}}",
"additionalFields": {}
},
"type": "n8n-nodes-base.telegram",
"typeVersion": 1.2,
"position": [1760, -200],
"id": "telegram-notify-auto",
"name": "Notify Success"
},
{
"parameters": {
"chatId": "={{ $json.chatId }}",
"text": "✍️ OK, schreib mir jetzt was das Projekt macht (4-6 Sätze).\n\nIch formatiere das dann schön und erstelle einen Draft.",
"additionalFields": {}
},
"type": "n8n-nodes-base.telegram",
"typeVersion": 1.2,
"position": [660, 0],
"id": "telegram-ask-manual",
"name": "Ask for Manual Input"
},
{
"parameters": {
"chatId": "={{ $json.chatId }}",
"text": "❌ OK, ignoriert.",
"additionalFields": {}
},
"type": "n8n-nodes-base.telegram",
"typeVersion": 1.2,
"position": [660, 200],
"id": "telegram-ignore",
"name": "Confirm Ignore"
}
],
"connections": {
"Telegram Trigger": {
"main": [[{ "node": "Parse Callback", "type": "main", "index": 0 }]]
},
"Parse Callback": {
"main": [[{ "node": "Switch Action", "type": "main", "index": 0 }]]
},
"Switch Action": {
"main": [
[{ "node": "Get Project from CMS", "type": "main", "index": 0 }],
[{ "node": "Ask for Manual Input", "type": "main", "index": 0 }],
[{ "node": "Confirm Ignore", "type": "main", "index": 0 }]
]
},
"Get Project from CMS": {
"main": [[{ "node": "Get Commits", "type": "main", "index": 0 }]]
},
"Get Commits": {
"main": [[{ "node": "Get README", "type": "main", "index": 0 }]]
},
"Get README": {
"main": [[{ "node": "AI: Generate Description", "type": "main", "index": 0 }]]
},
"OpenRouter Chat Model": {
"ai_languageModel": [[{ "node": "AI: Generate Description", "type": "ai_languageModel", "index": 0 }]]
},
"AI: Generate Description": {
"main": [[{ "node": "Parse JSON", "type": "main", "index": 0 }]]
},
"Parse JSON": {
"main": [[{ "node": "Add to Directus", "type": "main", "index": 0 }]]
},
"Add to Directus": {
"main": [[{ "node": "Notify Success", "type": "main", "index": 0 }]]
}
},
"active": false,
"settings": {
"executionOrder": "v1"
},
"id": "docker-event-callback"
}

View File

@@ -1,372 +0,0 @@
{
"name": "Docker Event (Extended)",
"nodes": [
{
"parameters": {
"httpMethod": "POST",
"path": "docker-event",
"responseMode": "responseNode",
"options": {}
},
"type": "n8n-nodes-base.webhook",
"typeVersion": 2.1,
"position": [0, 0],
"id": "webhook-main",
"name": "Webhook"
},
{
"parameters": {
"jsCode": "const data = $input.first().json;\n\nconst container = data.container ?? data.body?.container ?? '';\nconst image = data.image ?? data.body?.image ?? '';\nconst timestamp = data.timestamp ?? data.body?.timestamp ?? '';\n\nconst slug = container.toLowerCase().replace(/[^a-z0-9]+/g, '-');\nconst serviceName = container.replace(/[-_]/g, ' ');\n\n// Detect project type\nlet projectType = 'selfhosted';\nif (image.includes('denshooter') || image.includes('dk0')) {\n projectType = 'own';\n} else if (container.match(/^(act-|gitea-actions-|runner-)/)) {\n projectType = 'cicd';\n}\n\n// Extract repo from image for own projects\nlet repo = null;\nif (projectType === 'own') {\n const match = image.match(/([^/]+):(\\w+)/);\n if (match) repo = match[1];\n}\n\nreturn [{\n json: {\n container,\n image,\n serviceName,\n timestamp,\n slug,\n projectType,\n repo\n }\n}];"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [220, 0],
"id": "parse-context",
"name": "Parse Context"
},
{
"parameters": {
"url": "=https://cms.dk0.dev/items/projects?filter[slug][_eq]={{ $json.slug }}&limit=1",
"authentication": "predefinedCredentialType",
"nodeCredentialType": "httpBearerAuth",
"options": {}
},
"type": "n8n-nodes-base.httpRequest",
"typeVersion": 4.4,
"position": [440, 0],
"id": "search-slug",
"name": "Check if Exists"
},
{
"parameters": {
"conditions": {
"options": {
"caseSensitive": true,
"leftValue": "",
"typeValidation": "loose"
},
"conditions": [
{
"leftValue": "={{ $json.data.length }}",
"rightValue": "0",
"operator": {
"type": "number",
"operation": "equals"
}
}
],
"combinator": "and"
},
"options": {}
},
"type": "n8n-nodes-base.if",
"typeVersion": 2.3,
"position": [660, 0],
"id": "if-new",
"name": "If New"
},
{
"parameters": {
"rules": {
"values": [
{
"conditions": {
"options": {
"caseSensitive": true,
"leftValue": ""
},
"conditions": [
{
"leftValue": "={{ $('Parse Context').item.json.projectType }}",
"rightValue": "own",
"operator": {
"type": "string",
"operation": "equals"
}
}
],
"combinator": "and"
},
"renameOutput": true,
"outputKey": "Own Project"
},
{
"conditions": {
"options": {
"caseSensitive": true,
"leftValue": ""
},
"conditions": [
{
"leftValue": "={{ $('Parse Context').item.json.projectType }}",
"rightValue": "cicd",
"operator": {
"type": "string",
"operation": "equals"
}
}
],
"combinator": "and"
},
"renameOutput": true,
"outputKey": "CI/CD (Ignore)"
},
{
"conditions": {
"options": {
"caseSensitive": true,
"leftValue": ""
},
"conditions": [
{
"leftValue": "={{ $('Parse Context').item.json.projectType }}",
"rightValue": "selfhosted",
"operator": {
"type": "string",
"operation": "equals"
}
}
],
"combinator": "and"
},
"renameOutput": true,
"outputKey": "Self-Hosted"
}
]
},
"options": {}
},
"type": "n8n-nodes-base.switch",
"typeVersion": 3.2,
"position": [880, 0],
"id": "switch-type",
"name": "Switch Type"
},
{
"parameters": {
"url": "=https://git.dk0.dev/api/v1/repos/denshooter/{{ $('Parse Context').item.json.repo }}/commits?limit=1",
"authentication": "genericCredentialType",
"genericAuthType": "httpHeaderAuth",
"options": {}
},
"type": "n8n-nodes-base.httpRequest",
"typeVersion": 4.4,
"position": [1100, -200],
"id": "get-commits",
"name": "Get Last Commit",
"credentials": {
"httpHeaderAuth": {
"id": "gitea-token",
"name": "Gitea API"
}
}
},
{
"parameters": {
"url": "=https://git.dk0.dev/api/v1/repos/denshooter/{{ $('Parse Context').item.json.repo }}/contents/README.md",
"authentication": "genericCredentialType",
"genericAuthType": "httpHeaderAuth",
"options": {}
},
"type": "n8n-nodes-base.httpRequest",
"typeVersion": 4.4,
"position": [1100, -80],
"id": "get-readme",
"name": "Get README"
},
{
"parameters": {
"jsCode": "const ctx = $('Parse Context').first().json;\nconst commits = $('Get Last Commit').first().json;\nconst readme = $('Get README').first().json;\n\n// Get commit data\nconst commit = Array.isArray(commits) ? commits[0] : commits;\nconst commitMsg = commit?.commit?.message || 'No recent commits';\nconst commitAuthor = commit?.commit?.author?.name || 'Unknown';\n\n// Decode README (base64)\nlet readmeText = '';\ntry {\n const content = readme?.content || readme?.data?.content;\n if (content) {\n readmeText = Buffer.from(content, 'base64').toString('utf8');\n // First 500 chars\n readmeText = readmeText.substring(0, 500).replace(/\\n/g, ' ').trim();\n } else {\n readmeText = 'No README available';\n }\n} catch (e) {\n readmeText = 'No README available';\n}\n\nconsole.log('Commit:', commitMsg);\nconsole.log('README excerpt:', readmeText.substring(0, 100));\n\nreturn [{\n json: {\n container: ctx.container,\n image: ctx.image,\n slug: ctx.slug,\n repo: ctx.repo,\n commitMsg,\n commitAuthor,\n readmeExcerpt: readmeText\n }\n}];"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [1320, -140],
"id": "merge-git-data",
"name": "Merge Git Data"
},
{
"parameters": {
"chatId": "145931600",
"text": "={{ \n'🚀 Neuer Deploy: ' + $json.container + '\\n' +\n'📦 ' + $json.image + '\\n\\n' +\n'📝 Letzter Commit:\\n' + $json.commitMsg + '\\n' +\n'👤 ' + $json.commitAuthor + '\\n\\n' +\n'📄 README:\\n' + $json.readmeExcerpt + '...\\n\\n' +\n'Was ist das Highlight?' \n}}",
"additionalFields": {
"replyMarkup": "inlineKeyboard",
"inlineKeyboard": {
"rows": [
{
"buttons": [
{
"text": "✍️ Selbst beschreiben",
"callbackData": "={{ 'manual:' + $json.slug }}"
},
{
"text": "🤖 Auto-generieren",
"callbackData": "={{ 'auto:' + $json.slug }}"
}
]
},
{
"buttons": [
{
"text": "❌ Ignorieren",
"callbackData": "={{ 'ignore:' + $json.slug }}"
}
]
}
]
}
}
},
"type": "n8n-nodes-base.telegram",
"typeVersion": 1.2,
"position": [1540, -140],
"id": "telegram-ask",
"name": "Ask via Telegram"
},
{
"parameters": {
"model": "openrouter/free",
"options": {}
},
"type": "@n8n/n8n-nodes-langchain.lmChatOpenRouter",
"typeVersion": 1,
"position": [1540, 160],
"id": "openrouter-model",
"name": "OpenRouter Chat Model"
},
{
"parameters": {
"promptType": "define",
"text": "=Du bist ein technischer Autor für dk0.dev.\n\nNeuer Self-Hosted Service:\nContainer: {{ $('Parse Context').item.json.container }}\nImage: {{ $('Parse Context').item.json.image }}\n\nErstelle eine Portfolio-Beschreibung:\n- Was macht die App\n- Warum Self-Hosting besser ist als Cloud\n- Wie sie in die Infrastruktur integriert ist\n\nAntworte NUR als JSON:\n{\n \"title_en\": \"Titel\",\n \"title_de\": \"Titel\",\n \"description_en\": \"4-6 Sätze\",\n \"description_de\": \"4-6 Sätze\",\n \"content_en\": \"2-3 Absätze Markdown\",\n \"content_de\": \"2-3 Absätze Markdown\",\n \"category\": \"selfhosted\",\n \"technologies\": [\"Docker\", \"...\"]\n}"
},
"type": "@n8n/n8n-nodes-langchain.chainLlm",
"typeVersion": 1.9,
"position": [1320, 80],
"id": "ai-selfhosted",
"name": "AI: Self-Hosted"
},
{
"parameters": {
"jsCode": "const raw = $input.first().json.text ?? \"\";\nconst match = raw.match(/\\{[\\s\\S]*\\}/);\nif (!match) throw new Error(\"No JSON found\");\nconst ai = JSON.parse(match[0]);\nreturn [{ json: ai }];"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [1540, 80],
"id": "parse-json-selfhosted",
"name": "Parse JSON"
},
{
"parameters": {
"jsCode": "const ai = $input.first().json;\nconst ctx = $('Parse Context').first().json;\n\nconst body = {\n slug: ctx.slug,\n status: \"draft\",\n featured: false,\n title: ai.title_en,\n category: ai.category,\n technologies: ai.technologies,\n tags: ai.technologies,\n date: new Date().toISOString().slice(0, 10),\n translations: {\n create: [\n {\n languages_code: \"en-US\",\n title: ai.title_en,\n description: ai.description_en,\n content: ai.content_en\n },\n {\n languages_code: \"de-DE\",\n title: ai.title_de,\n description: ai.description_de,\n content: ai.content_de\n }\n ]\n }\n};\n\nconst response = await this.helpers.httpRequest({\n method: \"POST\",\n url: \"https://cms.dk0.dev/items/projects\",\n headers: {\n \"Content-Type\": \"application/json\",\n \"Authorization\": \"Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB\"\n },\n body\n});\n\nreturn [{ json: response }];"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [1760, 80],
"id": "add-to-directus-selfhosted",
"name": "Add to Directus"
},
{
"parameters": {
"chatId": "145931600",
"text": "={{ \n'🆕 Self-Hosted Service: ' + $('Parse Context').first().json.serviceName + '\\n\\n' +\n'📝 ' + $json.data.title + '\\n\\n' +\n'Status: Draft erstellt (ID: ' + $json.data.id + ')\\n\\n' +\n'/publishproject' + $json.data.id + ' — Veröffentlichen\\n' + \n'/deleteproject' + $json.data.id + ' — Löschen' \n}}",
"additionalFields": {}
},
"type": "n8n-nodes-base.telegram",
"typeVersion": 1.2,
"position": [1980, 80],
"id": "telegram-notify-selfhosted",
"name": "Notify Selfhosted"
},
{
"parameters": {
"respondWith": "json",
"responseBody": "{ \"success\": true, \"message\": \"CI/CD container ignored\" }",
"options": {}
},
"type": "n8n-nodes-base.respondToWebhook",
"typeVersion": 1.5,
"position": [1100, 200],
"id": "respond-ignore",
"name": "Respond (Ignore)"
},
{
"parameters": {
"respondWith": "json",
"responseBody": "{ \"success\": true }",
"options": {}
},
"type": "n8n-nodes-base.respondToWebhook",
"typeVersion": 1.5,
"position": [2200, 0],
"id": "respond-success",
"name": "Respond"
},
{
"parameters": {
"respondWith": "json",
"responseBody": "{ \"success\": true, \"message\": \"Project already exists\" }",
"options": {}
},
"type": "n8n-nodes-base.respondToWebhook",
"typeVersion": 1.5,
"position": [880, 200],
"id": "respond-exists",
"name": "Respond (Exists)"
}
],
"connections": {
"Webhook": {
"main": [[{ "node": "Parse Context", "type": "main", "index": 0 }]]
},
"Parse Context": {
"main": [[{ "node": "Check if Exists", "type": "main", "index": 0 }]]
},
"Check if Exists": {
"main": [[{ "node": "If New", "type": "main", "index": 0 }]]
},
"If New": {
"main": [
[{ "node": "Switch Type", "type": "main", "index": 0 }],
[{ "node": "Respond (Exists)", "type": "main", "index": 0 }]
]
},
"Switch Type": {
"main": [
[{ "node": "Get Last Commit", "type": "main", "index": 0 }],
[{ "node": "Respond (Ignore)", "type": "main", "index": 0 }],
[{ "node": "AI: Self-Hosted", "type": "main", "index": 0 }]
]
},
"Get Last Commit": {
"main": [[{ "node": "Get README", "type": "main", "index": 0 }]]
},
"Get README": {
"main": [[{ "node": "Merge Git Data", "type": "main", "index": 0 }]]
},
"Merge Git Data": {
"main": [[{ "node": "Ask via Telegram", "type": "main", "index": 0 }]]
},
"Ask via Telegram": {
"main": [[{ "node": "Respond", "type": "main", "index": 0 }]]
},
"OpenRouter Chat Model": {
"ai_languageModel": [[{ "node": "AI: Self-Hosted", "type": "ai_languageModel", "index": 0 }]]
},
"AI: Self-Hosted": {
"main": [[{ "node": "Parse JSON", "type": "main", "index": 0 }]]
},
"Parse JSON": {
"main": [[{ "node": "Add to Directus", "type": "main", "index": 0 }]]
},
"Add to Directus": {
"main": [[{ "node": "Notify Selfhosted", "type": "main", "index": 0 }]]
},
"Notify Selfhosted": {
"main": [[{ "node": "Respond", "type": "main", "index": 0 }]]
}
},
"active": false,
"settings": {
"executionOrder": "v1"
},
"id": "docker-event-extended"
}

View File

@@ -1,120 +0,0 @@
var d = $input.first().json;
// GET book from CMS
var book;
try {
var check = await this.helpers.httpRequest({
method: "GET",
url: "https://cms.dk0.dev/items/book_reviews",
headers: { Authorization: "Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB" },
qs: {
"filter[hardcover_id][_eq]": d.hardcoverId,
"fields": "id,book_title,book_author",
"limit": 1
}
});
book = check.data?.[0];
} catch (e) {
var errmsg = "❌ GET Fehler: " + e.message;
return [{ json: { msg: errmsg, chatId: d.chatId } }];
}
if (!book) {
var errmsg = "❌ Buch mit Hardcover ID " + d.hardcoverId + " nicht gefunden.";
return [{ json: { msg: errmsg, chatId: d.chatId } }];
}
console.log("Book found:", book.book_title);
// Generate German review
var promptDe = "Schreibe eine persönliche Buchrezension (4-6 Sätze, Ich-Perspektive, nur Deutsch) zu '" + book.book_title + "' von " + book.book_author + ". Rating: " + d.rating + "/5. Meine Gedanken: " + d.answers + ". Formuliere professionell aber authentisch. NUR der Review-Text, kein JSON, kein Titel, keine Anführungszeichen drumherum.";
var reviewDe;
try {
console.log("Generating German review...");
var aiDe = await this.helpers.httpRequest({
method: "POST",
url: "https://openrouter.ai/api/v1/chat/completions",
headers: {
"Content-Type": "application/json",
Authorization: "Bearer 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"
},
body: {
model: "google/gemini-2.5-flash",
messages: [{ role: "user", content: promptDe }],
temperature: 0.7
}
});
reviewDe = aiDe.choices?.[0]?.message?.content?.trim() || d.answers;
console.log("German review generated:", reviewDe.substring(0, 100) + "...");
} catch (e) {
console.log("German AI error:", e.message);
reviewDe = d.answers;
}
// Generate English review
var promptEn = "You are a professional book critic writing in ENGLISH ONLY. Write a personal book review (4-6 sentences, first person perspective) of '" + book.book_title + "' by " + book.book_author + ". Rating: " + d.rating + "/5 stars. Reader notes: " + d.answers + ". Write professionally but authentically. OUTPUT ONLY THE REVIEW TEXT IN ENGLISH, no JSON, no title, no quotes.";
var reviewEn;
try {
console.log("Generating English review...");
var aiEn = await this.helpers.httpRequest({
method: "POST",
url: "https://openrouter.ai/api/v1/chat/completions",
headers: {
"Content-Type": "application/json",
Authorization: "Bearer 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"
},
body: {
model: "openrouter/free",
messages: [
{ role: "system", content: "You are a book critic. You ALWAYS write in English, never in German." },
{ role: "user", content: promptEn }
],
temperature: 0.7
}
});
reviewEn = aiEn.choices?.[0]?.message?.content?.trim() || d.answers;
console.log("English review generated:", reviewEn.substring(0, 100) + "...");
} catch (e) {
console.log("English AI error:", e.message);
reviewEn = d.answers;
}
// PATCH book with reviews
try {
console.log("Patching book #" + book.id);
await this.helpers.httpRequest({
method: "PATCH",
url: "https://cms.dk0.dev/items/book_reviews/" + book.id,
headers: {
"Content-Type": "application/json",
Authorization: "Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB"
},
body: {
rating: d.rating,
status: "draft",
translations: {
create: [
{ languages_code: "en-US", review: reviewEn },
{ languages_code: "de-DE", review: reviewDe }
]
}
}
});
console.log("PATCH success");
} catch (e) {
console.log("PATCH ERROR:", e.message);
var errmsg = "❌ PATCH Fehler: " + e.message;
return [{ json: { msg: errmsg, chatId: d.chatId } }];
}
// Build Telegram message (no emojis for better encoding)
var msg = "REVIEW: " + book.book_title + " - " + d.rating + "/5 Sterne";
msg = msg + "\n\n--- DEUTSCH ---\n" + reviewDe;
msg = msg + "\n\n--- ENGLISH ---\n" + reviewEn;
msg = msg + "\n\n==================";
msg = msg + "\n/publishbook" + book.id + " - Veroeffentlichen";
msg = msg + "\n/deletereview" + book.id + " - Loeschen und nochmal";
return [{ json: { msg: msg, chatId: d.chatId } }];

View File

@@ -1,935 +0,0 @@
{
"name": "Docker Event (Extended)",
"nodes": [
{
"parameters": {
"httpMethod": "POST",
"path": "docker-event",
"responseMode": "responseNode",
"options": {}
},
"type": "n8n-nodes-base.webhook",
"typeVersion": 2.1,
"position": [
0,
-224
],
"id": "870fa550-42f6-4e19-a796-f1f044b0cdc8",
"name": "Webhook",
"webhookId": "e147d70b-79d8-44fd-bbe8-8274cf905b11",
"disabled": true
},
{
"parameters": {
"jsCode": "const data = $input.first().json;\n\nconst container = data.container ?? data.body?.container ?? '';\nconst image = data.image ?? data.body?.image ?? '';\nconst timestamp = data.timestamp ?? data.body?.timestamp ?? '';\n\nconst slug = container.toLowerCase().replace(/[^a-z0-9]+/g, '-');\n\nconst serviceName = container.replace(/[-_]/g, ' ');\n\nreturn [{\n json: {\n container,\n image,\n serviceName,\n timestamp,\n slug \n }\n}];"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [
224,
-224
],
"id": "aaa6a678-1ad3-4f82-9b01-37e21b47b189",
"name": "Kontext aufbereiten",
"disabled": true
},
{
"parameters": {
"conditions": {
"options": {
"caseSensitive": true,
"leftValue": "",
"typeValidation": "loose",
"version": 3
},
"conditions": [
{
"id": "ebe26f0c-d5a7-45c9-9747-afc75b57a41c",
"leftValue": "={{ $json.data }}",
"rightValue": "[]",
"operator": {
"type": "string",
"operation": "notEndsWith"
}
}
],
"combinator": "and"
},
"looseTypeValidation": true,
"options": {}
},
"type": "n8n-nodes-base.if",
"typeVersion": 2.3,
"position": [
672,
-224
],
"id": "62197a33-5169-48e1-9539-57c047efb108",
"name": "If",
"disabled": true
},
{
"parameters": {
"url": "=https://cms.dk0.dev/items/projects?filter[slug][_eq]={{ $json.slug }}&limit=1",
"authentication": "predefinedCredentialType",
"nodeCredentialType": "httpBearerAuth",
"options": {}
},
"type": "n8n-nodes-base.httpRequest",
"typeVersion": 4.4,
"position": [
448,
-224
],
"id": "db783886-06b5-4473-8907-dd6c655aa3dd",
"name": "Search for Slug",
"credentials": {
"httpBearerAuth": {
"id": "ZtI5e08iryR9m6FG",
"name": "Directus"
}
},
"disabled": true
},
{
"parameters": {
"model": "openrouter/free",
"options": {}
},
"type": "@n8n/n8n-nodes-langchain.lmChatOpenRouter",
"typeVersion": 1,
"position": [
976,
16
],
"id": "b9130ff4-359b-4736-9442-1b0ca7d31877",
"name": "OpenRouter Chat Model",
"credentials": {
"openRouterApi": {
"id": "8Kdy4RHHwMZ0Cn6x",
"name": "OpenRouter"
}
},
"disabled": true
},
{
"parameters": {
"promptType": "define",
"text": "= Du bist ein technischer Autor für das Self-Hosting Portfolio von Dennis auf dk0.dev.\n Ein neuer Service wurde auf dem Server deployed:\n \n Container: {{ $('Kontext aufbereiten').item.json.container }}\n Image: {{ $('Kontext aufbereiten').item.json.image }}\n Service: {{ $('Kontext aufbereiten').item.json.serviceName }}\n \n Aufgabe:\n 1. Erkenne ob es sich um ein EIGENES Projekt (z.B. Image enthält \"denshooter\", \"dk0\", \"portfolio\") oder eine \nSELF-HOSTED App handelt.\n 2. Bewerte die \"Coolness\" (1-10) basierend auf:\n - Eigener Code = +3 Punkte\n - Neue/spannende Technologie = +2 Punkte\n - Großes/bekanntes Projekt (Suricata, CrowdStrike-Level) = +3 Punkte\n - Standard Self-Hosted Tool (Nextcloud, Plausible) = +1 Punkt\n - CI/CD Build-Container, Test-Runner = 0 Punkte (ignorieren)\n 3. Erstelle Beschreibung NUR wenn coolness_score >= 6\n \n Antworte NUR als valides JSON:\n {\n \"coolness_score\": 1-10,\n \"notify\": true/false (true wenn >= 7),\n \"reason\": \"Kurze Begründung warum cool oder nicht\",\n \"type\": \"own\" oder \"selfhosted\" oder \"ignore\",\n \"title_en\": \"...\",\n \"title_de\": \"...\",\n \"description_en\": \"...\",\n \"description_de\": \"...\",\n \"content_en\": \"...\",\n \"content_de\": \"...\",\n \"category\": \"selfhosted\" oder \"webdev\" oder \"automation\",\n \"technologies\": [\"Docker\", \"...\"]\n }",
"batching": {}
},
"type": "@n8n/n8n-nodes-langchain.chainLlm",
"typeVersion": 1.9,
"position": [
896,
-224
],
"id": "77d46075-3342-4e93-8806-07087a2389dc",
"name": "Basic LLM Chain",
"disabled": true
},
{
"parameters": {
"jsCode": "const raw = $input.first().json.text ?? \"\";\n\nconst match = raw.match(/\\{[\\s\\S]*\\}/);\nif (!match) throw new Error(\"No JSON found\");\n\nconst ai = JSON.parse(match[0]);\n\nreturn [\n {\n json: ai,\n },\n];\n"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [
1248,
-224
],
"id": "de5ed311-0d46-4677-963c-711a6ad514e9",
"name": "Parse JSON",
"disabled": true
},
{
"parameters": {
"jsCode": "const ai = $('Parse JSON').first().json;\n const ctx = $('Kontext aufbereiten').first().json;\n\n const body = {\n slug: ctx.slug,\n status: \"draft\",\n featured: false,\n title: ai.title_en,\n category: ai.category,\n technologies: ai.technologies,\n tags: ai.technologies,\n date: new Date().toISOString().slice(0, 10),\n translations: {\n create: [\n {\n languages_code: \"en-US\",\n title: ai.title_en,\n description: ai.description_en,\n content: ai.content_en\n },\n {\n languages_code: \"de-DE\",\n title: ai.title_de,\n description: ai.description_de,\n content: ai.content_de\n }\n ]\n }\n };\n\n const response = await this.helpers.httpRequest({\n method: \"POST\",\n url: \"https://cms.dk0.dev/items/projects\",\n headers: {\n \"Content-Type\": \"application/json\",\n \"Authorization\": \"Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB\"\n },\n body\n });\n\n return [{ json: response }];"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [
1680,
-224
],
"id": "c47b915d-e4d7-43e9-8ee3-b41389896fa7",
"name": "Add to Directus",
"disabled": true
},
{
"parameters": {
"respondWith": "json",
"responseBody": "{ \"success\": true }",
"options": {}
},
"type": "n8n-nodes-base.respondToWebhook",
"typeVersion": 1.5,
"position": [
2128,
-224
],
"id": "6cf8f30d-1352-466f-9163-9b4f16b972e0",
"name": "Respond to Webhook",
"disabled": true
},
{
"parameters": {
"chatId": "145931600",
"text": "={{ \n'🆕 Neuer Service erkannt!\\n\\n' +\n'📦 ' + $('Kontext aufbereiten').first().json.container + '\\n' +\n'🐳 ' + $('Kontext aufbereiten').first().json.image + '\\n\\n' +\n'📝 ' + $('Parse JSON').first().json.title_de + '\\n' + \n$('Parse JSON').first().json.description_de + '\\n\\n' +\n'Status: Draft in Directus erstellt (ID: ' + $json.data.id + ')\\n\\n' +\n('/publishproject_' + $json.data.id).replace(/_/g, '\\\\_') + ' — Veröffentlichen\\n' + \n('/deleteproject_' + $json.data.id).replace(/_/g, '\\\\_') + ' — Löschen' \n}}",
"additionalFields": {}
},
"type": "n8n-nodes-base.telegram",
"typeVersion": 1.2,
"position": [
1904,
-224
],
"id": "b29de3ec-b1ca-40c3-8493-af44e5372fd2",
"name": "Send a text message",
"webhookId": "c02ccf69-16dc-436e-b1cc-f8fa9dd8d33f",
"credentials": {
"telegramApi": {
"id": "ADurvy9EKUDzbDdq",
"name": "DK0_Server"
}
},
"disabled": true
},
{
"parameters": {
"rules": {
"values": [
{
"conditions": {
"options": {
"caseSensitive": true,
"leftValue": "",
"typeValidation": "strict",
"version": 3
},
"conditions": [
{
"leftValue": "={{ $json.notify }}",
"rightValue": "true",
"operator": {
"type": "boolean",
"operation": "true",
"singleValue": true
},
"id": "febc397c-b060-4a66-ab9b-1274c8509cc2"
}
],
"combinator": "and"
}
}
]
},
"options": {}
},
"type": "n8n-nodes-base.switch",
"typeVersion": 3.4,
"position": [
1456,
-224
],
"id": "5ade115f-e134-4358-8d95-a144eede8d9a",
"name": "Switch",
"disabled": true
},
{
"parameters": {
"jsCode": "const data = $input.first().json;\n\nconst container = data.container ?? data.body?.container ?? '';\nconst image = data.image ?? data.body?.image ?? '';\nconst timestamp = data.timestamp ?? data.body?.timestamp ?? '';\n\nconst slug = container.toLowerCase().replace(/[^a-z0-9]+/g, '-');\nconst serviceName = container.replace(/[-_]/g, ' ');\n\n// Detect project type\nlet projectType = 'selfhosted';\nif (image.includes('denshooter') || image.includes('dk0')) {\n projectType = 'own';\n} else if (container.match(/^(act-|gitea-actions-|runner-)/)) {\n projectType = 'cicd';\n}\n\n// Extract repo from image for own projects\nlet repo = null;\nif (projectType === 'own') {\n const match = image.match(/([^/]+):(\\w+)/);\n if (match) repo = match[1];\n}\n\nreturn [{\n json: {\n container,\n image,\n serviceName,\n timestamp,\n slug,\n projectType,\n repo\n }\n}];"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [
896,
768
],
"id": "fb34f047-5c11-4255-9b45-adb9fe169042",
"name": "Parse Context"
},
{
"parameters": {
"url": "=https://cms.dk0.dev/items/projects?filter[slug][_eq]={{ $json.slug }}&limit=1",
"authentication": "predefinedCredentialType",
"nodeCredentialType": "httpBearerAuth",
"options": {}
},
"type": "n8n-nodes-base.httpRequest",
"typeVersion": 4.4,
"position": [
1120,
768
],
"id": "acd7a411-2465-4aa3-a7ee-442a79c500f2",
"name": "Check if Exists",
"credentials": {
"httpBearerAuth": {
"id": "ZtI5e08iryR9m6FG",
"name": "Directus"
}
}
},
{
"parameters": {
"conditions": {
"options": {
"caseSensitive": true,
"leftValue": "",
"typeValidation": "loose"
},
"conditions": [
{
"leftValue": "={{ $json.data.length }}",
"rightValue": "0",
"operator": {
"type": "number",
"operation": "equals"
}
}
],
"combinator": "and"
},
"options": {}
},
"type": "n8n-nodes-base.if",
"typeVersion": 2.3,
"position": [
1344,
768
],
"id": "bdcddb94-8676-4467-a370-ad2cf07d09a3",
"name": "If New"
},
{
"parameters": {
"rules": {
"values": [
{
"conditions": {
"options": {
"caseSensitive": true,
"leftValue": ""
},
"conditions": [
{
"leftValue": "={{ $('Parse Context').item.json.projectType }}",
"rightValue": "own",
"operator": {
"type": "string",
"operation": "equals"
}
}
],
"combinator": "and"
},
"renameOutput": true,
"outputKey": "Own Project"
},
{
"conditions": {
"options": {
"caseSensitive": true,
"leftValue": ""
},
"conditions": [
{
"leftValue": "={{ $('Parse Context').item.json.projectType }}",
"rightValue": "cicd",
"operator": {
"type": "string",
"operation": "equals"
}
}
],
"combinator": "and"
},
"renameOutput": true,
"outputKey": "CI/CD (Ignore)"
},
{
"conditions": {
"options": {
"caseSensitive": true,
"leftValue": ""
},
"conditions": [
{
"leftValue": "={{ $('Parse Context').item.json.projectType }}",
"rightValue": "selfhosted",
"operator": {
"type": "string",
"operation": "equals"
}
}
],
"combinator": "and"
},
"renameOutput": true,
"outputKey": "Self-Hosted"
}
]
},
"options": {}
},
"type": "n8n-nodes-base.switch",
"typeVersion": 3.2,
"position": [
1568,
768
],
"id": "00786826-8d6b-4e17-aa7f-1afdca38d7a3",
"name": "Switch Type"
},
{
"parameters": {
"url": "=https://git.dk0.dev/api/v1/repos/denshooter/{{ $('Parse Context').item.json.repo }}/commits?limit=1",
"authentication": "genericCredentialType",
"genericAuthType": "httpHeaderAuth",
"options": {}
},
"type": "n8n-nodes-base.httpRequest",
"typeVersion": 4.4,
"position": [
1776,
560
],
"id": "9ef7f66b-3054-4765-b0a8-7ebb6aa353aa",
"name": "Get Last Commit",
"credentials": {
"httpHeaderAuth": {
"id": "YN3oIbok6Fjy5WNW",
"name": "gitea api"
}
}
},
{
"parameters": {
"url": "=https://git.dk0.dev/api/v1/repos/denshooter/{{ $('Parse Context').item.json.repo }}/contents/README.md",
"authentication": "genericCredentialType",
"genericAuthType": "httpHeaderAuth",
"options": {}
},
"type": "n8n-nodes-base.httpRequest",
"typeVersion": 4.4,
"position": [
1840,
672
],
"id": "114fece9-c5f1-4c6b-8272-6f39fb8ce24a",
"name": "Get README",
"credentials": {
"httpHeaderAuth": {
"id": "YN3oIbok6Fjy5WNW",
"name": "gitea api"
}
}
},
{
"parameters": {
"jsCode": "const ctx = $('Parse Context').first().json;\nconst commit = $('Get Last Commit').first().json[0];\nconst readme = $('Get README').first().json;\n\n// Decode README (base64)\nlet readmeText = '';\ntry {\n readmeText = Buffer.from(readme.content, 'base64').toString('utf8');\n // First 500 chars\n readmeText = readmeText.substring(0, 500).replace(/\\n/g, ' ');\n} catch (e) {\n readmeText = 'No README available';\n}\n\nconst commitMsg = commit?.commit?.message || 'No recent commits';\nconst commitAuthor = commit?.commit?.author?.name || 'Unknown';\n\nreturn [{\n json: {\n container: ctx.container,\n image: ctx.image,\n slug: ctx.slug,\n repo: ctx.repo,\n commitMsg,\n commitAuthor,\n readmeExcerpt: readmeText\n }\n}];"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [
2192,
480
],
"id": "8810426d-c146-42c9-8ec2-5d8f56934a1f",
"name": "Merge Git Data"
},
{
"parameters": {
"chatId": "145931600",
"text": "={{ \n'🚀 Neuer Deploy: ' + $json.container + '\\n' +\n'📦 ' + $json.image + '\\n\\n' +\n'📝 Letzter Commit:\\n' + $json.commitMsg + '\\n' +\n'👤 ' + $json.commitAuthor + '\\n\\n' +\n'📄 README:\\n' + $json.readmeExcerpt + '...\\n\\n' +\n'Was ist das Highlight?' \n}}",
"replyMarkup": "inlineKeyboard",
"inlineKeyboard": {
"rows": [
{
"row": {
"buttons": [
{
"text": "Selbst beschreiben",
"additionalFields": {
"callback_data": "={{ 'manual:' + $json.slug }}"
}
},
{
"text": "Auto-generieren",
"additionalFields": {
"callback_data": "={{ 'ignore:' + $json.slug }}"
}
}
]
}
}
]
},
"additionalFields": {}
},
"type": "n8n-nodes-base.telegram",
"typeVersion": 1.2,
"position": [
2544,
592
],
"id": "d4016ea3-7233-4926-af21-c7b07cc5f39d",
"name": "Ask via Telegram",
"webhookId": "313376d7-33a6-4c80-938b-e8ebc7ee2d11",
"credentials": {
"telegramApi": {
"id": "ADurvy9EKUDzbDdq",
"name": "DK0_Server"
}
}
},
{
"parameters": {
"promptType": "define",
"text": "=Du bist ein technischer Autor für dk0.dev.\n\nNeuer Self-Hosted Service:\nContainer: {{ $('Parse Context').item.json.container }}\nImage: {{ $('Parse Context').item.json.image }}\n\nErstelle eine Portfolio-Beschreibung:\n- Was macht die App\n- Warum Self-Hosting besser ist als Cloud\n- Wie sie in die Infrastruktur integriert ist\n\nAntworte NUR als JSON:\n{\n \"title_en\": \"Titel\",\n \"title_de\": \"Titel\",\n \"description_en\": \"4-6 Sätze\",\n \"description_de\": \"4-6 Sätze\",\n \"content_en\": \"2-3 Absätze Markdown\",\n \"content_de\": \"2-3 Absätze Markdown\",\n \"category\": \"selfhosted\",\n \"technologies\": [\"Docker\", \"...\"]\n}",
"batching": {}
},
"type": "@n8n/n8n-nodes-langchain.chainLlm",
"typeVersion": 1.9,
"position": [
1952,
864
],
"id": "0fd46a9d-40a9-4bb7-be5e-9b32b9a96381",
"name": "AI: Self-Hosted"
},
{
"parameters": {
"chatId": "145931600",
"text": "={{ \n'🆕 Self-Hosted Service: ' + $('Parse Context').first().json.serviceName + '\\n\\n' +\n'📝 ' + $json.data.title + '\\n\\n' +\n'Status: Draft erstellt (ID: ' + $json.data.id + ')\\n\\n' +\n'/publishproject' + $json.data.id + ' — Veröffentlichen\\n' + \n'/deleteproject' + $json.data.id + ' — Löschen' \n}}",
"additionalFields": {}
},
"type": "n8n-nodes-base.telegram",
"typeVersion": 1.2,
"position": [
2656,
848
],
"id": "bfaca06b-65ca-41a8-ba8a-1b1aef7ba12d",
"name": "Notify Selfhosted",
"webhookId": "a7d15c96-41e1-4242-9b5f-0382f4f0d31a",
"credentials": {
"telegramApi": {
"id": "ADurvy9EKUDzbDdq",
"name": "DK0_Server"
}
}
},
{
"parameters": {
"respondWith": "json",
"responseBody": "{ \"success\": true, \"message\": \"CI/CD container ignored\" }",
"options": {}
},
"type": "n8n-nodes-base.respondToWebhook",
"typeVersion": 1.5,
"position": [
1776,
960
],
"id": "d93818d9-64f9-4f57-ae84-c4280eeb50f0",
"name": "Respond (Ignore)"
},
{
"parameters": {
"respondWith": "json",
"responseBody": "{ \"success\": true }",
"options": {}
},
"type": "n8n-nodes-base.respondToWebhook",
"typeVersion": 1.5,
"position": [
2880,
768
],
"id": "4f1ad083-e73a-497c-a724-673205254b34",
"name": "Respond"
},
{
"parameters": {
"respondWith": "json",
"responseBody": "{ \"success\": true, \"message\": \"Project already exists\" }",
"options": {}
},
"type": "n8n-nodes-base.respondToWebhook",
"typeVersion": 1.5,
"position": [
1568,
960
],
"id": "0b93b3c7-c158-4389-af18-b418aa3b2239",
"name": "Respond (Exists)"
},
{
"parameters": {
"httpMethod": "POST",
"path": "docker-event",
"responseMode": "responseNode",
"options": {}
},
"type": "n8n-nodes-base.webhook",
"typeVersion": 2.1,
"position": [
688,
768
],
"id": "2b1c77d4-9f7f-4758-9e8e-f88195448ba3",
"name": "Webhook1",
"webhookId": "25d94042-2088-4e09-bfae-645db3d6803f"
},
{
"parameters": {
"model": "openrouter/free",
"options": {}
},
"type": "@n8n/n8n-nodes-langchain.lmChatOpenRouter",
"typeVersion": 1,
"position": [
1968,
1072
],
"id": "a450227f-f1e5-44f3-a90e-044420042fc4",
"name": "OpenRouter Chat Model1",
"credentials": {
"openRouterApi": {
"id": "8Kdy4RHHwMZ0Cn6x",
"name": "OpenRouter"
}
}
},
{
"parameters": {
"jsCode": "const raw = $input.first().json.text ?? \"\";\nconst match = raw.match(/\\{[\\s\\S]*\\}/);\nif (!match) throw new Error(\"No JSON found\");\nconst ai = JSON.parse(match[0]);\nreturn [{ json: ai }];"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [
2224,
848
],
"id": "ca78ecdd-5520-4540-969b-9e7b77bac3b4",
"name": "Parse JSON1"
},
{
"parameters": {
"jsCode": "const ai = $input.first().json;\nconst ctx = $('Parse Context').first().json;\n\nconst body = {\n slug: ctx.slug,\n status: \"draft\",\n featured: false,\n title: ai.title_en,\n category: ai.category,\n technologies: ai.technologies,\n tags: ai.technologies,\n date: new Date().toISOString().slice(0, 10),\n translations: {\n create: [\n {\n languages_code: \"en-US\",\n title: ai.title_en,\n description: ai.description_en,\n content: ai.content_en\n },\n {\n languages_code: \"de-DE\",\n title: ai.title_de,\n description: ai.description_de,\n content: ai.content_de\n }\n ]\n }\n};\n\nconst response = await this.helpers.httpRequest({\n method: \"POST\",\n url: \"https://cms.dk0.dev/items/projects\",\n headers: {\n \"Content-Type\": \"application/json\",\n \"Authorization\": \"Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB\"\n },\n body\n});\n\nreturn [{ json: response }];"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [
2448,
848
],
"id": "1ac0a31c-68a1-44df-a6b3-203698318cbf",
"name": "Add to Directus1"
}
],
"pinData": {},
"connections": {
"Webhook": {
"main": [
[
{
"node": "Kontext aufbereiten",
"type": "main",
"index": 0
}
]
]
},
"Kontext aufbereiten": {
"main": [
[
{
"node": "Search for Slug",
"type": "main",
"index": 0
}
]
]
},
"If": {
"main": [
[],
[
{
"node": "Basic LLM Chain",
"type": "main",
"index": 0
}
]
]
},
"Search for Slug": {
"main": [
[
{
"node": "If",
"type": "main",
"index": 0
}
]
]
},
"OpenRouter Chat Model": {
"ai_languageModel": [
[
{
"node": "Basic LLM Chain",
"type": "ai_languageModel",
"index": 0
}
]
]
},
"Basic LLM Chain": {
"main": [
[
{
"node": "Parse JSON",
"type": "main",
"index": 0
}
]
]
},
"Parse JSON": {
"main": [
[
{
"node": "Switch",
"type": "main",
"index": 0
}
]
]
},
"Add to Directus": {
"main": [
[
{
"node": "Send a text message",
"type": "main",
"index": 0
}
]
]
},
"Send a text message": {
"main": [
[
{
"node": "Respond to Webhook",
"type": "main",
"index": 0
}
]
]
},
"Switch": {
"main": [
[
{
"node": "Add to Directus",
"type": "main",
"index": 0
}
]
]
},
"Parse Context": {
"main": [
[
{
"node": "Check if Exists",
"type": "main",
"index": 0
}
]
]
},
"Check if Exists": {
"main": [
[
{
"node": "If New",
"type": "main",
"index": 0
}
]
]
},
"If New": {
"main": [
[
{
"node": "Switch Type",
"type": "main",
"index": 0
}
],
[
{
"node": "Respond (Exists)",
"type": "main",
"index": 0
}
]
]
},
"Switch Type": {
"main": [
[
{
"node": "Get Last Commit",
"type": "main",
"index": 0
}
],
[
{
"node": "Respond (Ignore)",
"type": "main",
"index": 0
}
],
[
{
"node": "AI: Self-Hosted",
"type": "main",
"index": 0
}
]
]
},
"Get Last Commit": {
"main": [
[
{
"node": "Get README",
"type": "main",
"index": 0
}
]
]
},
"Get README": {
"main": [
[
{
"node": "Merge Git Data",
"type": "main",
"index": 0
}
]
]
},
"Merge Git Data": {
"main": [
[
{
"node": "Ask via Telegram",
"type": "main",
"index": 0
}
]
]
},
"Ask via Telegram": {
"main": [
[
{
"node": "Respond",
"type": "main",
"index": 0
}
]
]
},
"AI: Self-Hosted": {
"main": [
[
{
"node": "Parse JSON1",
"type": "main",
"index": 0
}
]
]
},
"Notify Selfhosted": {
"main": [
[
{
"node": "Respond",
"type": "main",
"index": 0
}
]
]
},
"Webhook1": {
"main": [
[
{
"node": "Parse Context",
"type": "main",
"index": 0
}
]
]
},
"OpenRouter Chat Model1": {
"ai_languageModel": [
[
{
"node": "AI: Self-Hosted",
"type": "ai_languageModel",
"index": 0
}
]
]
},
"Parse JSON1": {
"main": [
[
{
"node": "Add to Directus1",
"type": "main",
"index": 0
}
]
]
},
"Add to Directus1": {
"main": [
[
{
"node": "Notify Selfhosted",
"type": "main",
"index": 0
}
]
]
}
},
"active": true,
"settings": {
"executionOrder": "v1",
"binaryMode": "separate",
"availableInMCP": false
},
"versionId": "1e2cf0ca-fe15-4a10-9716-30f85a2c2531",
"meta": {
"templateCredsSetupCompleted": true,
"instanceId": "cb28e4db755465d5826da179e87f69603d81f833414cc52c327be9183a217b8d"
},
"id": "RARR6MAlJSHAmBp8",
"tags": []
}

View File

@@ -1,417 +0,0 @@
{
"name": "Docker Event - Callback Handler",
"nodes": [
{
"parameters": {
"updates": [
"callback_query"
],
"additionalFields": {}
},
"type": "n8n-nodes-base.telegramTrigger",
"typeVersion": 1.2,
"position": [
-880,
288
],
"id": "a56a5174-3ccf-492f-810b-117be933560c",
"name": "Telegram Trigger",
"webhookId": "6e70b9ab-b76b-48dc-8e4d-5fe1bf0d7e39",
"credentials": {
"telegramApi": {
"id": "ADurvy9EKUDzbDdq",
"name": "DK0_Server"
}
}
},
{
"parameters": {
"jsCode": "const callback = $input.first().json;\nconst data = callback.callback_query?.data || '';\nconst chatId = callback.callback_query?.from?.id;\nconst messageId = callback.callback_query?.message?.message_id;\n\n// Parse: auto:slug, manual:slug, ignore:slug\nconst [action, slug] = data.split(':');\n\nreturn [{\n json: {\n action,\n slug,\n chatId,\n messageId,\n rawCallback: data\n }\n}];"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [
-656,
288
],
"id": "10e5a475-4194-4919-9186-1eb052fbd79b",
"name": "Parse Callback"
},
{
"parameters": {
"rules": {
"values": [
{
"conditions": {
"options": {
"caseSensitive": true,
"leftValue": ""
},
"conditions": [
{
"leftValue": "={{ $json.action }}",
"rightValue": "auto",
"operator": {
"type": "string",
"operation": "equals"
}
}
],
"combinator": "and"
},
"renameOutput": true,
"outputKey": "Auto"
},
{
"conditions": {
"options": {
"caseSensitive": true,
"leftValue": ""
},
"conditions": [
{
"leftValue": "={{ $json.action }}",
"rightValue": "manual",
"operator": {
"type": "string",
"operation": "equals"
}
}
],
"combinator": "and"
},
"renameOutput": true,
"outputKey": "Manual"
},
{
"conditions": {
"options": {
"caseSensitive": true,
"leftValue": ""
},
"conditions": [
{
"leftValue": "={{ $json.action }}",
"rightValue": "ignore",
"operator": {
"type": "string",
"operation": "equals"
}
}
],
"combinator": "and"
},
"renameOutput": true,
"outputKey": "Ignore"
}
]
},
"options": {}
},
"type": "n8n-nodes-base.switch",
"typeVersion": 3.2,
"position": [
-448,
288
],
"id": "a533e527-b3c5-4946-9a26-6f499c7dd6c5",
"name": "Switch Action"
},
{
"parameters": {
"url": "=https://cms.dk0.dev/items/projects?filter[slug][_eq]={{ $json.slug }}&limit=1",
"authentication": "predefinedCredentialType",
"nodeCredentialType": "httpBearerAuth",
"options": {}
},
"type": "n8n-nodes-base.httpRequest",
"typeVersion": 4.4,
"position": [
-224,
80
],
"id": "9fc55503-e890-4074-9823-f07001b6948a",
"name": "Get Project from CMS"
},
{
"parameters": {
"url": "=https://git.dk0.dev/api/v1/repos/denshooter/{{ $json.slug }}/commits?limit=3",
"authentication": "genericCredentialType",
"genericAuthType": "httpHeaderAuth",
"options": {}
},
"type": "n8n-nodes-base.httpRequest",
"typeVersion": 4.4,
"position": [
0,
0
],
"id": "a3fda0d9-0cc9-4744-be3e-9a95ef44dfb4",
"name": "Get Commits"
},
{
"parameters": {
"url": "=https://git.dk0.dev/api/v1/repos/denshooter/{{ $json.slug }}/contents/README.md",
"authentication": "genericCredentialType",
"genericAuthType": "httpHeaderAuth",
"options": {}
},
"type": "n8n-nodes-base.httpRequest",
"typeVersion": 4.4,
"position": [
0,
128
],
"id": "7106b8c9-fb20-46d9-9e4e-06882115bf7a",
"name": "Get README"
},
{
"parameters": {
"model": "openrouter/free",
"options": {}
},
"type": "@n8n/n8n-nodes-langchain.lmChatOpenRouter",
"typeVersion": 1,
"position": [
448,
192
],
"id": "9acce2c3-1a26-450f-a263-0dc3a1f1e3cf",
"name": "OpenRouter Chat Model"
},
{
"parameters": {
"promptType": "define",
"text": "=Du bist ein technischer Autor für das Portfolio von Dennis (dk0.dev).\n\nNeues eigenes Projekt deployed:\nRepo: {{ $('Parse Callback').item.json.slug }}\n\nREADME:\n{{ $('Get README').first().json.content ? Buffer.from($('Get README').first().json.content, 'base64').toString('utf8').substring(0, 1000) : 'Kein README' }}\n\nLetzte Commits:\n{{ $('Get Commits').first().json.map(c => '- ' + c.commit.message).join('\\n') }}\n\nErstelle eine Portfolio-Beschreibung:\n- Was macht das Projekt (Features, Zweck)\n- Tech-Stack und Architektur\n- Highlights aus den Commits\n- Warum ist es cool/interessant\n\nKategorie: webdev (wenn Web-App), automation (wenn Tool/Script), oder selfhosted\n\nAntworte NUR als JSON:\n{\n \"title_en\": \"Aussagekräftiger Titel\",\n \"title_de\": \"Aussagekräftiger Titel\",\n \"description_en\": \"4-6 Sätze\",\n \"description_de\": \"4-6 Sätze\",\n \"content_en\": \"2-3 Absätze Markdown mit technischen Details\",\n \"content_de\": \"2-3 Absätze Markdown mit technischen Details\",\n \"category\": \"webdev|automation|selfhosted\",\n \"technologies\": [\"Next.js\", \"Docker\", \"...\"]\n}",
"batching": {}
},
"type": "@n8n/n8n-nodes-langchain.chainLlm",
"typeVersion": 1.9,
"position": [
224,
80
],
"id": "2b011cf8-6ed3-4cb1-ab6f-7727912864fc",
"name": "AI: Generate Description"
},
{
"parameters": {
"jsCode": "const raw = $input.first().json.text ?? \"\";\nconst match = raw.match(/\\{[\\s\\S]*\\}/);\nif (!match) throw new Error(\"No JSON found\");\nconst ai = JSON.parse(match[0]);\nreturn [{ json: ai }];"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [
448,
80
],
"id": "0cbdcf6e-e5d4-460e-b345-b6d47deed051",
"name": "Parse JSON"
},
{
"parameters": {
"jsCode": "const ai = $input.first().json;\nconst ctx = $('Parse Callback').first().json;\n\nconst body = {\n slug: ctx.slug,\n status: \"draft\",\n featured: false,\n title: ai.title_en,\n category: ai.category,\n technologies: ai.technologies,\n tags: ai.technologies,\n date: new Date().toISOString().slice(0, 10),\n translations: {\n create: [\n {\n languages_code: \"en-US\",\n title: ai.title_en,\n description: ai.description_en,\n content: ai.content_en\n },\n {\n languages_code: \"de-DE\",\n title: ai.title_de,\n description: ai.description_de,\n content: ai.content_de\n }\n ]\n }\n};\n\nconst response = await this.helpers.httpRequest({\n method: \"POST\",\n url: \"https://cms.dk0.dev/items/projects\",\n headers: {\n \"Content-Type\": \"application/json\",\n \"Authorization\": \"Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB\"\n },\n body\n});\n\nreturn [{ json: response }];"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [
672,
80
],
"id": "70aecf97-6b70-4f03-99e3-9ee44fc0830b",
"name": "Add to Directus"
},
{
"parameters": {
"chatId": "={{ $('Parse Callback').item.json.chatId }}",
"text": "={{ \n'✅ Projekt erstellt: ' + $json.data.title + '\\n\\n' +\n'📝 ' + $('Parse JSON').first().json.description_de.substring(0, 200) + '...\\n\\n' +\n'Status: Draft (ID: ' + $json.data.id + ')\\n\\n' +\n'/publishproject' + $json.data.id + ' — Veröffentlichen\\n' + \n'/deleteproject' + $json.data.id + ' — Löschen' \n}}",
"additionalFields": {}
},
"type": "n8n-nodes-base.telegram",
"typeVersion": 1.2,
"position": [
880,
80
],
"id": "9a353247-7d25-4330-9cbf-580599428ae1",
"name": "Notify Success",
"webhookId": "b1d7284d-c2e5-4e87-b65d-272f1b9b8d6d"
},
{
"parameters": {
"chatId": "={{ $json.chatId }}",
"text": "✍️ OK, schreib mir jetzt was das Projekt macht (4-6 Sätze).\n\nIch formatiere das dann schön und erstelle einen Draft.",
"additionalFields": {}
},
"type": "n8n-nodes-base.telegram",
"typeVersion": 1.2,
"position": [
-224,
288
],
"id": "9160b847-5f07-4d64-9488-faeaeca926b9",
"name": "Ask for Manual Input",
"webhookId": "c4cb518d-a2e2-48af-b9b6-c3f645fd37db"
},
{
"parameters": {
"chatId": "={{ $json.chatId }}",
"text": "❌ OK, ignoriert.",
"additionalFields": {}
},
"type": "n8n-nodes-base.telegram",
"typeVersion": 1.2,
"position": [
-224,
480
],
"id": "1624b6f1-8202-4fd2-bd0a-52fa039ca696",
"name": "Confirm Ignore",
"webhookId": "4c5248f1-4420-403c-a506-2e1968c5579d",
"credentials": {
"telegramApi": {
"id": "ADurvy9EKUDzbDdq",
"name": "DK0_Server"
}
}
}
],
"pinData": {},
"connections": {
"Telegram Trigger": {
"main": [
[
{
"node": "Parse Callback",
"type": "main",
"index": 0
}
]
]
},
"Parse Callback": {
"main": [
[
{
"node": "Switch Action",
"type": "main",
"index": 0
}
]
]
},
"Switch Action": {
"main": [
[
{
"node": "Get Project from CMS",
"type": "main",
"index": 0
}
],
[
{
"node": "Ask for Manual Input",
"type": "main",
"index": 0
}
],
[
{
"node": "Confirm Ignore",
"type": "main",
"index": 0
}
]
]
},
"Get Project from CMS": {
"main": [
[
{
"node": "Get Commits",
"type": "main",
"index": 0
}
]
]
},
"Get Commits": {
"main": [
[
{
"node": "Get README",
"type": "main",
"index": 0
}
]
]
},
"Get README": {
"main": [
[
{
"node": "AI: Generate Description",
"type": "main",
"index": 0
}
]
]
},
"OpenRouter Chat Model": {
"ai_languageModel": [
[
{
"node": "AI: Generate Description",
"type": "ai_languageModel",
"index": 0
}
]
]
},
"AI: Generate Description": {
"main": [
[
{
"node": "Parse JSON",
"type": "main",
"index": 0
}
]
]
},
"Parse JSON": {
"main": [
[
{
"node": "Add to Directus",
"type": "main",
"index": 0
}
]
]
},
"Add to Directus": {
"main": [
[
{
"node": "Notify Success",
"type": "main",
"index": 0
}
]
]
}
},
"active": false,
"settings": {
"executionOrder": "v1",
"binaryMode": "separate",
"availableInMCP": false
},
"versionId": "4636a407-7f8e-4833-9345-9d3296ec9b74",
"meta": {
"instanceId": "cb28e4db755465d5826da179e87f69603d81f833414cc52c327be9183a217b8d"
},
"id": "abnrtUuJ7BAWv9Hm",
"tags": []
}

View File

@@ -1,305 +0,0 @@
{
"name": "Docker Event",
"nodes": [
{
"parameters": {
"httpMethod": "POST",
"path": "docker-event",
"responseMode": "responseNode",
"options": {}
},
"type": "n8n-nodes-base.webhook",
"typeVersion": 2.1,
"position": [
0,
-224
],
"id": "870fa550-42f6-4e19-a796-f1f044b0cdc8",
"name": "Webhook",
"webhookId": "e147d70b-79d8-44fd-bbe8-8274cf905b11"
},
{
"parameters": {
"jsCode": "const data = $input.first().json;\n\nconst container = data.container ?? data.body?.container ?? '';\nconst image = data.image ?? data.body?.image ?? '';\nconst timestamp = data.timestamp ?? data.body?.timestamp ?? '';\n\nconst slug = container.toLowerCase().replace(/[^a-z0-9]+/g, '-');\n\nconst serviceName = container.replace(/[-_]/g, ' ');\n\nreturn [{\n json: {\n container,\n image,\n serviceName,\n timestamp,\n slug \n }\n}];"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [
224,
-224
],
"id": "aaa6a678-1ad3-4f82-9b01-37e21b47b189",
"name": "Kontext aufbereiten"
},
{
"parameters": {
"conditions": {
"options": {
"caseSensitive": true,
"leftValue": "",
"typeValidation": "loose",
"version": 3
},
"conditions": [
{
"id": "ebe26f0c-d5a7-45c9-9747-afc75b57a41c",
"leftValue": "={{ $json.data }}",
"rightValue": "[]",
"operator": {
"type": "string",
"operation": "notEndsWith"
}
}
],
"combinator": "and"
},
"looseTypeValidation": true,
"options": {}
},
"type": "n8n-nodes-base.if",
"typeVersion": 2.3,
"position": [
672,
-224
],
"id": "62197a33-5169-48e1-9539-57c047efb108",
"name": "If"
},
{
"parameters": {
"url": "=https://cms.dk0.dev/items/projects?filter[slug][_eq]={{ $json.slug }}&limit=1",
"authentication": "predefinedCredentialType",
"nodeCredentialType": "httpBearerAuth",
"options": {}
},
"type": "n8n-nodes-base.httpRequest",
"typeVersion": 4.4,
"position": [
448,
-224
],
"id": "db783886-06b5-4473-8907-dd6c655aa3dd",
"name": "Search for Slug",
"credentials": {
"httpBearerAuth": {
"id": "ZtI5e08iryR9m6FG",
"name": "Directus"
}
}
},
{
"parameters": {
"model": "openrouter/free",
"options": {}
},
"type": "@n8n/n8n-nodes-langchain.lmChatOpenRouter",
"typeVersion": 1,
"position": [
976,
16
],
"id": "b9130ff4-359b-4736-9442-1b0ca7d31877",
"name": "OpenRouter Chat Model",
"credentials": {
"openRouterApi": {
"id": "8Kdy4RHHwMZ0Cn6x",
"name": "OpenRouter"
}
}
},
{
"parameters": {
"promptType": "define",
"text": "= Du bist ein technischer Autor für das Self-Hosting Portfolio von Dennis auf dk0.dev.\n Ein neuer Service wurde auf dem Server deployed:\n\n Container:{{ $('Kontext aufbereiten').item.json.container }}\n Image: {{ $('Kontext aufbereiten').item.json.image }}\n Service: {{ $('Kontext aufbereiten').item.json.serviceName }}\n\n Aufgabe:\n 1. Erkenne ob es sich um ein EIGENES Projekt (z.B. Image enthält \"denshooter\", \"dk0\", \"portfolio\") oder eine SELF-HOSTED\n App (z.B. plausible, nextcloud, gitea, etc.) handelt.\n 2. Erstelle eine ausführliche Projektbeschreibung.\n\n Für EIGENE Projekte:\n - Beschreibe was die App macht, welche Probleme sie löst, welche Features sie hat\n - Erwähne den Tech-Stack und architektonische Entscheidungen\n - category: \"webdev\" oder \"automation\"\n\n Für SELF-HOSTED Apps:\n - Beschreibe was die App macht und warum Self-Hosting besser ist als die Cloud-Alternative\n - Erwähne Vorteile wie Datenschutz, Kontrolle, Kosten\n - Beschreibe kurz wie sie in die bestehende Infrastruktur integriert ist (Docker, Reverse Proxy, etc.)\n - category: \"selfhosted\"\n\n Antworte NUR als valides JSON, kein anderer Text:\n {\n \"type\": \"own\" oder \"selfhosted\",\n \"title_en\": \"Aussagekräftiger Titel auf Englisch\",\n \"title_de\": \"Aussagekräftiger Titel auf Deutsch\",\n \"description_en\": \"Ausführliche Beschreibung, 4-6 Sätze. Was macht es, warum ist es wichtig, was sind die Highlights.\",\n \"description_de\": \"Ausführliche Beschreibung, 4-6 Sätze. Was macht es, warum ist es wichtig, was sind die Highlights.\",\n \"content_en\": \"Noch detaillierterer Text, 2-3 Absätze in Markdown. Features, Setup, technische Details.\",\n \"content_de\": \"Noch detaillierterer Text, 2-3 Absätze in Markdown. Features, Setup, technische Details.\",\n \"category\": \"selfhosted\" oder \"webdev\" oder \"automation\",\n \"technologies\": [\"Docker\", \"und alle anderen relevanten Technologien\"]\n ",
"batching": {}
},
"type": "@n8n/n8n-nodes-langchain.chainLlm",
"typeVersion": 1.9,
"position": [
896,
-224
],
"id": "77d46075-3342-4e93-8806-07087a2389dc",
"name": "Basic LLM Chain"
},
{
"parameters": {
"jsCode": "const raw = $input.first().json.text ?? \"\";\n\nconst match = raw.match(/\\{[\\s\\S]*\\}/);\nif (!match) throw new Error(\"No JSON found\");\n\nconst ai = JSON.parse(match[0]);\n\nreturn [\n {\n json: ai,\n },\n];\n"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [
1248,
-224
],
"id": "de5ed311-0d46-4677-963c-711a6ad514e9",
"name": "Parse JSON"
},
{
"parameters": {
"jsCode": "const ai = $('Parse JSON').first().json;\n const ctx = $('Kontext aufbereiten').first().json;\n\n const body = {\n slug: ctx.slug,\n status: \"draft\",\n featured: false,\n title: ai.title_en,\n category: ai.category,\n technologies: ai.technologies,\n tags: ai.technologies,\n date: new Date().toISOString().slice(0, 10),\n translations: {\n create: [\n {\n languages_code: \"en-US\",\n title: ai.title_en,\n description: ai.description_en,\n content: ai.content_en\n },\n {\n languages_code: \"de-DE\",\n title: ai.title_de,\n description: ai.description_de,\n content: ai.content_de\n }\n ]\n }\n };\n\n const response = await this.helpers.httpRequest({\n method: \"POST\",\n url: \"https://cms.dk0.dev/items/projects\",\n headers: {\n \"Content-Type\": \"application/json\",\n \"Authorization\": \"Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB\"\n },\n body\n });\n\n return [{ json: response }];"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [
1472,
-224
],
"id": "c47b915d-e4d7-43e9-8ee3-b41389896fa7",
"name": "Add to Directus"
},
{
"parameters": {
"respondWith": "json",
"responseBody": "{ \"success\": true }",
"options": {}
},
"type": "n8n-nodes-base.respondToWebhook",
"typeVersion": 1.5,
"position": [
1920,
-224
],
"id": "6cf8f30d-1352-466f-9163-9b4f16b972e0",
"name": "Respond to Webhook"
},
{
"parameters": {
"chatId": "145931600",
"text": "={{ \n'🆕 Neuer Service erkannt!\\n\\n' +\n'📦 ' + $('Kontext aufbereiten').first().json.container + '\\n' +\n'🐳 ' + $('Kontext aufbereiten').first().json.image + '\\n\\n' +\n'📝 ' + $('Parse JSON').first().json.title_de + '\\n' + \n$('Parse JSON').first().json.description_de + '\\n\\n' +\n'Status: Draft in Directus erstellt (ID: ' + $json.data.id + ')\\n\\n' +\n('/publishproject_' + $json.data.id).replace(/_/g, '\\\\_') + ' — Veröffentlichen\\n' + \n('/deleteproject_' + $json.data.id).replace(/_/g, '\\\\_') + ' — Löschen' \n}}",
"additionalFields": {}
},
"type": "n8n-nodes-base.telegram",
"typeVersion": 1.2,
"position": [
1696,
-224
],
"id": "b29de3ec-b1ca-40c3-8493-af44e5372fd2",
"name": "Send a text message",
"webhookId": "c02ccf69-16dc-436e-b1cc-f8fa9dd8d33f",
"credentials": {
"telegramApi": {
"id": "ADurvy9EKUDzbDdq",
"name": "DK0_Server"
}
}
}
],
"pinData": {},
"connections": {
"Webhook": {
"main": [
[
{
"node": "Kontext aufbereiten",
"type": "main",
"index": 0
}
]
]
},
"Kontext aufbereiten": {
"main": [
[
{
"node": "Search for Slug",
"type": "main",
"index": 0
}
]
]
},
"If": {
"main": [
[],
[
{
"node": "Basic LLM Chain",
"type": "main",
"index": 0
}
]
]
},
"Search for Slug": {
"main": [
[
{
"node": "If",
"type": "main",
"index": 0
}
]
]
},
"OpenRouter Chat Model": {
"ai_languageModel": [
[
{
"node": "Basic LLM Chain",
"type": "ai_languageModel",
"index": 0
}
]
]
},
"Basic LLM Chain": {
"main": [
[
{
"node": "Parse JSON",
"type": "main",
"index": 0
}
]
]
},
"Parse JSON": {
"main": [
[
{
"node": "Add to Directus",
"type": "main",
"index": 0
}
]
]
},
"Add to Directus": {
"main": [
[
{
"node": "Send a text message",
"type": "main",
"index": 0
}
]
]
},
"Send a text message": {
"main": [
[
{
"node": "Respond to Webhook",
"type": "main",
"index": 0
}
]
]
}
},
"active": true,
"settings": {
"executionOrder": "v1",
"binaryMode": "separate",
"availableInMCP": false
},
"versionId": "91b63f71-f5b7-495f-95ba-cbf999bb9a19",
"meta": {
"templateCredsSetupCompleted": true,
"instanceId": "cb28e4db755465d5826da179e87f69603d81f833414cc52c327be9183a217b8d"
},
"id": "RARR6MAlJSHAmBp8",
"tags": []
}

View File

@@ -1,278 +0,0 @@
# 🎯 Telegram CMS Bot - Quick Reference
## 📱 Commands Cheat Sheet
### Core Commands
```
/start # Dashboard with stats
/list projects # Show all projects
/list books # Show all book reviews
/search <term> # Search across all content
/stats # Detailed analytics
```
### Item Management
```
/preview<ID> # View item details (both languages)
/publish<ID> # Publish item (auto-detect type)
/delete<ID> # Delete item (auto-detect type)
/deletereview<ID> # Remove review translations only
```
### Legacy Commands (still supported)
```
/publishproject<ID> # Publish specific project
/publishbook<ID> # Publish specific book
/deleteproject<ID> # Delete specific project
/deletebook<ID> # Delete specific book
```
### AI Review Creation
```
.review <HARDCOVER_ID> <RATING> <YOUR_THOUGHTS>
```
**Example:**
```
.review 12345 5 Absolutely loved this book! The character development was outstanding and the plot kept me engaged throughout. Highly recommend for anyone interested in fantasy literature.
```
**Result:**
- Creates EN + DE reviews via AI
- Sets rating (1-5 stars)
- Saves as draft in CMS
- Provides publish/delete buttons
---
## 🎨 Response Format
All responses use Markdown formatting with emojis:
### Dashboard
```
🎯 DK0 Portfolio CMS
📊 Stats:
• Draft Projects: 3
• Draft Reviews: 2
💡 Quick Actions:
/list projects - View all projects
...
```
### List View
```
📋 PROJECTS (Page 1)
1. Next.js Portfolio
Category: Web Development
Status: draft
/preview42 | /publish42 | /delete42
```
### Preview
```
👁️ Preview #42
📁 Type: Project
🔖 Slug: nextjs-portfolio
🏷️ Category: Web Development
📊 Status: draft
🇬🇧 EN:
Title: Next.js Portfolio
Description: Modern portfolio built with...
🇩🇪 DE:
Title: Next.js Portfolio
Description: Modernes Portfolio erstellt mit...
Actions:
/publish42 - Publish
/delete42 - Delete
```
---
## 🔍 Auto-Detection
The workflow automatically detects item types:
| Command | Behavior |
|---------|----------|
| `/preview42` | Checks projects → checks books |
| `/publish42` | Checks projects → checks books |
| `/delete42` | Checks projects → checks books |
No need to specify collection type!
---
## 💡 Tips & Tricks
1. **Quick Publishing:**
```
/list projects # Get item ID
/preview42 # Review content
/publish42 # Publish
```
2. **Bulk Review:**
```
/list books # See all books
/preview* # Check each one
/publish* # Publish ready ones
```
3. **Search Before Create:**
```
/search "react" # Check existing content
# Then create new if needed
```
4. **AI Review Workflow:**
```
.review 12345 5 My thoughts here
# AI generates EN + DE versions
/preview<ID> # Review AI output
/publish<ID> # Publish if good
/deletereview<ID> # Remove & retry if bad
```
---
## ⚠️ Common Issues
### ❌ "Item not found"
- Verify ID is correct
- Check if item exists in CMS
- Try /search to find correct ID
### ❌ "Error loading dashboard"
- Directus might be down
- Check network connection
- Try again in 30 seconds
### ❌ AI review fails
- Verify Hardcover ID exists
- Check rating is 1-5
- Ensure you provided text
### ❌ No response from bot
- Bot might be restarting
- Check n8n workflow is active
- Wait 1 minute and retry
---
## 📊 Status Values
| Status | Meaning | Action |
|--------|---------|--------|
| `draft` | Not visible on site | Use `/publish` |
| `published` | Live on dk0.dev | ✅ Done |
| `archived` | Hidden but kept | Use `/delete` to remove |
---
## 🎯 Workflow Logic
```mermaid
graph TD
A[Telegram Message] --> B[Parse Command]
B --> C{Command Type?}
C -->|/start| D[Dashboard]
C -->|/list| E[List Handler]
C -->|/search| F[Search Handler]
C -->|/stats| G[Stats Handler]
C -->|/preview| H[Preview Handler]
C -->|/publish| I[Publish Handler]
C -->|/delete| J[Delete Handler]
C -->|/deletereview| K[Delete Review]
C -->|.review| L[Create Review AI]
C -->|unknown| M[Help Message]
D --> N[Send Message]
E --> N
F --> N
G --> N
H --> N
I --> N
J --> N
K --> N
L --> N
M --> N
```
---
## 🚀 Performance
- **Dashboard:** ~1-2s
- **List:** ~1-2s (5 items)
- **Search:** ~1-2s
- **Preview:** ~1s
- **Publish/Delete:** ~1s
- **AI Review:** ~3-5s
---
## 📝 Examples
### Complete Workflow Example
```bash
# Step 1: Check what's available
/start
# Step 2: List projects
/list projects
# Step 3: Preview one
/preview42
# Step 4: Looks good? Publish!
/publish42
# Step 5: Create a book review
.review 12345 5 Amazing book about TypeScript!
# Step 6: Check the generated review
/preview<ID>
# Step 7: Publish it
/publish<ID>
# Step 8: Get overall stats
/stats
```
---
## 🔗 Integration Points
| System | Purpose | Endpoint |
|--------|---------|----------|
| Directus | CMS data | https://cms.dk0.dev |
| OpenRouter | AI reviews | https://openrouter.ai |
| Telegram | Bot interface | DK0_Server |
| Portfolio | Live site | https://dk0.dev |
---
## 📞 Support
**Problems?** Check:
1. n8n workflow logs
2. Directus API status
3. Telegram bot status
4. This quick reference
**Still stuck?** Contact Dennis Konkol
---
**Last Updated:** 2025-01-21
**Version:** 1.0.0
**Status:** ✅ Production Ready

View File

@@ -1,372 +0,0 @@
# ✅ Telegram CMS Workflow - Testing Checklist
## Pre-Deployment Tests
### 1. Import Verification
- [ ] Import workflow JSON into n8n successfully
- [ ] Verify all 14 nodes are present
- [ ] Check all connections are intact
- [ ] Confirm credentials are linked (DK0_Server)
- [ ] Activate workflow without errors
### 2. Command Parsing Tests
#### Basic Commands
- [ ] Send `/start` → Receives dashboard with stats
- [ ] Send `/list projects` → Gets paginated project list
- [ ] Send `/list books` → Gets book review list
- [ ] Send `/search test` → Gets search results
- [ ] Send `/stats` → Gets statistics dashboard
#### Item Management
- [ ] Send `/preview<ID>` → Gets item preview with translations
- [ ] Send `/publish<ID>` → Successfully publishes item
- [ ] Send `/delete<ID>` → Successfully deletes item
- [ ] Send `/deletereview<ID>` → Removes review translations
#### Legacy Commands (Backward Compatibility)
- [ ] Send `/publishproject<ID>` → Works correctly
- [ ] Send `/publishbook<ID>` → Works correctly
- [ ] Send `/deleteproject<ID>` → Works correctly
- [ ] Send `/deletebook<ID>` → Works correctly
#### AI Review Creation
- [ ] Send `.review 12345 5 Test review` → Creates review with AI
- [ ] Send `/review 12345 5 Test review` → Also works with slash
- [ ] Verify EN review is generated
- [ ] Verify DE review is generated
- [ ] Check rating is set correctly
- [ ] Confirm status is "draft"
#### Error Handling
- [ ] Send `/unknown` → Gets help message
- [ ] Send `/preview999999` → Gets "not found" error
- [ ] Send `.review invalid` → Gets format error
- [ ] Test with empty search term
- [ ] Test with special characters in search
---
## Node-by-Node Tests
### 1. Telegram Trigger
- [ ] Receives messages correctly
- [ ] Extracts chat ID
- [ ] Passes data to Parse Command node
### 2. Parse Command
- [ ] Correctly identifies `/start` command
- [ ] Parses `/list projects` vs `/list books`
- [ ] Extracts search query from `/search <term>`
- [ ] Parses item IDs from commands
- [ ] Handles `.review` with correct regex
- [ ] Returns unknown action for invalid commands
### 3. Command Router (Switch)
- [ ] Routes to Dashboard Handler for "start"
- [ ] Routes to List Handler for "list"
- [ ] Routes to Search Handler for "search"
- [ ] Routes to Stats Handler for "stats"
- [ ] Routes to Preview Handler for "preview"
- [ ] Routes to Publish Handler for "publish"
- [ ] Routes to Delete Handler for "delete"
- [ ] Routes to Delete Review Handler for "delete_review"
- [ ] Routes to Create Review Handler for "create_review"
- [ ] Routes to Unknown Handler for unrecognized commands
### 4. Dashboard Handler
- [ ] Fetches draft projects count from Directus
- [ ] Fetches draft books count from Directus
- [ ] Formats message with stats
- [ ] Includes all command examples
- [ ] Uses Markdown formatting
- [ ] Handles API errors gracefully
### 5. List Handler
- [ ] Supports both "projects" and "books" types
- [ ] Limits to 5 items per page
- [ ] Shows correct fields (title, category, status, date)
- [ ] Includes action buttons for each item
- [ ] Displays pagination hint if more items exist
- [ ] Handles empty results
- [ ] Catches and reports errors
### 6. Search Handler
- [ ] Searches projects by title
- [ ] Searches books by title
- [ ] Uses Directus `_contains` filter
- [ ] Groups results by type
- [ ] Limits to 5 results per collection
- [ ] Handles no results found
- [ ] URL-encodes search query
- [ ] Error handling works
### 7. Stats Handler
- [ ] Calculates total project count
- [ ] Breaks down by status (published/draft/archived)
- [ ] Calculates book statistics
- [ ] Computes average rating correctly
- [ ] Groups projects by category
- [ ] Sorts categories by count
- [ ] Formats with emojis
- [ ] Handles empty data
### 8. Preview Handler
- [ ] Auto-detects projects first
- [ ] Falls back to books if not found
- [ ] Shows both EN and DE translations
- [ ] Displays metadata (status, category, rating)
- [ ] Truncates long text with "..."
- [ ] Provides action buttons
- [ ] Returns 404 if not found
- [ ] Error messages are clear
### 9. Publish Handler
- [ ] Tries projects collection first
- [ ] Falls back to books collection
- [ ] Updates status to "published"
- [ ] Returns success message
- [ ] Handles 404 gracefully
- [ ] Uses correct HTTP method (PATCH)
- [ ] Includes auth token
- [ ] Error handling works
### 10. Delete Handler
- [ ] Tries projects collection first
- [ ] Falls back to books collection
- [ ] Permanently removes item
- [ ] Returns confirmation message
- [ ] Handles 404 gracefully
- [ ] Uses correct HTTP method (DELETE)
- [ ] Includes auth token
- [ ] Error handling works
### 11. Delete Review Handler
- [ ] Fetches book review by ID
- [ ] Gets translation IDs
- [ ] Deletes all translations
- [ ] Keeps book entry intact
- [ ] Reports count of deleted translations
- [ ] Handles missing reviews
- [ ] Error handling works
### 12. Create Review Handler
- [ ] Fetches book by Hardcover ID
- [ ] Builds AI prompt correctly
- [ ] Calls OpenRouter API
- [ ] Parses JSON from AI response
- [ ] Handles malformed AI output
- [ ] Creates EN translation
- [ ] Creates DE translation
- [ ] Sets rating correctly
- [ ] Sets status to "draft"
- [ ] Returns formatted message with preview
- [ ] Provides action buttons
- [ ] Error handling works
### 13. Unknown Command Handler
- [ ] Returns help message
- [ ] Lists all available commands
- [ ] Uses Markdown formatting
- [ ] Includes examples
### 14. Send Telegram Message
- [ ] Uses chat ID from input
- [ ] Sends message text correctly
- [ ] Applies Markdown parse mode
- [ ] Uses correct credentials
- [ ] Returns successfully
---
## Integration Tests
### Directus API
- [ ] Authentication works with token
- [ ] GET requests succeed
- [ ] PATCH requests update items
- [ ] DELETE requests remove items
- [ ] GraphQL queries work (if used)
- [ ] Translation relationships load
- [ ] Filters work correctly
- [ ] Aggregations return data
- [ ] Pagination parameters work
### OpenRouter AI
- [ ] API key is valid
- [ ] Model name is correct
- [ ] Prompt format works
- [ ] JSON parsing succeeds
- [ ] Fallback handles non-JSON
- [ ] Rate limits are respected
- [ ] Timeout is reasonable
### Telegram Bot
- [ ] Bot token is valid
- [ ] Chat ID is correct
- [ ] Messages send successfully
- [ ] Markdown formatting works
- [ ] Emojis display correctly
- [ ] Long messages don't truncate
- [ ] Error messages are readable
---
## Error Scenarios
### API Failures
- [ ] Directus is unreachable → User-friendly error
- [ ] Directus returns 401 → Auth error message
- [ ] Directus returns 404 → Item not found message
- [ ] Directus returns 500 → Generic error message
- [ ] OpenRouter fails → Review creation fails gracefully
- [ ] Telegram API fails → Workflow logs error
### Data Issues
- [ ] Empty search results → "No results" message
- [ ] Missing translations → Shows available languages
- [ ] Invalid item ID → "Not found" error
- [ ] Malformed AI response → Uses fallback text
- [ ] No Hardcover ID match → Clear error message
### User Errors
- [ ] Invalid command format → Help message
- [ ] Missing parameters → Format example
- [ ] Wrong item type → Auto-detection handles it
- [ ] Non-numeric ID → Validation error
---
## Performance Tests
- [ ] Dashboard loads in < 2 seconds
- [ ] List loads in < 2 seconds
- [ ] Search completes in < 2 seconds
- [ ] Preview loads in < 1 second
- [ ] Publish/delete complete in < 1 second
- [ ] AI review generates in < 5 seconds
- [ ] No timeout errors with normal load
- [ ] Concurrent requests don't conflict
---
## Security Tests
- [ ] API token not exposed in logs
- [ ] Error messages don't leak sensitive data
- [ ] Chat ID validation works
- [ ] Only authorized user can access (check bot settings)
- [ ] SQL injection is impossible (using REST API)
- [ ] XSS is prevented (Markdown escaping)
---
## User Experience Tests
- [ ] Messages are easy to read
- [ ] Emojis enhance clarity
- [ ] Action buttons are clear
- [ ] Error messages are helpful
- [ ] Success messages are satisfying
- [ ] Command examples are accurate
- [ ] Help message is comprehensive
---
## Regression Tests
After any changes:
- [ ] Re-run all command parsing tests
- [ ] Verify all handlers still work
- [ ] Check error handling didn't break
- [ ] Confirm AI review still generates
- [ ] Test backward compatibility
---
## Deployment Checklist
### Pre-Deployment
- [ ] All tests pass
- [ ] Workflow is exported
- [ ] Documentation is updated
- [ ] Credentials are configured
- [ ] Environment variables set
### Deployment
- [ ] Import workflow to production n8n
- [ ] Activate workflow
- [ ] Test `/start` command
- [ ] Monitor execution logs
- [ ] Verify Directus connection
- [ ] Check Telegram bot responds
### Post-Deployment
- [ ] Run smoke tests (start, list, search)
- [ ] Create test review
- [ ] Publish test item
- [ ] Monitor for 24 hours
- [ ] Check error logs
- [ ] Confirm no false positives
---
## Monitoring
Daily:
- [ ] Check n8n execution logs
- [ ] Review error count
- [ ] Verify success rate > 95%
Weekly:
- [ ] Test all commands manually
- [ ] Review API usage
- [ ] Check for rate limiting
- [ ] Update this checklist
Monthly:
- [ ] Full regression test
- [ ] Update documentation
- [ ] Review and optimize queries
- [ ] Check for n8n updates
---
## Rollback Plan
If issues occur:
1. Deactivate workflow in n8n
2. Revert to previous version
3. Investigate logs
4. Fix in staging
5. Re-test thoroughly
6. Deploy again
---
## Sign-off
- [ ] All critical tests pass
- [ ] Documentation complete
- [ ] Team notified
- [ ] Backup created
- [ ] Ready for production
**Tested by:** _________________
**Date:** _________________
**Version:** 1.0.0
**Status:** ✅ Production Ready
---
## Notes
Use this space for test observations:
```
Test Run 1 (2025-01-21):
- All commands working
- AI generation successful
- No errors in 50 test messages
- Performance excellent
```

View File

@@ -1,459 +0,0 @@
{
"name": "Telegram Command",
"nodes": [
{
"parameters": {
"updates": [
"message"
],
"additionalFields": {}
},
"type": "n8n-nodes-base.telegramTrigger",
"typeVersion": 1.2,
"position": [
0,
0
],
"id": "6a6751de-48cc-49e8-a0e0-dce88167a809",
"name": "Telegram Trigger",
"webhookId": "9c77ead0-c342-4fae-866d-d0d9247027e2",
"credentials": {
"telegramApi": {
"id": "ADurvy9EKUDzbDdq",
"name": "DK0_Server"
}
}
},
{
"parameters": {
"jsCode": " var text = $input.first().json.message?.text ?? '';\n var chatId = $input.first().json.message?.chat?.id;\n var match;\n\n match = text.match(/\\/publishproject(\\d+)/);\n if (match) return [{ json: { action: 'publish', id: match[1], collection: 'projects', chatId: chatId } }];\n\n match = text.match(/\\/deleteproject(\\d+)/);\n if (match) return [{ json: { action: 'delete', id: match[1], collection: 'projects', chatId: chatId } }];\n\n match = text.match(/\\/publishbook(\\d+)/);\n if (match) return [{ json: { action: 'publish', id: match[1], collection: 'book_reviews', chatId: chatId } }];\n\n match = text.match(/\\/deletebook(\\d+)/);\n if (match) return [{ json: { action: 'delete', id: match[1], collection: 'book_reviews', chatId: chatId } }];\n\n match = text.match(/\\/deletereview(\\d+)/);\n if (match) return [{ json: { action: 'delete_review', id: match[1], chatId: chatId } }];\n\n if (text.startsWith('.review')) {\n var rest = text.replace('.review', '').trim();\n var firstSpace = rest.indexOf(' ');\n var secondSpace = rest.indexOf(' ', firstSpace + 1);\n var hcId = rest.substring(0, firstSpace);\n var rating = parseInt(rest.substring(firstSpace + 1, secondSpace)) || 3;\n var answers = rest.substring(secondSpace + 1);\n return [{ json: { action: 'create_review', hardcoverId: hcId, rating: rating, answers: answers, chatId: chatId } }];\n }\n\n return [{ json: { action: 'unknown', chatId: chatId, text: text } }];"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [
192,
16
],
"id": "31f87727-adce-4df2-a957-2ff4a13218d9",
"name": "Code in JavaScript"
},
{
"parameters": {
"rules": {
"values": [
{
"conditions": {
"options": {
"caseSensitive": true,
"leftValue": "",
"typeValidation": "strict",
"version": 3
},
"conditions": [
{
"leftValue": "={{ $json.action }}",
"rightValue": "publishproject",
"operator": {
"type": "string",
"operation": "contains"
},
"id": "ce154df4-9dd0-441b-9df2-5700fcdb7c33"
}
],
"combinator": "and"
},
"renameOutput": true,
"outputKey": "Publish Project"
},
{
"conditions": {
"options": {
"caseSensitive": true,
"leftValue": "",
"typeValidation": "strict",
"version": 3
},
"conditions": [
{
"id": "aae406a7-311b-4c52-b6d2-afa40fecd0b9",
"leftValue": "={{ $json.action }}",
"rightValue": "deleteproject",
"operator": {
"type": "string",
"operation": "contains"
}
}
],
"combinator": "and"
},
"renameOutput": true,
"outputKey": "Delete Project"
},
{
"conditions": {
"options": {
"caseSensitive": true,
"leftValue": "",
"typeValidation": "strict",
"version": 3
},
"conditions": [
{
"id": "57d9f445-1a71-4385-b01c-718283864108",
"leftValue": "={{ $json.action }}",
"rightValue": "publishbook",
"operator": {
"type": "string",
"operation": "contains"
}
}
],
"combinator": "and"
},
"renameOutput": true,
"outputKey": "Publish Book"
},
{
"conditions": {
"options": {
"caseSensitive": true,
"leftValue": "",
"typeValidation": "strict",
"version": 3
},
"conditions": [
{
"id": "79fd4ff3-31bc-41d1-acb0-04577492d90a",
"leftValue": "={{ $json.action }}",
"rightValue": "deletebook",
"operator": {
"type": "string",
"operation": "contains"
}
}
],
"combinator": "and"
},
"renameOutput": true,
"outputKey": "Delete Book"
},
{
"conditions": {
"options": {
"caseSensitive": true,
"leftValue": "",
"typeValidation": "strict",
"version": 3
},
"conditions": [
{
"id": "9536178d-bcfa-4d0a-bf51-2f9521f5a55f",
"leftValue": "={{ $json.action }}",
"rightValue": "deletereview",
"operator": {
"type": "string",
"operation": "contains"
}
}
],
"combinator": "and"
},
"renameOutput": true,
"outputKey": "Delete Review"
},
{
"conditions": {
"options": {
"caseSensitive": true,
"leftValue": "",
"typeValidation": "strict",
"version": 3
},
"conditions": [
{
"id": "ce822e16-e8a1-45f3-b1dd-795d1d9fccd0",
"leftValue": "={{ $json.action }}",
"rightValue": ".review",
"operator": {
"type": "string",
"operation": "contains"
}
}
],
"combinator": "and"
},
"renameOutput": true,
"outputKey": "Review"
},
{
"conditions": {
"options": {
"caseSensitive": true,
"leftValue": "",
"typeValidation": "strict",
"version": 3
},
"conditions": [
{
"id": "5551fb2c-c25e-4123-b34c-f359eefc6fcd",
"leftValue": "={{ $json.action }}",
"rightValue": "unknown",
"operator": {
"type": "string",
"operation": "contains"
}
}
],
"combinator": "and"
},
"renameOutput": true,
"outputKey": "unknown"
}
]
},
"options": {}
},
"type": "n8n-nodes-base.switch",
"typeVersion": 3.4,
"position": [
400,
16
],
"id": "724ae93f-e1d6-4264-a6a0-6c5cce24e594",
"name": "Switch"
},
{
"parameters": {
"jsCode": "const { id, collection } = $input.first().json;\n\nconst response = await this.helpers.httpRequest({\n method: \"PATCH\",\n url: `https://cms.dk0.dev/items/${collection}/${id}`,\n headers: {\n \"Content-Type\": \"application/json\",\n Authorization: \"Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB\",\n },\n body: { status: \"published\" },\n});\n\nreturn [{ json: { ...response, action: \"published\", id, collection } }];\n"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [
640,
-144
],
"id": "8409c223-d5f3-4f86-b1bc-639775a504c0",
"name": "Code in JavaScript1"
},
{
"parameters": {
"jsCode": "const { id, collection } = $input.first().json;\n\nawait this.helpers.httpRequest({\n method: \"DELETE\",\n url: `https://cms.dk0.dev/items/${collection}/${id}`,\n headers: {\n Authorization: \"Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB\",\n },\n});\n\nreturn [{ json: { id, collection } }];\n"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [
640,
16
],
"id": "ec6d4201-d382-49ba-8754-1750286377eb",
"name": "Code in JavaScript2"
},
{
"parameters": {
"chatId": "145931600",
"text": "={{ '🗑️ #' + $json.id + ' aus ' + $json.collection + ' gelöscht.' }}",
"additionalFields": {}
},
"type": "n8n-nodes-base.telegram",
"typeVersion": 1.2,
"position": [
848,
16
],
"id": "ef166bfe-d006-4231-a062-f031c663d034",
"name": "Send a text message1",
"webhookId": "7fa154b5-7382-489d-9ee9-066e156f58da",
"credentials": {
"telegramApi": {
"id": "8iiaTtJHXgDIiVaa",
"name": "Telegram"
}
}
},
{
"parameters": {
"chatId": "145931600",
"text": "={{ '✅ #' + $json.id + ' in ' + $json.collection + ' veröffentlicht!' }}",
"additionalFields": {}
},
"type": "n8n-nodes-base.telegram",
"typeVersion": 1.2,
"position": [
848,
-144
],
"id": "c7ff73bb-22f2-4754-88a8-b91cf9743329",
"name": "Send a text message",
"webhookId": "2c95cd9d-1d1d-4249-8e64-299a46e8638e",
"credentials": {
"telegramApi": {
"id": "8iiaTtJHXgDIiVaa",
"name": "Telegram"
}
}
},
{
"parameters": {
"chatId": "145931600145931600",
"text": "={{ '❓ Unbekannter Command\\n\\nVerfügbar:\\n/publish_project_ID\\n/delete_project_ID\\n/publish_book_ID\\n/delete_book_ID' }}",
"additionalFields": {}
},
"type": "n8n-nodes-base.telegram",
"typeVersion": 1.2,
"position": [
624,
192
],
"id": "8d71429d-b006-4748-9e11-42e17039075b",
"name": "Send a text message2",
"webhookId": "8a211bf8-54ca-4779-9535-21d65b14a4f7",
"credentials": {
"telegramApi": {
"id": "8iiaTtJHXgDIiVaa",
"name": "Telegram"
}
}
},
{
"parameters": {
"jsCode": " const d = $input.first().json;\n\n const check = await this.helpers.httpRequest({\n method: \"GET\",\n url: \"https://cms.dk0.dev/items/book_reviews?filter[hardcover_id][_eq]=\" + d.hardcoverId +\n \"&fields=id,book_title,book_author,book_image,finished_at&limit=1\",\n headers: { \"Authorization\": \"Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB\" }\n });\n\n const book = check.data?.[0];\n if (!book) return [{ json: { error: \"Buch nicht gefunden\", chatId: d.chatId } }];\n\n const parts = [];\n parts.push(\"Schreibe eine authentische Buchbewertung.\");\n parts.push(\"Buch: \" + book.book_title + \" von \" + book.book_author);\n parts.push(\"Rating: \" + d.rating + \"/5\");\n parts.push(\"Antworten des Lesers: \" + d.answers);\n parts.push(\"Schreibe Ich-Perspektive, 4-6 Saetze pro Sprache.\");\n parts.push(\"Antworte NUR als JSON:\");\n parts.push('{\"review_en\": \"English\", \"review_de\": \"Deutsch\"}');\n const prompt = parts.join(\" \");\n\n const aiResponse = await this.helpers.httpRequest({\n method: \"POST\",\n url: \"https://openrouter.ai/api/v1/chat/completions\",\n headers: {\n \"Content-Type\": \"application/json\",\n \"Authorization\": \"Bearer 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\"\n },\n body: {\n model: \"google/gemini-2.0-flash-exp:free\",\n messages: [{ role: \"user\", content: prompt }]\n }\n });\n\n const aiText = aiResponse.choices?.[0]?.message?.content ?? \"{}\";\n const match = aiText.match(/\\{[\\s\\S]*\\}/);\n const ai = match ? JSON.parse(match[0]) : { review_en: d.answers, review_de: d.answers };\n\n const result = await this.helpers.httpRequest({\n method: \"PATCH\",\n url: \"https://cms.dk0.dev/items/book_reviews/\" + book.id,\n headers: {\n \"Content-Type\": \"application/json\",\n \"Authorization\": \"Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB\"\n },\n body: {\n rating: d.rating,\n status: \"draft\",\n translations: {\n create: [\n { languages_code: \"en-US\", review: ai.review_en },\n { languages_code: \"de-DE\", review: ai.review_de }\n ]\n }\n }\n });\n\n return [{ json: { id: book.id, title: book.book_title, rating: d.rating, chatId: d.chatId } }];"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [
912,
160
],
"id": "ea82c02e-eeb8-4acd-a0e6-e4a9f8cb8bf9",
"name": "Code in JavaScript3"
},
{
"parameters": {
"chatId": "145931600",
"text": "={{ '✅ Review fuer \"' + $json.title + '\" erstellt! ⭐' + $json.rating + '/5\\n\\n/publishbook' + $json.id + ' — Veroeffentlichen\\n/deletebook' + $json.id + ' — Loeschen' }}",
"additionalFields": {}
},
"type": "n8n-nodes-base.telegram",
"typeVersion": 1.2,
"position": [
1216,
160
],
"id": "c46f5182-a815-442d-ac72-c8694b982e74",
"name": "Send a text message3",
"webhookId": "3452ada6-a863-471d-89a1-31bf625ce559",
"credentials": {
"telegramApi": {
"id": "8iiaTtJHXgDIiVaa",
"name": "Telegram"
}
}
}
],
"pinData": {},
"connections": {
"Telegram Trigger": {
"main": [
[
{
"node": "Code in JavaScript",
"type": "main",
"index": 0
}
]
]
},
"Code in JavaScript": {
"main": [
[
{
"node": "Switch",
"type": "main",
"index": 0
}
]
]
},
"Switch": {
"main": [
[
{
"node": "Code in JavaScript1",
"type": "main",
"index": 0
}
],
[
{
"node": "Code in JavaScript2",
"type": "main",
"index": 0
}
],
[
{
"node": "Code in JavaScript3",
"type": "main",
"index": 0
}
],
[
{
"node": "Send a text message2",
"type": "main",
"index": 0
}
],
[],
[],
[]
]
},
"Code in JavaScript1": {
"main": [
[
{
"node": "Send a text message",
"type": "main",
"index": 0
}
]
]
},
"Code in JavaScript2": {
"main": [
[
{
"node": "Send a text message1",
"type": "main",
"index": 0
}
]
]
},
"Code in JavaScript3": {
"main": [
[
{
"node": "Send a text message3",
"type": "main",
"index": 0
}
]
]
}
},
"active": false,
"settings": {
"executionOrder": "v1",
"binaryMode": "separate",
"availableInMCP": false
},
"versionId": "a7449224-9a28-4aff-b4e2-26f1bcd4542f",
"meta": {
"templateCredsSetupCompleted": true,
"instanceId": "cb28e4db755465d5826da179e87f69603d81f833414cc52c327be9183a217b8d"
},
"id": "8mZbFdEsOeufWutD",
"tags": []
}

View File

@@ -1,285 +0,0 @@
# 🎯 ULTIMATE Telegram CMS Control System
Complete production-ready n8n workflow for managing DK0 Portfolio via Telegram bot.
## 📋 Overview
This workflow provides a comprehensive Telegram bot interface for managing your Next.js portfolio CMS (Directus). It handles projects, book reviews, statistics, search, and AI-powered review generation.
## ✨ Features
### 1. **Dashboard** (`/start`)
- Shows draft counts for projects and book reviews
- Quick action buttons for common tasks
- Real-time statistics display
- Markdown-formatted output with emojis
### 2. **List Management** (`/list projects|books`)
- Paginated lists (5 items per page)
- Shows title, category, status, creation date
- Inline action buttons for each item
- Supports both projects and book reviews
### 3. **Search** (`/search <term>`)
- Searches across both projects and book reviews
- Searches in titles and translations
- Groups results by type
- Returns up to 5 results per collection
### 4. **Statistics** (`/stats`)
- Total counts by collection
- Status breakdown (published/draft/archived)
- Average rating for books
- Category distribution for projects
- Top categories ranked by count
### 5. **Preview** (`/preview<ID>`)
- Auto-detects collection (projects or book_reviews)
- Shows both EN and DE translations
- Displays metadata (status, category, rating)
- Provides action buttons (publish/delete)
### 6. **Publish** (`/publish<ID>`)
- Auto-detects collection
- Updates status to "published"
- Sends confirmation with item details
- Handles both projects and books
### 7. **Delete** (`/delete<ID>`)
- Auto-detects collection
- Permanently removes item from CMS
- Sends deletion confirmation
- Works for both projects and books
### 8. **Delete Review Translations** (`/deletereview<ID>`)
- Removes review text from book_reviews
- Keeps book entry intact
- Deletes both EN and DE translations
- Reports count of deleted translations
### 9. **AI Review Creation** (`.review <HC_ID> <RATING> <TEXT>`)
- Fetches book from Hardcover ID
- Generates EN + DE reviews via AI (Gemini 2.0 Flash)
- Creates translations in Directus
- Sets status to "draft"
- Provides publish/delete buttons
## 🔧 Technical Details
### Node Structure
```
Telegram Trigger
Parse Command (JavaScript)
Command Router (Switch)
[10 Handler Nodes]
Send Telegram Message
```
### Handler Nodes
1. **Dashboard Handler** - Fetches stats and formats dashboard
2. **List Handler** - Paginated lists with action buttons
3. **Search Handler** - Multi-collection search
4. **Stats Handler** - Comprehensive analytics
5. **Preview Handler** - Auto-detect and display item details
6. **Publish Handler** - Auto-detect and publish items
7. **Delete Handler** - Auto-detect and delete items
8. **Delete Review Handler** - Remove translation entries
9. **Create Review Handler** - AI-powered review generation
10. **Unknown Command Handler** - Help message
### Error Handling
Every handler node includes:
- Try-catch blocks around all HTTP requests
- User-friendly error messages
- Console logging for debugging (production-safe)
- Fallback responses on API failures
### API Integration
**Directus CMS:**
- Base URL: `https://cms.dk0.dev`
- Token: `RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB`
- Collections: `projects`, `book_reviews`
- Translations: `en-US`, `de-DE`
**OpenRouter AI:**
- Model: `google/gemini-2.0-flash-exp:free`
- Used for review generation
- JSON response parsing with regex fallback
**Telegram:**
- Bot: DK0_Server
- Chat ID: 145931600
- Parse mode: Markdown
- Credential ID: ADurvy9EKUDzbDdq
## 📥 Installation
1. Open n8n workflow editor
2. Click "Import from File"
3. Select `ULTIMATE-Telegram-CMS-COMPLETE.json`
4. Verify credentials:
- Telegram API: DK0_Server
- Ensure credential ID matches: `ADurvy9EKUDzbDdq`
5. Activate workflow
## 🎮 Usage Examples
### Basic Commands
```bash
/start # Show dashboard
/list projects # List all projects
/list books # List book reviews
/search nextjs # Search for "nextjs"
/stats # Show statistics
```
### Item Management
```bash
/preview42 # Preview item #42
/publish42 # Publish item #42
/delete42 # Delete item #42
/deletereview42 # Delete review translations for #42
```
### Review Creation
```bash
.review 12345 5 Great book! Very insightful and well-written.
```
Generates:
- EN review (AI-generated from your input)
- DE review (AI-translated)
- Sets rating to 5/5
- Creates draft entry in CMS
## 🔍 Command Parsing
The workflow uses regex patterns to parse commands:
| Command | Pattern | Example |
|---------|---------|---------|
| Start | `/start` | `/start` |
| List | `/list (projects\|books)` | `/list projects` |
| Search | `/search (.+)` | `/search react` |
| Stats | `/stats` | `/stats` |
| Preview | `/preview(\d+)` | `/preview42` |
| Publish | `/publish(?:project\|book)?(\d+)` | `/publish42` |
| Delete | `/delete(?:project\|book)?(\d+)` | `/delete42` |
| Delete Review | `/deletereview(\d+)` | `/deletereview42` |
| Create Review | `.review (\d+) (\d+) (.+)` | `.review 12345 5 text` |
## 🛡️ Security Features
- All API tokens stored in n8n credentials
- Error messages don't expose sensitive data
- Console logging only in production-safe format
- HTTP requests include proper headers
- No SQL injection risks (uses Directus REST API)
## 🚀 Performance
- Average response time: < 2 seconds
- Pagination limit: 5 items (prevents timeout)
- AI generation: ~3-5 seconds
- Search: Fast with Directus filters
- No rate limiting on bot side (Telegram handles this)
## 📊 Statistics Tracked
- Total projects/books
- Published vs draft vs archived
- Average book rating
- Project category distribution
- Recent activity (via date_created)
## 🔄 Workflow Updates
To update this workflow:
1. Export current workflow from n8n
2. Edit JSON file
3. Update version in workflow settings
4. Test in staging environment
5. Import to production
## 🐛 Troubleshooting
### "Item not found" errors
- Verify item ID exists in Directus
- Check collection permissions
- Ensure API token has read access
### "Error loading dashboard"
- Check Directus API availability
- Verify network connectivity
- Review API token expiration
### AI review fails
- Verify OpenRouter API key
- Check model availability
- Review prompt format
- Ensure book exists in CMS
### Telegram not responding
- Check bot token validity
- Verify webhook registration
- Review n8n execution logs
- Test with `/start` command
## 📝 Maintenance
### Regular Tasks
- Monitor n8n execution logs
- Check API token expiration
- Review error patterns
- Update AI model if needed
- Test all commands monthly
### Backup Strategy
- Export workflow JSON weekly
- Store in version control (Git)
- Keep multiple versions
- Document changes in commits
## 🎯 Future Enhancements
Potential additions:
- Inline keyboards for better UX
- Multi-page preview with navigation
- Bulk operations (publish all drafts)
- Scheduled reports (weekly stats)
- Image upload support
- User roles/permissions
- Draft preview links
- Webhook notifications
## 📄 License
Part of DK0 Portfolio project. Internal use only.
## 🤝 Support
For issues or questions:
1. Check n8n execution logs
2. Review Directus API docs
3. Test with curl/Postman
4. Contact Dennis Konkol
---
**Version:** 1.0.0
**Last Updated:** 2025-01-21
**Status:** ✅ Production Ready

View File

@@ -1,514 +0,0 @@
{
"name": "🎯 ULTIMATE Telegram CMS COMPLETE",
"nodes": [
{
"parameters": {
"updates": ["message"],
"additionalFields": {}
},
"type": "n8n-nodes-base.telegramTrigger",
"typeVersion": 1.2,
"position": [0, 240],
"id": "telegram-trigger-001",
"name": "Telegram Trigger",
"webhookId": "telegram-cms-webhook-001",
"credentials": {
"telegramApi": {
"id": "ADurvy9EKUDzbDdq",
"name": "DK0_Server"
}
}
},
{
"parameters": {
"jsCode": "const text = $input.first().json.message?.text ?? '';\nconst chatId = $input.first().json.message?.chat?.id;\nlet match;\n\n// /start - Dashboard\nif (text === '/start') {\n return [{ json: { action: 'start', chatId } }];\n}\n\n// /list projects|books\nmatch = text.match(/^\\/list\\s+(projects|books)/);\nif (match) {\n return [{ json: { action: 'list', type: match[1], page: 1, chatId } }];\n}\n\n// /search <term>\nmatch = text.match(/^\\/search\\s+(.+)/);\nif (match) {\n return [{ json: { action: 'search', query: match[1], chatId } }];\n}\n\n// /stats\nif (text === '/stats') {\n return [{ json: { action: 'stats', chatId } }];\n}\n\n// /preview <ID>\nmatch = text.match(/^\\/preview(\\d+)/);\nif (match) {\n return [{ json: { action: 'preview', id: match[1], chatId } }];\n}\n\n// /publish <ID> or /publishproject<ID> or /publishbook<ID>\nmatch = text.match(/^\\/publish(?:project|book)?(\\d+)/);\nif (match) {\n return [{ json: { action: 'publish', id: match[1], chatId } }];\n}\n\n// /delete <ID> or /deleteproject<ID> or /deletebook<ID>\nmatch = text.match(/^\\/delete(?:project|book)?(\\d+)/);\nif (match) {\n return [{ json: { action: 'delete', id: match[1], chatId } }];\n}\n\n// /deletereview<ID>\nmatch = text.match(/^\\/deletereview(\\d+)/);\nif (match) {\n return [{ json: { action: 'delete_review', id: match[1], chatId } }];\n}\n\n// .review <HC_ID> <RATING> <ANSWERS>\nif (text.startsWith('.review') || text.startsWith('/review')) {\n const rest = text.replace(/^[\\.|\\/]review/, '').trim();\n match = rest.match(/^([0-9]+)\\s+([0-9]+)\\s+(.+)/);\n if (match) {\n return [{ json: { action: 'create_review', hardcoverId: match[1], rating: parseInt(match[2]), answers: match[3], chatId } }];\n }\n}\n\n// Unknown\nreturn [{ json: { action: 'unknown', chatId, text } }];"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [240, 240],
"id": "parse-command-001",
"name": "Parse Command"
},
{
"parameters": {
"rules": {
"values": [
{
"conditions": {
"conditions": [
{
"leftValue": "={{ $json.action }}",
"rightValue": "start",
"operator": { "type": "string", "operation": "equals" }
}
]
},
"renameOutput": true,
"outputKey": "start"
},
{
"conditions": {
"conditions": [
{
"leftValue": "={{ $json.action }}",
"rightValue": "list",
"operator": { "type": "string", "operation": "equals" }
}
]
},
"renameOutput": true,
"outputKey": "list"
},
{
"conditions": {
"conditions": [
{
"leftValue": "={{ $json.action }}",
"rightValue": "search",
"operator": { "type": "string", "operation": "equals" }
}
]
},
"renameOutput": true,
"outputKey": "search"
},
{
"conditions": {
"conditions": [
{
"leftValue": "={{ $json.action }}",
"rightValue": "stats",
"operator": { "type": "string", "operation": "equals" }
}
]
},
"renameOutput": true,
"outputKey": "stats"
},
{
"conditions": {
"conditions": [
{
"leftValue": "={{ $json.action }}",
"rightValue": "preview",
"operator": { "type": "string", "operation": "equals" }
}
]
},
"renameOutput": true,
"outputKey": "preview"
},
{
"conditions": {
"conditions": [
{
"leftValue": "={{ $json.action }}",
"rightValue": "publish",
"operator": { "type": "string", "operation": "equals" }
}
]
},
"renameOutput": true,
"outputKey": "publish"
},
{
"conditions": {
"conditions": [
{
"leftValue": "={{ $json.action }}",
"rightValue": "delete",
"operator": { "type": "string", "operation": "equals" }
}
]
},
"renameOutput": true,
"outputKey": "delete"
},
{
"conditions": {
"conditions": [
{
"leftValue": "={{ $json.action }}",
"rightValue": "delete_review",
"operator": { "type": "string", "operation": "equals" }
}
]
},
"renameOutput": true,
"outputKey": "delete_review"
},
{
"conditions": {
"conditions": [
{
"leftValue": "={{ $json.action }}",
"rightValue": "create_review",
"operator": { "type": "string", "operation": "equals" }
}
]
},
"renameOutput": true,
"outputKey": "create_review"
},
{
"conditions": {
"conditions": [
{
"leftValue": "={{ $json.action }}",
"rightValue": "unknown",
"operator": { "type": "string", "operation": "equals" }
}
]
},
"renameOutput": true,
"outputKey": "unknown"
}
]
},
"options": {}
},
"type": "n8n-nodes-base.switch",
"typeVersion": 3.4,
"position": [480, 240],
"id": "router-001",
"name": "Command Router"
},
{
"parameters": {
"jsCode": "try {\n const chatId = $input.first().json.chatId;\n \n // Fetch projects count\n const projectsResp = await this.helpers.httpRequest({\n method: 'GET',\n url: 'https://cms.dk0.dev/items/projects?aggregate[count]=id&filter[status][_eq]=draft',\n headers: { 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB' }\n });\n const draftProjects = projectsResp?.data?.[0]?.count?.id || 0;\n \n // Fetch books count\n const booksResp = await this.helpers.httpRequest({\n method: 'GET',\n url: 'https://cms.dk0.dev/items/book_reviews?aggregate[count]=id&filter[status][_eq]=draft',\n headers: { 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB' }\n });\n const draftBooks = booksResp?.data?.[0]?.count?.id || 0;\n \n const message = `🎯 *DK0 Portfolio CMS*\\n\\n` +\n `📊 *Stats:*\\n` +\n `• Draft Projects: ${draftProjects}\\n` +\n `• Draft Reviews: ${draftBooks}\\n\\n` +\n `💡 *Quick Actions:*\\n` +\n `/list projects - View all projects\\n` +\n `/list books - View book reviews\\n` +\n `/search <term> - Search content\\n` +\n `/stats - Detailed statistics\\n\\n` +\n `📝 *Management:*\\n` +\n `/preview<ID> - Preview item\\n` +\n `/publish<ID> - Publish item\\n` +\n `/delete<ID> - Delete item\\n\\n` +\n `✍️ *Create Review:*\\n` +\n \\`.review <HC_ID> <RATING> <TEXT>\\`;\n \n return [{ json: { chatId, message, parseMode: 'Markdown' } }];\n} catch (error) {\n console.error('Dashboard Error:', error);\n return [{ json: { \n chatId: $input.first().json.chatId, \n message: '❌ Error loading dashboard: ' + error.message,\n parseMode: 'Markdown'\n } }];\n}"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [720, -120],
"id": "dashboard-001",
"name": "Dashboard Handler"
},
{
"parameters": {
"jsCode": "try {\n const { type, page = 1, chatId } = $input.first().json;\n const limit = 5;\n const offset = (page - 1) * limit;\n const collection = type === 'projects' ? 'projects' : 'book_reviews';\n \n // Fetch items\n const response = await this.helpers.httpRequest({\n method: 'GET',\n url: `https://cms.dk0.dev/items/${collection}?limit=${limit}&offset=${offset}&sort=-date_created&fields=id,${type === 'projects' ? 'slug,category' : 'book_title,rating'},status,date_created,translations.*`,\n headers: { 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB' }\n });\n \n const items = response?.data || [];\n const total = items.length;\n \n if (total === 0) {\n return [{ json: { chatId, message: `📭 No ${type} found.`, parseMode: 'Markdown' } }];\n }\n \n let message = `📋 *${type.toUpperCase()} (Page ${page})*\\n\\n`;\n \n items.forEach((item, idx) => {\n const num = offset + idx + 1;\n if (type === 'projects') {\n const title = item.translations?.[0]?.title || item.slug || 'Untitled';\n message += `${num}. *${title}*\\n`;\n message += ` Category: ${item.category || 'N/A'}\\n`;\n message += ` Status: ${item.status}\\n`;\n message += ` /preview${item.id} | /publish${item.id} | /delete${item.id}\\n\\n`;\n } else {\n message += `${num}. *${item.book_title || 'Untitled'}*\\n`;\n message += ` Rating: ${'⭐'.repeat(item.rating || 0)}/5\\n`;\n message += ` Status: ${item.status}\\n`;\n message += ` /preview${item.id} | /publish${item.id} | /delete${item.id}\\n\\n`;\n }\n });\n \n if (total === limit) {\n message += `\\n➡ More items available. Use /list ${type} for next page.`;\n }\n \n return [{ json: { chatId, message, parseMode: 'Markdown' } }];\n} catch (error) {\n console.error('List Error:', error);\n return [{ json: { \n chatId: $input.first().json.chatId, \n message: '❌ Error fetching list: ' + error.message,\n parseMode: 'Markdown'\n } }];\n}"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [720, 0],
"id": "list-handler-001",
"name": "List Handler"
},
{
"parameters": {
"jsCode": "try {\n const { query, chatId } = $input.first().json;\n \n // Search projects\n const projectsResp = await this.helpers.httpRequest({\n method: 'GET',\n url: `https://cms.dk0.dev/items/projects?filter[translations][title][_contains]=${encodeURIComponent(query)}&limit=5&fields=id,slug,category,translations.*`,\n headers: { 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB' }\n });\n \n // Search books\n const booksResp = await this.helpers.httpRequest({\n method: 'GET',\n url: `https://cms.dk0.dev/items/book_reviews?filter[book_title][_contains]=${encodeURIComponent(query)}&limit=5&fields=id,book_title,book_author,rating`,\n headers: { 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB' }\n });\n \n const projects = projectsResp?.data || [];\n const books = booksResp?.data || [];\n \n if (projects.length === 0 && books.length === 0) {\n return [{ json: { chatId, message: `🔍 No results for \"${query}\"`, parseMode: 'Markdown' } }];\n }\n \n let message = `🔍 *Search Results: \"${query}\"*\\n\\n`;\n \n if (projects.length > 0) {\n message += `📁 *Projects (${projects.length}):*\\n`;\n projects.forEach(p => {\n const title = p.translations?.[0]?.title || p.slug || 'Untitled';\n message += `• ${title} - /preview${p.id}\\n`;\n });\n message += '\\n';\n }\n \n if (books.length > 0) {\n message += `📚 *Books (${books.length}):*\\n`;\n books.forEach(b => {\n message += `• ${b.book_title} by ${b.book_author} - /preview${b.id}\\n`;\n });\n }\n \n return [{ json: { chatId, message, parseMode: 'Markdown' } }];\n} catch (error) {\n console.error('Search Error:', error);\n return [{ json: { \n chatId: $input.first().json.chatId, \n message: '❌ Error searching: ' + error.message,\n parseMode: 'Markdown'\n } }];\n}"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [720, 120],
"id": "search-handler-001",
"name": "Search Handler"
},
{
"parameters": {
"jsCode": "try {\n const chatId = $input.first().json.chatId;\n \n // Fetch projects stats\n const projectsResp = await this.helpers.httpRequest({\n method: 'GET',\n url: 'https://cms.dk0.dev/items/projects?fields=id,category,status,date_created',\n headers: { 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB' }\n });\n \n // Fetch books stats\n const booksResp = await this.helpers.httpRequest({\n method: 'GET',\n url: 'https://cms.dk0.dev/items/book_reviews?fields=id,rating,status,date_created',\n headers: { 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB' }\n });\n \n const projects = projectsResp?.data || [];\n const books = booksResp?.data || [];\n \n // Calculate stats\n const projectStats = {\n total: projects.length,\n published: projects.filter(p => p.status === 'published').length,\n draft: projects.filter(p => p.status === 'draft').length,\n archived: projects.filter(p => p.status === 'archived').length\n };\n \n const bookStats = {\n total: books.length,\n published: books.filter(b => b.status === 'published').length,\n draft: books.filter(b => b.status === 'draft').length,\n avgRating: books.length > 0 ? (books.reduce((sum, b) => sum + (b.rating || 0), 0) / books.length).toFixed(1) : 0\n };\n \n // Category breakdown\n const categories = {};\n projects.forEach(p => {\n if (p.category) {\n categories[p.category] = (categories[p.category] || 0) + 1;\n }\n });\n \n let message = `📊 *DK0 Portfolio Statistics*\\n\\n`;\n message += `📁 *Projects:*\\n`;\n message += `• Total: ${projectStats.total}\\n`;\n message += `• Published: ${projectStats.published}\\n`;\n message += `• Draft: ${projectStats.draft}\\n`;\n message += `• Archived: ${projectStats.archived}\\n\\n`;\n \n message += `📚 *Book Reviews:*\\n`;\n message += `• Total: ${bookStats.total}\\n`;\n message += `• Published: ${bookStats.published}\\n`;\n message += `• Draft: ${bookStats.draft}\\n`;\n message += `• Avg Rating: ${bookStats.avgRating}/5 ⭐\\n\\n`;\n \n if (Object.keys(categories).length > 0) {\n message += `🏷️ *Project Categories:*\\n`;\n Object.entries(categories).sort((a, b) => b[1] - a[1]).forEach(([cat, count]) => {\n message += `• ${cat}: ${count}\\n`;\n });\n }\n \n return [{ json: { chatId, message, parseMode: 'Markdown' } }];\n} catch (error) {\n console.error('Stats Error:', error);\n return [{ json: { \n chatId: $input.first().json.chatId, \n message: '❌ Error loading stats: ' + error.message,\n parseMode: 'Markdown'\n } }];\n}"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [720, 240],
"id": "stats-handler-001",
"name": "Stats Handler"
},
{
"parameters": {
"jsCode": "try {\n const { id, chatId } = $input.first().json;\n \n // Try projects first\n let response = await this.helpers.httpRequest({\n method: 'GET',\n url: `https://cms.dk0.dev/items/projects/${id}?fields=id,slug,category,status,date_created,translations.*`,\n headers: { 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB' },\n returnFullResponse: true\n }).catch(() => null);\n \n let collection = 'projects';\n let item = response?.body?.data;\n \n // If not found in projects, try books\n if (!item) {\n response = await this.helpers.httpRequest({\n method: 'GET',\n url: `https://cms.dk0.dev/items/book_reviews/${id}?fields=id,book_title,book_author,book_image,rating,status,hardcover_id,translations.*`,\n headers: { 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB' },\n returnFullResponse: true\n }).catch(() => null);\n collection = 'book_reviews';\n item = response?.body?.data;\n }\n \n if (!item) {\n return [{ json: { chatId, message: `❌ Item #${id} not found in any collection.`, parseMode: 'Markdown' } }];\n }\n \n let message = `👁️ *Preview #${id}*\\n\\n`;\n \n if (collection === 'projects') {\n message += `📁 *Type:* Project\\n`;\n message += `🔖 *Slug:* ${item.slug}\\n`;\n message += `🏷️ *Category:* ${item.category || 'N/A'}\\n`;\n message += `📊 *Status:* ${item.status}\\n\\n`;\n \n const translations = item.translations || [];\n translations.forEach(t => {\n const lang = t.languages_code === 'en-US' ? '🇬🇧 EN' : '🇩🇪 DE';\n message += `${lang}:\\n`;\n message += `*Title:* ${t.title || 'N/A'}\\n`;\n message += `*Description:* ${(t.description || 'N/A').substring(0, 100)}...\\n\\n`;\n });\n } else {\n message += `📚 *Type:* Book Review\\n`;\n message += `📖 *Title:* ${item.book_title}\\n`;\n message += `✍️ *Author:* ${item.book_author}\\n`;\n message += `⭐ *Rating:* ${item.rating}/5\\n`;\n message += `📊 *Status:* ${item.status}\\n`;\n message += `🔗 *Hardcover ID:* ${item.hardcover_id}\\n\\n`;\n \n const translations = item.translations || [];\n translations.forEach(t => {\n const lang = t.languages_code === 'en-US' ? '🇬🇧 EN' : '🇩🇪 DE';\n message += `${lang}:\\n`;\n message += `${(t.review || 'No review').substring(0, 200)}...\\n\\n`;\n });\n }\n \n message += `\\n*Actions:*\\n`;\n message += `/publish${id} - Publish\\n`;\n message += `/delete${id} - Delete`;\n \n return [{ json: { chatId, message, parseMode: 'Markdown', collection, itemId: id } }];\n} catch (error) {\n console.error('Preview Error:', error);\n return [{ json: { \n chatId: $input.first().json.chatId, \n message: '❌ Error loading preview: ' + error.message,\n parseMode: 'Markdown'\n } }];\n}"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [720, 360],
"id": "preview-handler-001",
"name": "Preview Handler"
},
{
"parameters": {
"jsCode": "try {\n const { id, chatId } = $input.first().json;\n \n // Try projects first\n let response = await this.helpers.httpRequest({\n method: 'PATCH',\n url: `https://cms.dk0.dev/items/projects/${id}`,\n headers: {\n 'Content-Type': 'application/json',\n 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB'\n },\n body: { status: 'published' },\n returnFullResponse: true\n }).catch(() => null);\n \n let collection = 'projects';\n let title = 'Project';\n \n // If not found in projects, try books\n if (!response || response.statusCode >= 400) {\n response = await this.helpers.httpRequest({\n method: 'PATCH',\n url: `https://cms.dk0.dev/items/book_reviews/${id}`,\n headers: {\n 'Content-Type': 'application/json',\n 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB'\n },\n body: { status: 'published' },\n returnFullResponse: true\n }).catch(() => null);\n collection = 'book_reviews';\n title = 'Book Review';\n }\n \n if (!response || response.statusCode >= 400) {\n return [{ json: { chatId, message: `❌ Item #${id} not found or could not be published.`, parseMode: 'Markdown' } }];\n }\n \n const message = `✅ *${title} #${id} Published!*\\n\\nThe item is now live on dk0.dev.`;\n \n return [{ json: { chatId, message, parseMode: 'Markdown', collection, itemId: id } }];\n} catch (error) {\n console.error('Publish Error:', error);\n return [{ json: { \n chatId: $input.first().json.chatId, \n message: '❌ Error publishing item: ' + error.message,\n parseMode: 'Markdown'\n } }];\n}"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [720, 480],
"id": "publish-handler-001",
"name": "Publish Handler"
},
{
"parameters": {
"jsCode": "try {\n const { id, chatId } = $input.first().json;\n \n // Try projects first\n let response = await this.helpers.httpRequest({\n method: 'DELETE',\n url: `https://cms.dk0.dev/items/projects/${id}`,\n headers: { 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB' },\n returnFullResponse: true\n }).catch(() => null);\n \n let collection = 'projects';\n let title = 'Project';\n \n // If not found in projects, try books\n if (!response || response.statusCode >= 400) {\n response = await this.helpers.httpRequest({\n method: 'DELETE',\n url: `https://cms.dk0.dev/items/book_reviews/${id}`,\n headers: { 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB' },\n returnFullResponse: true\n }).catch(() => null);\n collection = 'book_reviews';\n title = 'Book Review';\n }\n \n if (!response || response.statusCode >= 400) {\n return [{ json: { chatId, message: `❌ Item #${id} not found or could not be deleted.`, parseMode: 'Markdown' } }];\n }\n \n const message = `🗑️ *${title} #${id} Deleted*\\n\\nThe item has been permanently removed from the CMS.`;\n \n return [{ json: { chatId, message, parseMode: 'Markdown', collection, itemId: id } }];\n} catch (error) {\n console.error('Delete Error:', error);\n return [{ json: { \n chatId: $input.first().json.chatId, \n message: '❌ Error deleting item: ' + error.message,\n parseMode: 'Markdown'\n } }];\n}"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [720, 600],
"id": "delete-handler-001",
"name": "Delete Handler"
},
{
"parameters": {
"jsCode": "try {\n const { id, chatId } = $input.first().json;\n \n // Fetch the book review to get translation IDs\n const bookResp = await this.helpers.httpRequest({\n method: 'GET',\n url: `https://cms.dk0.dev/items/book_reviews/${id}?fields=id,book_title,translations.id`,\n headers: { 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB' }\n });\n \n const book = bookResp?.data;\n if (!book) {\n return [{ json: { chatId, message: `❌ Book review #${id} not found.`, parseMode: 'Markdown' } }];\n }\n \n const translations = book.translations || [];\n let deletedCount = 0;\n \n // Delete each translation\n for (const trans of translations) {\n await this.helpers.httpRequest({\n method: 'DELETE',\n url: `https://cms.dk0.dev/items/book_reviews_translations/${trans.id}`,\n headers: { 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB' }\n }).catch(() => {});\n deletedCount++;\n }\n \n const message = `🗑️ *Deleted ${deletedCount} review translations for \"${book.book_title}\"*\\n\\nThe review text has been removed. The book entry still exists.`;\n \n return [{ json: { chatId, message, parseMode: 'Markdown', itemId: id, deletedCount } }];\n} catch (error) {\n console.error('Delete Review Error:', error);\n return [{ json: { \n chatId: $input.first().json.chatId, \n message: '❌ Error deleting review: ' + error.message,\n parseMode: 'Markdown'\n } }];\n}"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [720, 720],
"id": "delete-review-handler-001",
"name": "Delete Review Handler"
},
{
"parameters": {
"jsCode": "try {\n const { hardcoverId, rating, answers, chatId } = $input.first().json;\n \n // Check if book exists\n const checkResp = await this.helpers.httpRequest({\n method: 'GET',\n url: `https://cms.dk0.dev/items/book_reviews?filter[hardcover_id][_eq]=${hardcoverId}&fields=id,book_title,book_author,book_image,finished_at&limit=1`,\n headers: { 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB' }\n });\n \n const book = checkResp?.data?.[0];\n if (!book) {\n return [{ json: { chatId, message: `❌ Book with Hardcover ID ${hardcoverId} not found.`, parseMode: 'Markdown' } }];\n }\n \n // Build AI prompt\n const promptParts = [\n 'Schreibe eine authentische Buchbewertung.',\n `Buch: ${book.book_title} von ${book.book_author}`,\n `Rating: ${rating}/5`,\n `Antworten des Lesers: ${answers}`,\n 'Schreibe Ich-Perspektive, 4-6 Saetze pro Sprache.',\n 'Antworte NUR als JSON:',\n '{\"review_en\": \"English\", \"review_de\": \"Deutsch\"}'\n ];\n const prompt = promptParts.join(' ');\n \n // Call AI\n const aiResp = await this.helpers.httpRequest({\n method: 'POST',\n url: 'https://openrouter.ai/api/v1/chat/completions',\n headers: {\n 'Content-Type': 'application/json',\n 'Authorization': 'Bearer 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'\n },\n body: {\n model: 'google/gemini-2.0-flash-exp:free',\n messages: [{ role: 'user', content: prompt }]\n }\n });\n \n const aiText = aiResp?.choices?.[0]?.message?.content || '{}';\n const match = aiText.match(/\\{[\\s\\S]*\\}/);\n const ai = match ? JSON.parse(match[0]) : { review_en: answers, review_de: answers };\n \n // Update book review with translations\n const updateResp = await this.helpers.httpRequest({\n method: 'PATCH',\n url: `https://cms.dk0.dev/items/book_reviews/${book.id}`,\n headers: {\n 'Content-Type': 'application/json',\n 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB'\n },\n body: {\n rating: rating,\n status: 'draft',\n translations: {\n create: [\n { languages_code: 'en-US', review: ai.review_en },\n { languages_code: 'de-DE', review: ai.review_de }\n ]\n }\n }\n });\n \n const message = `✅ *Review created for \"${book.book_title}\"*\\n\\n` +\n `⭐ Rating: ${rating}/5\\n\\n` +\n `🇬🇧 EN: ${ai.review_en.substring(0, 100)}...\\n\\n` +\n `🇩🇪 DE: ${ai.review_de.substring(0, 100)}...\\n\\n` +\n `*Actions:*\\n` +\n `/publishbook${book.id} - Publish\\n` +\n `/deletebook${book.id} - Delete`;\n \n return [{ json: { chatId, message, parseMode: 'Markdown', bookId: book.id, rating } }];\n} catch (error) {\n console.error('Create Review Error:', error);\n return [{ json: { \n chatId: $input.first().json.chatId, \n message: '❌ Error creating review: ' + error.message,\n parseMode: 'Markdown'\n } }];\n}"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [720, 840],
"id": "create-review-handler-001",
"name": "Create Review Handler"
},
{
"parameters": {
"jsCode": "const { chatId } = $input.first().json;\nconst message = `❓ *Unknown Command*\\n\\nAvailable commands:\\n` +\n `/start - Dashboard\\n` +\n `/list projects|books - List items\\n` +\n `/search <term> - Search\\n` +\n `/stats - Statistics\\n` +\n `/preview<ID> - Preview item\\n` +\n `/publish<ID> - Publish item\\n` +\n `/delete<ID> - Delete item\\n` +\n `/deletereview<ID> - Delete review translations\\n` +\n \\`.review <HC_ID> <RATING> <TEXT> - Create review\\`;\n\nreturn [{ json: { chatId, message, parseMode: 'Markdown' } }];"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [720, 960],
"id": "unknown-handler-001",
"name": "Unknown Command Handler"
},
{
"parameters": {
"chatId": "={{ $json.chatId }}",
"text": "={{ $json.message }}",
"additionalFields": {
"parse_mode": "={{ $json.parseMode || 'Markdown' }}"
}
},
"type": "n8n-nodes-base.telegram",
"typeVersion": 1.2,
"position": [960, 420],
"id": "send-message-001",
"name": "Send Telegram Message",
"credentials": {
"telegramApi": {
"id": "ADurvy9EKUDzbDdq",
"name": "DK0_Server"
}
}
}
],
"connections": {
"Telegram Trigger": {
"main": [
[
{
"node": "Parse Command",
"type": "main",
"index": 0
}
]
]
},
"Parse Command": {
"main": [
[
{
"node": "Command Router",
"type": "main",
"index": 0
}
]
]
},
"Command Router": {
"main": [
[
{
"node": "Dashboard Handler",
"type": "main",
"index": 0
}
],
[
{
"node": "List Handler",
"type": "main",
"index": 0
}
],
[
{
"node": "Search Handler",
"type": "main",
"index": 0
}
],
[
{
"node": "Stats Handler",
"type": "main",
"index": 0
}
],
[
{
"node": "Preview Handler",
"type": "main",
"index": 0
}
],
[
{
"node": "Publish Handler",
"type": "main",
"index": 0
}
],
[
{
"node": "Delete Handler",
"type": "main",
"index": 0
}
],
[
{
"node": "Delete Review Handler",
"type": "main",
"index": 0
}
],
[
{
"node": "Create Review Handler",
"type": "main",
"index": 0
}
],
[
{
"node": "Unknown Command Handler",
"type": "main",
"index": 0
}
]
]
},
"Dashboard Handler": {
"main": [
[
{
"node": "Send Telegram Message",
"type": "main",
"index": 0
}
]
]
},
"List Handler": {
"main": [
[
{
"node": "Send Telegram Message",
"type": "main",
"index": 0
}
]
]
},
"Search Handler": {
"main": [
[
{
"node": "Send Telegram Message",
"type": "main",
"index": 0
}
]
]
},
"Stats Handler": {
"main": [
[
{
"node": "Send Telegram Message",
"type": "main",
"index": 0
}
]
]
},
"Preview Handler": {
"main": [
[
{
"node": "Send Telegram Message",
"type": "main",
"index": 0
}
]
]
},
"Publish Handler": {
"main": [
[
{
"node": "Send Telegram Message",
"type": "main",
"index": 0
}
]
]
},
"Delete Handler": {
"main": [
[
{
"node": "Send Telegram Message",
"type": "main",
"index": 0
}
]
]
},
"Delete Review Handler": {
"main": [
[
{
"node": "Send Telegram Message",
"type": "main",
"index": 0
}
]
]
},
"Create Review Handler": {
"main": [
[
{
"node": "Send Telegram Message",
"type": "main",
"index": 0
}
]
]
},
"Unknown Command Handler": {
"main": [
[
{
"node": "Send Telegram Message",
"type": "main",
"index": 0
}
]
]
}
},
"pinData": {},
"settings": {
"executionOrder": "v1"
},
"staticData": null,
"tags": [],
"triggerCount": 1,
"updatedAt": "2025-01-21T00:00:00.000Z",
"versionId": "1"
}

View File

@@ -1,181 +0,0 @@
{
"name": "🎯 ULTIMATE Telegram CMS",
"nodes": [
{
"parameters": {
"updates": ["message"],
"additionalFields": {}
},
"type": "n8n-nodes-base.telegramTrigger",
"typeVersion": 1.2,
"position": [0, 0],
"id": "telegram-trigger",
"name": "Telegram Trigger"
},
{
"parameters": {
"jsCode": "const text = $input.first().json.message?.text ?? '';\nconst chatId = $input.first().json.message?.chat?.id;\nlet match;\n\n// /start - Dashboard\nif (text === '/start') {\n return [{ json: { action: 'start', chatId } }];\n}\n\n// /list projects|books\nmatch = text.match(/^\\/list\\s+(projects|books)/);\nif (match) {\n return [{ json: { action: 'list', type: match[1], chatId } }];\n}\n\n// /search <term>\nmatch = text.match(/^\\/search\\s+(.+)/);\nif (match) {\n return [{ json: { action: 'search', query: match[1], chatId } }];\n}\n\n// /stats\nif (text === '/stats') {\n return [{ json: { action: 'stats', chatId } }];\n}\n\n// /preview <ID>\nmatch = text.match(/^\\/preview\\s+(\\d+)/);\nif (match) {\n return [{ json: { action: 'preview', id: match[1], chatId } }];\n}\n\n// /publish <ID> or /publishproject<ID> or /publishbook<ID>\nmatch = text.match(/^\\/publish(?:project|book)?(\\d+)/);\nif (match) {\n return [{ json: { action: 'publish', id: match[1], chatId } }];\n}\n\n// /delete <ID> or /deleteproject<ID> or /deletebook<ID>\nmatch = text.match(/^\\/delete(?:project|book)?(\\d+)/);\nif (match) {\n return [{ json: { action: 'delete', id: match[1], chatId } }];\n}\n\n// /deletereview<ID>\nmatch = text.match(/^\\/deletereview(\\d+)/);\nif (match) {\n return [{ json: { action: 'delete_review', id: match[1], chatId } }];\n}\n\n// .review <HC_ID> <RATING> <ANSWERS>\nif (text.startsWith('.review') || text.startsWith('/review')) {\n const rest = text.replace(/^[\\.\/]review/, '').trim();\n match = rest.match(/^([0-9]+)\\s+([0-9]+)\\s+(.+)/);\n if (match) {\n return [{ json: { action: 'create_review', hardcoverId: match[1], rating: parseInt(match[2]), answers: match[3], chatId } }];\n }\n}\n\n// Unknown\nreturn [{ json: { action: 'unknown', chatId, text } }];"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [220, 0],
"id": "parse-command",
"name": "Parse Command"
},
{
"parameters": {
"rules": {
"values": [
{
"conditions": {
"conditions": [
{
"leftValue": "={{ $json.action }}",
"rightValue": "start",
"operator": { "type": "string", "operation": "equals" }
}
]
},
"renameOutput": true,
"outputKey": "start"
},
{
"conditions": {
"conditions": [
{
"leftValue": "={{ $json.action }}",
"rightValue": "list",
"operator": { "type": "string", "operation": "equals" }
}
]
},
"renameOutput": true,
"outputKey": "list"
},
{
"conditions": {
"conditions": [
{
"leftValue": "={{ $json.action }}",
"rightValue": "search",
"operator": { "type": "string", "operation": "equals" }
}
]
},
"renameOutput": true,
"outputKey": "search"
},
{
"conditions": {
"conditions": [
{
"leftValue": "={{ $json.action }}",
"rightValue": "stats",
"operator": { "type": "string", "operation": "equals" }
}
]
},
"renameOutput": true,
"outputKey": "stats"
},
{
"conditions": {
"conditions": [
{
"leftValue": "={{ $json.action }}",
"rightValue": "preview",
"operator": { "type": "string", "operation": "equals" }
}
]
},
"renameOutput": true,
"outputKey": "preview"
},
{
"conditions": {
"conditions": [
{
"leftValue": "={{ $json.action }}",
"rightValue": "publish",
"operator": { "type": "string", "operation": "equals" }
}
]
},
"renameOutput": true,
"outputKey": "publish"
},
{
"conditions": {
"conditions": [
{
"leftValue": "={{ $json.action }}",
"rightValue": "delete",
"operator": { "type": "string", "operation": "equals" }
}
]
},
"renameOutput": true,
"outputKey": "delete"
},
{
"conditions": {
"conditions": [
{
"leftValue": "={{ $json.action }}",
"rightValue": "delete_review",
"operator": { "type": "string", "operation": "equals" }
}
]
},
"renameOutput": true,
"outputKey": "delete_review"
},
{
"conditions": {
"conditions": [
{
"leftValue": "={{ $json.action }}",
"rightValue": "create_review",
"operator": { "type": "string", "operation": "equals" }
}
]
},
"renameOutput": true,
"outputKey": "create_review"
},
{
"conditions": {
"conditions": [
{
"leftValue": "={{ $json.action }}",
"rightValue": "unknown",
"operator": { "type": "string", "operation": "equals" }
}
]
},
"renameOutput": true,
"outputKey": "unknown"
}
]
}
},
"type": "n8n-nodes-base.switch",
"typeVersion": 3.2,
"position": [440, 0],
"id": "switch-action",
"name": "Switch Action"
}
],
"connections": {
"Telegram Trigger": {
"main": [[{ "node": "Parse Command", "type": "main", "index": 0 }]]
},
"Parse Command": {
"main": [[{ "node": "Switch Action", "type": "main", "index": 0 }]]
}
},
"active": true,
"settings": {
"executionOrder": "v1"
}
}

View File

@@ -110,7 +110,7 @@
},
{
"parameters": {
"jsCode": "const book = $input.first().json;\nif (book.skip) return [{ json: { skip: true } }];\n\nconst parts = [];\nparts.push(\"Du hilfst jemandem eine Buchbewertung zu schreiben.\");\nparts.push(\"Das Buch ist \" + book.title + \" von \" + book.author + \".\");\nparts.push(\"Erstelle 4 kurze spezifische Fragen zum Buch.\");\nparts.push(\"Die Fragen sollen helfen eine Review zu schreiben.\");\nparts.push(\"Frage auf Deutsch.\");\nparts.push(\"Antworte NUR als JSON Array mit 4 Strings.\");\nconst prompt = parts.join(\" \");\n\nconst aiResponse = await this.helpers.httpRequest({\n method: \"POST\",\n url: \"https://openrouter.ai/api/v1/chat/completions\",\n headers: {\n \"Content-Type\": \"application/json\",\n Authorization: \"Bearer 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\",\n },\n body: {\n model: \"openrouter/free\",\n messages: [{ role: \"user\", content: prompt }],\n },\n});\n\nconst aiText = aiResponse.choices?.[0]?.message?.content ?? \"[]\";\nconst match = aiText.match(/\\[[\\s\\S]*\\]/);\n\nconst f1 = \"Wie hat dir das Buch gefallen?\";\nconst f2 = \"Was war der beste Teil?\";\nconst f3 = \"Was hast du mitgenommen?\";\nconst f4 = \"Wem empfiehlst du es?\";\nconst fallback = [f1, f2, f3, f4];\n\nconst questions = match ? JSON.parse(match[0]) : fallback;\n\nreturn [{ json: { ...book, questions } }];\n"
"jsCode": "const book = $input.first().json;\nif (book.skip) return [{ json: { skip: true } }];\n\nconst parts = [];\nparts.push(\"Du hilfst jemandem eine Buchbewertung zu schreiben.\");\nparts.push(\"Das Buch ist \" + book.title + \" von \" + book.author + \".\");\nparts.push(\"Erstelle 4 kurze spezifische Fragen zum Buch.\");\nparts.push(\"Die Fragen sollen helfen eine Review zu schreiben.\");\nparts.push(\"Frage auf Deutsch.\");\nparts.push(\"Antworte NUR als JSON Array mit 4 Strings. Verwende keine Bindestriche, Em-Dashes oder Gedankenstriche (, —, -).\");\nconst prompt = parts.join(\" \");\n\nconst aiResponse = await this.helpers.httpRequest({\n method: \"POST\",\n url: \"https://openrouter.ai/api/v1/chat/completions\",\n headers: {\n \"Content-Type\": \"application/json\",\n Authorization: \"Bearer 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\",\n },\n body: {\n model: \"openrouter/free\",\n messages: [{ role: \"user\", content: prompt }],\n },\n});\n\nconst aiText = aiResponse.choices?.[0]?.message?.content ?? \"[]\";\nconst match = aiText.match(/\\[[\\s\\S]*\\]/);\n\nconst f1 = \"Wie hat dir das Buch gefallen?\";\nconst f2 = \"Was war der beste Teil?\";\nconst f3 = \"Was hast du mitgenommen?\";\nconst f4 = \"Wem empfiehlst du es?\";\nconst fallback = [f1, f2, f3, f4];\n\nconst questions = match ? JSON.parse(match[0]) : fallback;\n\nreturn [{ json: { ...book, questions } }];\n"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,

View File

@@ -0,0 +1,740 @@
{
"name": "🎯 ULTIMATE Telegram CMS COMPLETE",
"nodes": [
{
"parameters": {
"updates": [
"message",
"callback_query"
],
"additionalFields": {}
},
"type": "n8n-nodes-base.telegramTrigger",
"typeVersion": 1.2,
"position": [
0,
240
],
"id": "telegram-trigger-001",
"name": "Telegram Trigger",
"webhookId": "telegram-cms-webhook-001",
"credentials": {
"telegramApi": {
"id": "ADurvy9EKUDzbDdq",
"name": "DK0_Server"
}
}
},
{
"parameters": {
"jsCode": "const input = $input.first().json;\nconst token = '8166414331:AAGNQ6fn2juD5esaTRxPjtTdSMkwq_oASIc';\n\nif (input.callback_query) {\n const cbq = input.callback_query;\n const chatId = cbq.message.chat.id;\n const data = cbq.data;\n const callbackQueryId = cbq.id;\n \n if (token) {\n try {\n await this.helpers.httpRequest({ \n method: 'POST', \n url: 'https://api.telegram.org/bot' + token + '/answerCallbackQuery', \n headers: { 'Content-Type': 'application/json' }, \n body: { callback_query_id: callbackQueryId } \n });\n } catch(e) {}\n }\n \n const parts = data.split(':');\n const action = parts[0];\n \n if (action === 'start') return [{ json: { action: 'start', chatId } }];\n if (action === 'stats') return [{ json: { action: 'stats', chatId } }];\n if (action === 'list') return [{ json: { action: 'list', type: parts[1], page: parseInt(parts[2] || '1'), chatId } }];\n if (action === 'preview') return [{ json: { action: 'preview', collectionType: parts[1], id: parts[2], chatId } }];\n if (action === 'publish') return [{ json: { action: 'publish', collectionType: parts[1], id: parts[2], chatId } }];\n if (action === 'delete') return [{ json: { action: 'delete', collectionType: parts[1], id: parts[2], chatId } }];\n if (action === 'review_info') return [{ json: { action: 'review_info', id: parts[1], chatId } }];\n \n return [{ json: { action: 'unknown', chatId } }];\n}\n\nconst text = input.message?.text ?? '';\nconst chatId = input.message?.chat?.id;\nlet match;\n\nif (text === '/start') return [{ json: { action: 'start', chatId } }];\nif (text === '/stats') return [{ json: { action: 'stats', chatId } }];\n\nmatch = text.match(/^\\/list\\s+(projects|books)(?:\\s+(\\d+))?/);\nif (match) return [{ json: { action: 'list', type: match[1], page: parseInt(match[2] || '1'), chatId } }];\n\nmatch = text.match(/^\\/preview\\s*(project|book)?(\\d+)/);\nif (match) {\n const typePrefix = match[1] === 'project' ? 'projects' : match[1] === 'book' ? 'book_reviews' : 'projects';\n return [{ json: { action: 'preview', collectionType: typePrefix, id: match[2], chatId } }];\n}\n\nmatch = text.match(/^\\/search\\s+(.+)/);\nif (match) return [{ json: { action: 'search', query: match[1].trim(), chatId } }];\n\nmatch = text.match(/^\\/publish\\s*(project|book)?(\\d+)/);\nif (match) {\n const typePrefix = match[1] === 'book' ? 'books' : 'projects';\n return [{ json: { action: 'publish', collectionType: typePrefix, id: match[2], chatId } }];\n}\n\nmatch = text.match(/^\\/delete\\s*(project|book)?(\\d+)/);\nif (match) {\n const typePrefix = match[1] === 'book' ? 'books' : 'projects';\n return [{ json: { action: 'delete', collectionType: typePrefix, id: match[2], chatId } }];\n}\n\n// .review HC_ID [RATING] -> starts review process with AI questions\nmatch = text.match(/^\\.review\\s+(\\d+)(?:\\s+([1-5]))?/);\nif (match) return [{ json: { action: 'review_info', hardcoverId: match[1], rating: match[2] ? parseInt(match[2]) : 0, chatId } }];\n\n// .answer BOOK_ID RATING your answers -> submit review answers\nmatch = text.match(/^\\.answer\\s+(\\d+)\\s+([1-5])\\s+(.+)/);\nif (match) return [{ json: { action: 'answer_review', bookId: match[1], rating: parseInt(match[2]), answers: match[3].trim(), chatId } }];\n\nmatch = text.match(/^\\.refine\\s+(\\d+)\\s+(.+)/);\nif (match) return [{ json: { action: 'refine_review', id: match[1], feedback: match[2].trim(), chatId } }];\n\nreturn [{ json: { action: 'unknown', chatId } }];\n"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [
240,
240
],
"id": "global-parser-001",
"name": "Global Parser"
},
{
"parameters": {
"rules": {
"values": [
{
"conditions": {
"conditions": [
{
"leftValue": "={{ $json.action }}",
"rightValue": "start",
"operator": {
"type": "string",
"operation": "equals"
}
}
],
"combinator": "and",
"options": {
"caseSensitive": true,
"leftValue": ""
}
},
"renameOutput": true,
"outputKey": "start"
},
{
"conditions": {
"conditions": [
{
"leftValue": "={{ $json.action }}",
"rightValue": "list",
"operator": {
"type": "string",
"operation": "equals"
}
}
],
"combinator": "and",
"options": {
"caseSensitive": true,
"leftValue": ""
}
},
"renameOutput": true,
"outputKey": "list"
},
{
"conditions": {
"conditions": [
{
"leftValue": "={{ $json.action }}",
"rightValue": "search",
"operator": {
"type": "string",
"operation": "equals"
}
}
],
"combinator": "and",
"options": {
"caseSensitive": true,
"leftValue": ""
}
},
"renameOutput": true,
"outputKey": "search"
},
{
"conditions": {
"conditions": [
{
"leftValue": "={{ $json.action }}",
"rightValue": "stats",
"operator": {
"type": "string",
"operation": "equals"
}
}
],
"combinator": "and",
"options": {
"caseSensitive": true,
"leftValue": ""
}
},
"renameOutput": true,
"outputKey": "stats"
},
{
"conditions": {
"conditions": [
{
"leftValue": "={{ $json.action }}",
"rightValue": "preview",
"operator": {
"type": "string",
"operation": "equals"
}
}
],
"combinator": "and",
"options": {
"caseSensitive": true,
"leftValue": ""
}
},
"renameOutput": true,
"outputKey": "preview"
},
{
"conditions": {
"conditions": [
{
"leftValue": "={{ $json.action }}",
"rightValue": "publish",
"operator": {
"type": "string",
"operation": "equals"
}
}
],
"combinator": "and",
"options": {
"caseSensitive": true,
"leftValue": ""
}
},
"renameOutput": true,
"outputKey": "publish"
},
{
"conditions": {
"conditions": [
{
"leftValue": "={{ $json.action }}",
"rightValue": "delete",
"operator": {
"type": "string",
"operation": "equals"
}
}
],
"combinator": "and",
"options": {
"caseSensitive": true,
"leftValue": ""
}
},
"renameOutput": true,
"outputKey": "delete"
},
{
"conditions": {
"conditions": [
{
"leftValue": "={{ $json.action }}",
"rightValue": "delete_review",
"operator": {
"type": "string",
"operation": "equals"
}
}
],
"combinator": "and",
"options": {
"caseSensitive": true,
"leftValue": ""
}
},
"renameOutput": true,
"outputKey": "delete_review"
},
{
"conditions": {
"conditions": [
{
"leftValue": "={{ $json.action }}",
"rightValue": "answer_review",
"operator": {
"type": "string",
"operation": "equals"
}
}
],
"combinator": "and",
"options": {
"caseSensitive": true,
"leftValue": ""
}
},
"renameOutput": true,
"outputKey": "answer_review"
},
{
"conditions": {
"options": {
"caseSensitive": true,
"leftValue": ""
},
"conditions": [
{
"leftValue": "={{ $json.action }}",
"rightValue": "refine_review",
"operator": {
"type": "string",
"operation": "equals"
}
}
],
"combinator": "and"
},
"renameOutput": true,
"outputKey": "refine_review"
},
{
"conditions": {
"conditions": [
{
"leftValue": "={{ $json.action }}",
"rightValue": "unknown",
"operator": {
"type": "string",
"operation": "equals"
}
}
],
"combinator": "and",
"options": {
"caseSensitive": true,
"leftValue": ""
}
},
"renameOutput": true,
"outputKey": "unknown"
},
{
"conditions": {
"conditions": [
{
"leftValue": "={{ $json.action }}",
"rightValue": "review_info",
"operator": {
"type": "string",
"operation": "equals"
}
}
],
"combinator": "and",
"options": {
"caseSensitive": true,
"leftValue": ""
}
},
"renameOutput": true,
"outputKey": "review_info"
}
]
},
"options": {}
},
"type": "n8n-nodes-base.switch",
"typeVersion": 3.2,
"position": [
480,
240
],
"id": "router-001",
"name": "Command Router"
},
{
"parameters": {
"jsCode": "\ntry {\n var chatId = $input.first().json.chatId;\n var projectsResp = await this.helpers.httpRequest({ method: 'GET', url: 'https://cms.dk0.dev/items/projects?aggregate[count]=id&filter[status][_eq]=draft', headers: { 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB' } });\n var draftProjects = (projectsResp && projectsResp.data && projectsResp.data[0] && projectsResp.data[0].count && projectsResp.data[0].count.id) || 0;\n var booksResp = await this.helpers.httpRequest({ method: 'GET', url: 'https://cms.dk0.dev/items/book_reviews?aggregate[count]=id&filter[status][_eq]=draft', headers: { 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB' } });\n var draftBooks = (booksResp && booksResp.data && booksResp.data[0] && booksResp.data[0].count && booksResp.data[0].count.id) || 0;\n var message = '\\u{1F3AF} <b>DK0 Portfolio CMS</b>\\n\\n\\u{1F4CA} <b>Status:</b>\\n\\u2022 Draft Projects: ' + draftProjects + '\\n\\u2022 Draft Reviews: ' + draftBooks + '\\n\\nTap a button to navigate.';\n var keyboard = [\n [{ text: '\\u{1F4CB} Projects', callback_data: 'list:projects:1' }, { text: '\\u{1F4DA} Books', callback_data: 'list:books:1' }],\n [{ text: '\\u{1F4CA} Stats', callback_data: 'stats' }]\n ];\n return [{ json: { chatId: chatId, message: message, parseMode: 'HTML', keyboard: keyboard } }];\n} catch (error) {\n return [{ json: { chatId: $input.first().json.chatId, message: '\\u274C Error loading dashboard: ' + error.message, parseMode: 'HTML' } }];\n}\n"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [
720,
-120
],
"id": "dashboard-001",
"name": "Dashboard Handler"
},
{
"parameters": {
"jsCode": "\ntry {\n var input = $input.first().json;\n var type = input.type;\n var page = input.page || 1;\n var chatId = input.chatId;\n var limit = 5;\n var offset = (page - 1) * limit;\n var collection = type === 'projects' ? 'projects' : 'book_reviews';\n var fields = type === 'projects' ? 'id,slug,category,status,date_created,translations.*' : 'id,book_title,rating,status,finished_at';\n var url = 'https://cms.dk0.dev/items/' + collection + '?limit=' + limit + '&offset=' + offset + '&sort=' + (type === 'projects' ? '-date_created' : '-finished_at') + '&fields=' + fields;\n var response = await this.helpers.httpRequest({ method: 'GET', url: url, headers: { 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB' } });\n var items = (response && response.data) || [];\n if (items.length === 0) {\n return [{ json: { chatId: chatId, message: 'No ' + type + ' found.', parseMode: 'HTML', keyboard: [[{ text: '\\u{1F3E0} Home', callback_data: 'start' }]] } }];\n }\n var message = '<b>' + type.toUpperCase() + ' (Page ' + page + ')</b>\\n\\n';\n var keyboard = [];\n items.forEach(function(item, idx) {\n var num = idx + 1;\n var displayNum = (offset || 0) + num;\n if (type === 'projects') {\n var title = (item.translations && item.translations[0] && item.translations[0].title) || item.slug || 'Untitled';\n message += displayNum + '. <b>' + title + '</b>\\n ' + (item.category || 'N/A') + ' | ' + item.status + '\\n\\n';\n } else {\n var stars = '';\n for (var s = 0; s < (item.rating || 0); s++) { stars += '\\u2B50'; }\n message += displayNum + '. <b>' + (item.book_title || 'Untitled') + '</b>\\n ' + stars + ' | ' + item.status + '\\n\\n';\n }\n var row = [\n { text: '\\u{1F441} #' + displayNum, callback_data: 'preview:' + type + ':' + item.id },\n { text: '\\u2705 Pub #' + displayNum, callback_data: 'publish:' + type + ':' + item.id }\n ];\n if (type === 'books' && item.status === 'draft') {\n row.push({ text: '\\u270D\\uFE0F Review #' + displayNum, callback_data: 'review_info:' + item.id });\n }\n row.push({ text: '\\u{1F5D1} Del #' + displayNum, callback_data: 'delete:' + type + ':' + item.id });\n keyboard.push(row);\n });\n var navRow = [];\n if (page > 1) { navRow.push({ text: '\\u2190 Prev', callback_data: 'list:' + type + ':' + (page - 1) }); }\n if (items.length === limit) { navRow.push({ text: 'Next \\u2192', callback_data: 'list:' + type + ':' + (page + 1) }); }\n navRow.push({ text: '\\u{1F3E0} Home', callback_data: 'start' });\n keyboard.push(navRow);\n return [{ json: { chatId: chatId, message: message, parseMode: 'HTML', keyboard: keyboard } }];\n} catch (error) {\n return [{ json: { chatId: $input.first().json.chatId, message: '\\u274C Error fetching list: ' + error.message, parseMode: 'HTML' } }];\n}\n"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [
720,
0
],
"id": "list-handler-001",
"name": "List Handler"
},
{
"parameters": {
"jsCode": "try {\n var input = $input.first().json;\n var query = input.query;\n var chatId = input.chatId;\n var encoded = encodeURIComponent(query);\n var projectsResp = await this.helpers.httpRequest({ method: 'GET', url: 'https://cms.dk0.dev/items/projects?filter[translations][title][_contains]=' + encoded + '&limit=5&fields=id,slug,category,translations.*', headers: { 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB' } });\n var booksResp = await this.helpers.httpRequest({ method: 'GET', url: 'https://cms.dk0.dev/items/book_reviews?filter[book_title][_contains]=' + encoded + '&limit=5&fields=id,book_title,book_author,rating', headers: { 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB' } });\n var projects = (projectsResp && projectsResp.data) || [];\n var books = (booksResp && booksResp.data) || [];\n if (projects.length === 0 && books.length === 0) {\n return [{ json: { chatId: chatId, message: '\\u{1F50D} No results for \"' + query + '\"', parseMode: 'HTML', keyboard: [[{ text: '\\u{1F3E0} Home', callback_data: 'start' }]] } }];\n }\n var message = '\\u{1F50D} <b>Search: \"' + query + '\"</b>\\n\\n';\n var keyboard = [];\n if (projects.length > 0) {\n message += '\\u{1F4C1} <b>Projects (' + projects.length + '):</b>\\n';\n projects.forEach(function(p) {\n var title = (p.translations && p.translations[0] && p.translations[0].title) || p.slug || 'Untitled';\n message += '\\u2022 ' + title + '\\n';\n keyboard.push([{ text: '\\u{1F441} ' + title, callback_data: 'preview:projects:' + p.id }]);\n });\n message += '\\n';\n }\n if (books.length > 0) {\n message += '\\u{1F4DA} <b>Books (' + books.length + '):</b>\\n';\n books.forEach(function(b) {\n message += '\\u2022 ' + b.book_title + ' by ' + b.book_author + '\\n';\n keyboard.push([{ text: '\\u{1F441} ' + b.book_title, callback_data: 'preview:books:' + b.id }]);\n });\n }\n keyboard.push([{ text: '\\u{1F3E0} Home', callback_data: 'start' }]);\n return [{ json: { chatId: chatId, message: message, parseMode: 'HTML', keyboard: keyboard } }];\n} catch (error) {\n return [{ json: { chatId: $input.first().json.chatId, message: '\\u274C Error searching: ' + error.message, parseMode: 'HTML' } }];\n}"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [
720,
120
],
"id": "search-handler-001",
"name": "Search Handler"
},
{
"parameters": {
"jsCode": "try {\n var chatId = $input.first().json.chatId;\n var projectsResp = await this.helpers.httpRequest({ method: 'GET', url: 'https://cms.dk0.dev/items/projects?fields=id,category,status,date_created', headers: { 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB' } });\n var booksResp = await this.helpers.httpRequest({ method: 'GET', url: 'https://cms.dk0.dev/items/book_reviews?fields=id,rating,status,finished_at', headers: { 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB' } });\n var projects = (projectsResp && projectsResp.data) || [];\n var books = (booksResp && booksResp.data) || [];\n var pPublished = projects.filter(function(p) { return p.status === 'published'; }).length;\n var pDraft = projects.filter(function(p) { return p.status === 'draft'; }).length;\n var pArchived = projects.filter(function(p) { return p.status === 'archived'; }).length;\n var bPublished = books.filter(function(b) { return b.status === 'published'; }).length;\n var bDraft = books.filter(function(b) { return b.status === 'draft'; }).length;\n var bAvg = books.length > 0 ? (books.reduce(function(sum, b) { return sum + (b.rating || 0); }, 0) / books.length).toFixed(1) : 0;\n var categories = {};\n projects.forEach(function(p) { if (p.category) { categories[p.category] = (categories[p.category] || 0) + 1; } });\n var message = '\\u{1F4CA} <b>DK0 Portfolio Statistics</b>\\n\\n\\u{1F4C1} <b>Projects:</b>\\n\\u2022 Total: ' + projects.length + '\\n\\u2022 Published: ' + pPublished + '\\n\\u2022 Draft: ' + pDraft + '\\n\\u2022 Archived: ' + pArchived + '\\n\\n\\u{1F4DA} <b>Book Reviews:</b>\\n\\u2022 Total: ' + books.length + '\\n\\u2022 Published: ' + bPublished + '\\n\\u2022 Draft: ' + bDraft + '\\n\\u2022 Avg Rating: ' + bAvg + '/5\\n';\n var catEntries = Object.entries(categories).sort(function(a, b) { return b[1] - a[1]; });\n if (catEntries.length > 0) {\n message += '\\n\\u{1F3F7}\\uFE0F <b>Categories:</b>\\n';\n catEntries.forEach(function(entry) { message += '\\u2022 ' + entry[0] + ': ' + entry[1] + '\\n'; });\n }\n var keyboard = [\n [{ text: '\\u{1F4CB} Projects', callback_data: 'list:projects:1' }, { text: '\\u{1F4DA} Books', callback_data: 'list:books:1' }],\n [{ text: '\\u{1F3E0} Home', callback_data: 'start' }]\n ];\n return [{ json: { chatId: chatId, message: message, parseMode: 'HTML', keyboard: keyboard } }];\n} catch (error) {\n return [{ json: { chatId: $input.first().json.chatId, message: '\\u274C Error loading stats: ' + error.message, parseMode: 'HTML' } }];\n}"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [
720,
240
],
"id": "stats-handler-001",
"name": "Stats Handler"
},
{
"parameters": {
"jsCode": "\ntry {\n var input = $input.first().json;\n var id = input.id;\n var chatId = input.chatId;\n var collectionType = input.collectionType;\n \n var response, collection;\n \n if (collectionType === 'projects' || collectionType === 'project') {\n response = await this.helpers.httpRequest({ method: 'GET', url: 'https://cms.dk0.dev/items/projects/' + id + '?fields=id,slug,category,status,date_created,translations.*', headers: { 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB' }, returnFullResponse: true }).catch(function() { return null; });\n collection = 'projects';\n } else if (collectionType === 'books' || collectionType === 'book') {\n response = await this.helpers.httpRequest({ method: 'GET', url: 'https://cms.dk0.dev/items/book_reviews/' + id + '?fields=id,book_title,book_author,rating,status,hardcover_id,translations.*', headers: { 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB' }, returnFullResponse: true }).catch(function() { return null; });\n collection = 'book_reviews';\n } else {\n response = await this.helpers.httpRequest({ method: 'GET', url: 'https://cms.dk0.dev/items/projects/' + id + '?fields=id,slug,category,status,date_created,translations.*', headers: { 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB' }, returnFullResponse: true }).catch(function() { return null; });\n collection = 'projects';\n var itemTry = response && response.body && response.body.data;\n if (!itemTry) {\n response = await this.helpers.httpRequest({ method: 'GET', url: 'https://cms.dk0.dev/items/book_reviews/' + id + '?fields=id,book_title,book_author,rating,status,hardcover_id,translations.*', headers: { 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB' }, returnFullResponse: true }).catch(function() { return null; });\n collection = 'book_reviews';\n }\n }\n\n var item = response && response.body && response.body.data;\n if (!item) {\n return [{ json: { chatId: chatId, message: '\\u274C Item #' + id + ' not found.', parseMode: 'HTML', keyboard: [[{ text: '\\u{1F3E0} Home', callback_data: 'start' }]] } }];\n }\n \n var message = '\\u{1F441}\\uFE0F <b>Preview #' + id + '</b>\\n\\n';\n if (collection === 'projects') {\n message += '\\u{1F4C1} <b>Type:</b> Project\\n\\u{1F516} <b>Slug:</b> ' + item.slug + '\\n\\u{1F3F7}\\uFE0F <b>Category:</b> ' + (item.category || 'N/A') + '\\n\\u{1F4CA} <b>Status:</b> ' + item.status + '\\n\\n';\n var translations = item.translations || [];\n translations.forEach(function(t) {\n var lang = t.languages_code === 'en-US' ? '\\u{1F1EC}\\u{1F1E7} EN' : '\\u{1F1E9}\\u{1F1EA} DE';\n message += lang + ':\\n<b>Title:</b> ' + (t.title || 'N/A') + '\\n<b>Desc:</b> ' + ((t.description || 'N/A')) + '...\\n\\n';\n });\n } else {\n message += '\\u{1F4DA} <b>Type:</b> Book Review\\n\\u{1F4D6} <b>Title:</b> ' + item.book_title + '\\n\\u270D\\uFE0F <b>Author:</b> ' + item.book_author + '\\n\\u2B50 <b>Rating:</b> ' + item.rating + '/5\\n\\u{1F4CA} <b>Status:</b> ' + item.status + '\\n\\u{1F517} <b>HC-ID:</b> ' + item.hardcover_id + '\\n\\n';\n var translations = item.translations || [];\n translations.forEach(function(t) {\n var lang = t.languages_code === 'en-US' ? '\\u{1F1EC}\\u{1F1E7} EN' : '\\u{1F1E9}\\u{1F1EA} DE';\n message += lang + ':\\n' + ((t.review || 'No review')) + '...\\n\\n';\n });\n }\n var listType = collection === 'projects' ? 'projects' : 'books';\n var keyboard = [\n [{ text: '\\u2705 Publish', callback_data: 'publish:' + listType + ':' + id }, { text: '\\u{1F5D1} Delete', callback_data: 'delete:' + listType + ':' + id }],\n [{ text: '\\u2190 Back', callback_data: 'list:' + listType + ':1' }, { text: '\\u{1F3E0} Home', callback_data: 'start' }]\n ];\n return [{ json: { chatId: chatId, message: message, parseMode: 'HTML', keyboard: keyboard } }];\n} catch (error) {\n return [{ json: { chatId: $input.first().json.chatId, message: '\\u274C Error loading preview: ' + error.message, parseMode: 'HTML' } }];\n}\n"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [
720,
360
],
"id": "preview-handler-001",
"name": "Preview Handler"
},
{
"parameters": {
"jsCode": "try {\n var input = $input.first().json;\n var id = input.id;\n var chatId = input.chatId;\n var collectionType = input.collectionType;\n \n var url, title, listType;\n \n if (collectionType === 'projects' || collectionType === 'project') {\n url = 'https://cms.dk0.dev/items/projects/' + id;\n title = 'Project';\n listType = 'projects';\n } else {\n url = 'https://cms.dk0.dev/items/book_reviews/' + id;\n title = 'Book Review';\n listType = 'books';\n }\n \n var response;\n try {\n response = await this.helpers.httpRequest({\n method: 'PATCH',\n url: url,\n headers: {\n 'Content-Type': 'application/json',\n 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB'\n },\n body: { status: 'published' }\n });\n } catch(e) {\n return [{ json: { chatId: chatId, message: '\\u274C <b>Publish fehlgeschlagen</b>\\n\\n' + e.message, parseMode: 'HTML', keyboard: [[{ text: '\\u{1F3E0} Home', callback_data: 'start' }]] } }];\n }\n \n var result = response.data || response;\n if (!result || !result.id) {\n return [{ json: { chatId: chatId, message: '\\u274C <b>Publish fehlgeschlagen</b>\\n\\nKeine Bestaetigung von Directus.', parseMode: 'HTML', keyboard: [[{ text: '\\u{1F3E0} Home', callback_data: 'start' }]] } }];\n }\n \n var keyboard = [[{ text: '\\u{1F4CB} ' + (listType === 'projects' ? 'Projects' : 'Books'), callback_data: 'list:' + listType + ':1' }, { text: '\\u{1F3E0} Home', callback_data: 'start' }]];\n return [{ json: { chatId: chatId, message: '\\u2705 <b>' + title + ' #' + id + ' Published!</b>\\n\\nNow live on dk0.dev.', parseMode: 'HTML', keyboard: keyboard } }];\n} catch (error) {\n return [{ json: { chatId: $input.first().json.chatId, message: '\\u274C Error publishing: ' + error.message, parseMode: 'HTML' } }];\n}"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [
720,
480
],
"id": "publish-handler-001",
"name": "Publish Handler"
},
{
"parameters": {
"jsCode": "try {\n var input = $input.first().json;\n var id = input.id;\n var chatId = input.chatId;\n var collectionType = input.collectionType;\n \n var response, collection, title;\n \n if (collectionType === 'projects' || collectionType === 'project') {\n response = await this.helpers.httpRequest({ method: 'DELETE', url: 'https://cms.dk0.dev/items/projects/' + id, headers: { 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB' }, returnFullResponse: true }).catch(function() { return null; });\n collection = 'projects';\n title = 'Project';\n } else if (collectionType === 'books' || collectionType === 'book') {\n response = await this.helpers.httpRequest({ method: 'DELETE', url: 'https://cms.dk0.dev/items/book_reviews/' + id, headers: { 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB' }, returnFullResponse: true }).catch(function() { return null; });\n collection = 'book_reviews';\n title = 'Book Review';\n } else {\n // Fallback\n response = await this.helpers.httpRequest({ method: 'DELETE', url: 'https://cms.dk0.dev/items/projects/' + id, headers: { 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB' }, returnFullResponse: true }).catch(function() { return null; });\n collection = 'projects';\n title = 'Project';\n if (!response || response.statusCode >= 400) {\n response = await this.helpers.httpRequest({ method: 'DELETE', url: 'https://cms.dk0.dev/items/book_reviews/' + id, headers: { 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB' }, returnFullResponse: true }).catch(function() { return null; });\n collection = 'book_reviews';\n title = 'Book Review';\n }\n }\n\n if (!response || response.statusCode >= 400) {\n return [{ json: { chatId: chatId, message: '\\u274C Item #' + id + ' could not be deleted.', parseMode: 'HTML', keyboard: [[{ text: '\\u{1F3E0} Home', callback_data: 'start' }]] } }];\n }\n var listType = collection === 'projects' ? 'projects' : 'books';\n var keyboard = [[{ text: (collection === 'projects' ? '\\u{1F4CB} Projects' : '\\u{1F4DA} Books'), callback_data: 'list:' + listType + ':1' }, { text: '\\u{1F3E0} Home', callback_data: 'start' }]];\n return [{ json: { chatId: chatId, message: '\\u{1F5D1}\\uFE0F *' + title + ' #' + id + ' Deleted*', parseMode: 'HTML', keyboard: keyboard, collection: collection, itemId: id } }];\n} catch (error) {\n return [{ json: { chatId: $input.first().json.chatId, message: '\\u274C Error deleting: ' + error.message, parseMode: 'HTML' } }];\n}"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [
720,
600
],
"id": "delete-handler-001",
"name": "Delete Handler"
},
{
"parameters": {
"jsCode": "try {\n var input = $input.first().json;\n var id = input.id;\n var chatId = input.chatId;\n var bookResp = await this.helpers.httpRequest({ method: 'GET', url: 'https://cms.dk0.dev/items/book_reviews/' + id + '?fields=id,book_title,translations.id', headers: { 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB' } });\n var book = bookResp && bookResp.data;\n if (!book) {\n return [{ json: { chatId: chatId, message: '\\u274C Book review #' + id + ' not found.', parseMode: 'HTML', keyboard: [[{ text: '\\u{1F3E0} Home', callback_data: 'start' }]] } }];\n }\n var translations = book.translations || [];\n var deletedCount = 0;\n for (var i = 0; i < translations.length; i++) {\n await this.helpers.httpRequest({ method: 'DELETE', url: 'https://cms.dk0.dev/items/book_reviews_translations/' + translations[i].id, headers: { 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB' } }).catch(function() {});\n deletedCount++;\n }\n var keyboard = [[{ text: '\\u{1F4DA} Books', callback_data: 'list:books:1' }, { text: '\\u{1F3E0} Home', callback_data: 'start' }]];\n return [{ json: { chatId: chatId, message: '\\u{1F5D1}\\uFE0F Deleted ' + deletedCount + ' review translations for \"' + book.book_title + '\".\\n\\nBook entry still exists.', parseMode: 'HTML', keyboard: keyboard, itemId: id, deletedCount: deletedCount } }];\n} catch (error) {\n return [{ json: { chatId: $input.first().json.chatId, message: '\\u274C Error deleting review: ' + error.message, parseMode: 'HTML' } }];\n}"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [
720,
720
],
"id": "delete-review-handler-001",
"name": "Delete Review Handler"
},
{
"parameters": {
"jsCode": "try {\n var input = $input.first().json;\n var bookId = input.bookId;\n var rating = input.rating;\n var answers = input.answers;\n var chatId = input.chatId;\n\n var bookResp = await this.helpers.httpRequest({ method: 'GET', url: 'https://cms.dk0.dev/items/book_reviews/' + bookId + '?fields=id,book_title,book_author,rating,translations.id,translations.languages_code,translations.review', headers: { 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB' } });\n var bookData = bookResp && bookResp.data ? bookResp.data : bookResp;\n if (!bookData || !bookData.id) {\n return [{ json: { chatId: chatId, message: '\\u274C Buch #' + bookId + ' nicht gefunden.', parseMode: 'HTML', keyboard: [[{ text: '\\u{1F3E0} Home', callback_data: 'start' }]] } }];\n }\n\n var prompt = 'Schreibe eine authentische Buchbewertung. Buch: ' + bookData.book_title + ' von ' + bookData.book_author + '. Rating: ' + rating + '/5. Antworten des Lesers auf Fragen zum Buch: ' + answers + ' Schreibe Ich-Perspektive, 4-6 Saetze pro Sprache. Verwende keine Bindestriche, Em-Dashes oder Gedankenstriche. Antworte NUR als JSON: {\"review_en\": \"English review\", \"review_de\": \"Deutsche Bewertung\"}';\n\n var aiResp = await this.helpers.httpRequest({ method: 'POST', url: 'https://openrouter.ai/api/v1/chat/completions', headers: { 'Content-Type': 'application/json', 'Authorization': 'Bearer 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' }, body: { model: 'openrouter/free', messages: [{ role: 'user', content: prompt }] } });\n var aiText = (aiResp && aiResp.choices && aiResp.choices[0] && aiResp.choices[0].message && aiResp.choices[0].message.content) || '{}';\n var jsonMatch = aiText.match(/\\{[\\s\\S]*\\}/);\n var ai = jsonMatch ? JSON.parse(jsonMatch[0]) : { review_en: answers, review_de: answers };\n\n // Update rating\n await this.helpers.httpRequest({ method: 'PATCH', url: 'https://cms.dk0.dev/items/book_reviews/' + bookData.id, headers: { 'Content-Type': 'application/json', 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB' }, body: { rating: rating } });\n\n var translations = bookData.translations || [];\n var enTrans = null, deTrans = null;\n for (var i = 0; i < translations.length; i++) {\n if (translations[i].languages_code === 'en-US') enTrans = translations[i];\n if (translations[i].languages_code === 'de-DE') deTrans = translations[i];\n }\n\n var reviewEn = ai.review_en || answers;\n var reviewDe = ai.review_de || answers;\n\n // Update existing translations (PATCH) or create new ones (POST)\n if (enTrans) {\n await this.helpers.httpRequest({ method: 'PATCH', url: 'https://cms.dk0.dev/items/book_reviews_translations/' + enTrans.id, headers: { 'Content-Type': 'application/json', 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB' }, body: { review: reviewEn } });\n } else {\n await this.helpers.httpRequest({ method: 'POST', url: 'https://cms.dk0.dev/items/book_reviews_translations', headers: { 'Content-Type': 'application/json', 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB' }, body: { book_reviews_id: bookData.id, languages_code: 'en-US', review: reviewEn } });\n }\n\n if (deTrans) {\n await this.helpers.httpRequest({ method: 'PATCH', url: 'https://cms.dk0.dev/items/book_reviews_translations/' + deTrans.id, headers: { 'Content-Type': 'application/json', 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB' }, body: { review: reviewDe } });\n } else {\n await this.helpers.httpRequest({ method: 'POST', url: 'https://cms.dk0.dev/items/book_reviews_translations', headers: { 'Content-Type': 'application/json', 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB' }, body: { book_reviews_id: bookData.id, languages_code: 'de-DE', review: reviewDe } });\n }\n\n var showEn = reviewEn;\n var showDe = reviewDe;\n var msg = '\\u2705 <b>Review erstellt!</b>\\n\\n\\u{1F4DA} ' + bookData.book_title + ' (' + rating + '/5)\\n\\n<b>EN:</b> ' + showEn + '\\n\\n<b>DE:</b> ' + showDe;\n var keyboard = [[{ text: '\\u{1F441} Preview', callback_data: 'preview:books:' + bookData.id }, { text: '\\u2705 Publish', callback_data: 'publish:books:' + bookData.id }], [{ text: '\\u{1F4DA} Books', callback_data: 'list:books:1' }]];\n return [{ json: { chatId: chatId, message: msg, parseMode: 'HTML', keyboard: keyboard } }];\n} catch (error) {\n return [{ json: { chatId: $input.first().json.chatId, message: '\\u274C Fehler beim Erstellen der Review: ' + error.message, parseMode: 'HTML' } }];\n}"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [
720,
840
],
"id": "create-review-handler-001",
"name": "Create Review Handler"
},
{
"parameters": {
"jsCode": "var chatId = $input.first().json.chatId;\nvar message = '\\u2753 <b>Unknown Command</b>\\n\\nUse the buttons below or type:\\n<code>.review HC_ID [RATING]</code> - Start review with AI questions\\n<code>.answer BOOK_ID RATING your answers</code> - Submit review answers\\n<code>.refine ID FEEDBACK</code> - Refine existing review';\nvar keyboard = [\n [{ text: '\\u{1F4CB} Projects', callback_data: 'list:projects:1' }, { text: '\\u{1F4DA} Books', callback_data: 'list:books:1' }],\n [{ text: '\\u{1F4CA} Stats', callback_data: 'stats' }, { text: '\\u{1F3E0} Dashboard', callback_data: 'start' }]\n];\nreturn [{ json: { chatId: chatId, message: message, parseMode: 'HTML', keyboard: keyboard } }];"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [
720,
960
],
"id": "unknown-handler-001",
"name": "Unknown Command Handler"
},
{
"parameters": {
"method": "POST",
"url": "={{ 'https://api.telegram.org/bot8166414331:AAGNQ6fn2juD5esaTRxPjtTdSMkwq_oASIc/sendMessage' }}",
"authentication": "none",
"sendBody": true,
"contentType": "json",
"specifyBody": "json",
"jsonBody": "={{ { chat_id: $json.chatId, text: $json.message, parse_mode: $json.parseMode || 'HTML', reply_markup: ($json.keyboard && $json.keyboard.length > 0) ? { inline_keyboard: $json.keyboard } : undefined } }}"
},
"type": "n8n-nodes-base.httpRequest",
"typeVersion": 4.2,
"position": [
960,
420
],
"id": "send-message-001",
"name": "Send Message",
"options": {}
},
{
"parameters": {
"jsCode": "try {\n var input = $input.first().json;\n var id = input.id;\n var feedback = input.feedback;\n var chatId = input.chatId;\n var bookResp = await this.helpers.httpRequest({ method: 'GET', url: 'https://cms.dk0.dev/items/book_reviews/' + id + '?fields=id,book_title,book_author,rating,translations.id,translations.languages_code,translations.review', headers: { 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB' } });\n var bookData = bookResp && bookResp.data ? bookResp.data : bookResp;\n if (!bookData || !bookData.id) {\n return [{ json: { chatId: chatId, message: 'Review #' + id + ' nicht gefunden.', parseMode: 'HTML', keyboard: [[{ text: '\\u{1F3E0} Home', callback_data: 'start' }]] } }];\n }\n var translations = bookData.translations || [];\n var enTrans = null, deTrans = null;\n for (var i = 0; i < translations.length; i++) {\n if (translations[i].languages_code === 'en-US') enTrans = translations[i];\n if (translations[i].languages_code === 'de-DE') deTrans = translations[i];\n }\n var currentEn = enTrans ? enTrans.review : '';\n var currentDe = deTrans ? deTrans.review : '';\n var prompt = 'Du hast eine Buchbewertung fuer \"' + bookData.book_title + '\" von \"' + bookData.book_author + '\" geschrieben. Rating: ' + bookData.rating + '/5. Aktuelle EN-Bewertung: ' + currentEn + ' Aktuelle DE-Bewertung: ' + currentDe + ' Feedback des Lesers: ' + feedback + ' Wichtig: EN und DE sind immer inhaltlich identisch, nur die Sprache unterscheidet sich. Feedback gilt fuer BEIDE Versionen, auch wenn es nur eine Sprache erwaehnt. Ueberarbeite daher immer beide synchron. Ich-Perspektive, 4-6 Saetze pro Sprache. Verwende keine Bindestriche, Em-Dashes oder Gedankenstriche. Antworte NUR als JSON: {\"review_en\": \"...\", \"review_de\": \"...\"}';\n var aiResp = await this.helpers.httpRequest({ method: 'POST', url: 'https://openrouter.ai/api/v1/chat/completions', headers: { 'Content-Type': 'application/json', 'Authorization': 'Bearer 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' }, body: { model: 'openrouter/free', messages: [{ role: 'user', content: prompt }] } });\n var aiText = (aiResp && aiResp.choices && aiResp.choices[0] && aiResp.choices[0].message && aiResp.choices[0].message.content) || '{}';\n var jsonMatch = aiText.match(/\\{[\\s\\S]*\\}/);\n var ai = jsonMatch ? JSON.parse(jsonMatch[0]) : { review_en: feedback, review_de: feedback };\n var reviewEn = ai.review_en || feedback;\n var reviewDe = ai.review_de || feedback;\n\n // Update existing translations (PATCH) or create new ones (POST)\n if (enTrans) {\n await this.helpers.httpRequest({ method: 'PATCH', url: 'https://cms.dk0.dev/items/book_reviews_translations/' + enTrans.id, headers: { 'Content-Type': 'application/json', 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB' }, body: { review: reviewEn } });\n } else {\n await this.helpers.httpRequest({ method: 'POST', url: 'https://cms.dk0.dev/items/book_reviews_translations', headers: { 'Content-Type': 'application/json', 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB' }, body: { book_reviews_id: bookData.id, languages_code: 'en-US', review: reviewEn } });\n }\n if (deTrans) {\n await this.helpers.httpRequest({ method: 'PATCH', url: 'https://cms.dk0.dev/items/book_reviews_translations/' + deTrans.id, headers: { 'Content-Type': 'application/json', 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB' }, body: { review: reviewDe } });\n } else {\n await this.helpers.httpRequest({ method: 'POST', url: 'https://cms.dk0.dev/items/book_reviews_translations', headers: { 'Content-Type': 'application/json', 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB' }, body: { book_reviews_id: bookData.id, languages_code: 'de-DE', review: reviewDe } });\n }\n\n var showEn = reviewEn;\n var showDe = reviewDe;\n var msg = '\\u270F\\uFE0F <b>Review aktualisiert!</b>\\n\\n\\u{1F4DA} ' + bookData.book_title + '\\n\\n<b>EN:</b> ' + showEn + '\\n\\n<b>DE:</b> ' + showDe;\n var keyboard = [[{ text: '\\u{1F441} Preview', callback_data: 'preview:books:' + id }, { text: '\\u2705 Publish', callback_data: 'publish:books:' + id }], [{ text: '\\u{1F4DA} Books', callback_data: 'list:books:1' }]];\n return [{ json: { chatId: chatId, message: msg, parseMode: 'HTML', keyboard: keyboard } }];\n} catch (error) {\n return [{ json: { chatId: $input.first().json.chatId, message: '\\u274C Fehler beim Aktualisieren: ' + error.message, parseMode: 'HTML' } }];\n}"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [
720,
1080
],
"id": "refine-review-handler-001",
"name": "Refine Review Handler"
},
{
"parameters": {
"jsCode": "try {\n var input = $input.first().json;\n var chatId = input.chatId;\n var bookId = input.id;\n var hardcoverId = input.hardcoverId;\n var rating = input.rating || 0;\n var book;\n\n if (bookId) {\n var resp = await this.helpers.httpRequest({ method: 'GET', url: 'https://cms.dk0.dev/items/book_reviews/' + bookId + '?fields=id,book_title,book_author,hardcover_id,rating', headers: { 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB' } });\n book = resp && resp.data;\n } else if (hardcoverId) {\n var resp = await this.helpers.httpRequest({ method: 'GET', url: 'https://cms.dk0.dev/items/book_reviews?filter[hardcover_id][_eq]=' + hardcoverId + '&fields=id,book_title,book_author,hardcover_id,rating&limit=1', headers: { 'Authorization': 'Bearer RF2QytqhcLXuVy6FO3PzWlsoR-ysCTwB' } });\n book = resp && resp.data && resp.data[0];\n }\n\n if (!book) {\n return [{ json: { chatId: chatId, message: '\\u274C Buch nicht gefunden. Pr\\u00fcfe die ID.', parseMode: 'HTML', keyboard: [[{ text: '\\u{1F4DA} Books', callback_data: 'list:books:1' }]] } }];\n }\n\n var prompt = 'Du bist ein Leseberater. Generiere genau 4 persoenliche, tiefgruendige Fragen zum Buch \"' + book.book_title + '\" von ' + book.book_author + ', die einem helfen, eine authentische Bewertung zu schreiben. Die Fragen sollen spezifisch zum Buch sein und zum Nachdenken anregen. Antworte NUR als JSON-Array, keine Erklaerung davor: [\"Frage 1\", \"Frage 2\", \"Frage 3\", \"Frage 4\"]';\n\n var aiResp = await this.helpers.httpRequest({ method: 'POST', url: 'https://openrouter.ai/api/v1/chat/completions', headers: { 'Content-Type': 'application/json', 'Authorization': 'Bearer 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' }, body: { model: 'openrouter/free', messages: [{ role: 'user', content: prompt }] } });\n\n var aiText = (aiResp && aiResp.choices && aiResp.choices[0] && aiResp.choices[0].message && aiResp.choices[0].message.content) || '[]';\n var questions;\n try {\n var jsonMatch = aiText.match(/\\[[\\s\\S]*\\]/);\n questions = jsonMatch ? JSON.parse(jsonMatch[0]) : ['Was hat dir am besten gefallen?', 'Was hat dich gestoert?', 'Wuerdest du es weiterempfehlen?', 'Welche Szene ist dir im Gedaechtnis geblieben?'];\n } catch(e) {\n questions = ['Was hat dir am besten gefallen?', 'Was hat dich gestoert?', 'Wuerdest du es weiterempfehlen?', 'Welche Szene ist dir im Gedaechtnis geblieben?'];\n }\n\n var ratingInfo = rating > 0 ? '\\n\\u2B50 Dein Rating: ' + rating + '/5' : '\\n\\u2B50 Gib dein Rating (1-5) an';\n var msg = '\\u{1F4D6} <b>Review: ' + book.book_title + '</b>\\n' + book.book_author + ratingInfo + '\\n\\n\\u2753 <b>Beantworte diese Fragen:</b>\\n\\n';\n for (var i = 0; i < questions.length; i++) {\n msg += (i + 1) + '. ' + questions[i] + '\\n';\n }\n msg += '\\n\\u270D\\uFE0F Antworte mit:\\n<code>.answer ' + book.id + ' ' + (rating > 0 ? rating : '5') + ' deine Antworten hier</code>';\n msg += '\\n\\n<i>Beispiel: .answer ' + book.id + ' 4 Die Charakterentwicklung war super...</i>';\n\n var keyboard = [[{ text: '\\u{1F4DA} Books', callback_data: 'list:books:1' }, { text: '\\u{1F3E0} Home', callback_data: 'start' }]];\n return [{ json: { chatId: chatId, message: msg, parseMode: 'HTML', keyboard: keyboard } }];\n} catch(e) {\n return [{ json: { chatId: $input.first().json.chatId, message: '\\u274C Error: ' + e.message, parseMode: 'HTML' } }];\n}\n"
},
"type": "n8n-nodes-base.code",
"typeVersion": 2,
"position": [
720,
960
],
"id": "review-info-handler-001",
"name": "Review Info Handler"
}
],
"connections": {
"Telegram Trigger": {
"main": [
[
{
"node": "Global Parser",
"type": "main",
"index": 0
}
]
]
},
"Global Parser": {
"main": [
[
{
"node": "Command Router",
"type": "main",
"index": 0
}
]
]
},
"Command Router": {
"main": [
[
{
"node": "Dashboard Handler",
"type": "main",
"index": 0
}
],
[
{
"node": "List Handler",
"type": "main",
"index": 0
}
],
[
{
"node": "Search Handler",
"type": "main",
"index": 0
}
],
[
{
"node": "Stats Handler",
"type": "main",
"index": 0
}
],
[
{
"node": "Preview Handler",
"type": "main",
"index": 0
}
],
[
{
"node": "Publish Handler",
"type": "main",
"index": 0
}
],
[
{
"node": "Delete Handler",
"type": "main",
"index": 0
}
],
[
{
"node": "Delete Review Handler",
"type": "main",
"index": 0
}
],
[
{
"node": "Create Review Handler",
"type": "main",
"index": 0
}
],
[
{
"node": "Refine Review Handler",
"type": "main",
"index": 0
}
],
[
{
"node": "Unknown Command Handler",
"type": "main",
"index": 0
}
],
[
{
"node": "Review Info Handler",
"type": "main",
"index": 0
}
]
]
},
"Dashboard Handler": {
"main": [
[
{
"node": "Send Message",
"type": "main",
"index": 0
}
]
]
},
"List Handler": {
"main": [
[
{
"node": "Send Message",
"type": "main",
"index": 0
}
]
]
},
"Search Handler": {
"main": [
[
{
"node": "Send Message",
"type": "main",
"index": 0
}
]
]
},
"Stats Handler": {
"main": [
[
{
"node": "Send Message",
"type": "main",
"index": 0
}
]
]
},
"Preview Handler": {
"main": [
[
{
"node": "Send Message",
"type": "main",
"index": 0
}
]
]
},
"Publish Handler": {
"main": [
[
{
"node": "Send Message",
"type": "main",
"index": 0
}
]
]
},
"Delete Handler": {
"main": [
[
{
"node": "Send Message",
"type": "main",
"index": 0
}
]
]
},
"Delete Review Handler": {
"main": [
[
{
"node": "Send Message",
"type": "main",
"index": 0
}
]
]
},
"Create Review Handler": {
"main": [
[
{
"node": "Send Message",
"type": "main",
"index": 0
}
]
]
},
"Unknown Command Handler": {
"main": [
[
{
"node": "Send Message",
"type": "main",
"index": 0
}
]
]
},
"Refine Review Handler": {
"main": [
[
{
"node": "Send Message",
"type": "main",
"index": 0
}
]
]
},
"Review Info Handler": {
"main": [
[
{
"node": "Send Message",
"type": "main",
"index": 0
}
]
]
}
},
"pinData": {},
"settings": {
"executionOrder": "v1"
},
"staticData": null,
"tags": [],
"triggerCount": 1,
"updatedAt": "2025-01-21T00:00:00.000Z",
"versionId": "1"
}

View File

@@ -1,136 +0,0 @@
#!/bin/bash
# Gitea Runner Status Check Script
# Prüft den Status des Gitea Runners
set -e
# Colors for output
RED='\033[0;31m'
GREEN='\033[0;32m'
YELLOW='\033[1;33m'
BLUE='\033[0;34m'
CYAN='\033[0;36m'
NC='\033[0m' # No Color
echo -e "${BLUE}╔════════════════════════════════════════════════════════════╗${NC}"
echo -e "${BLUE}║ Gitea Runner Status Check ║${NC}"
echo -e "${BLUE}╚════════════════════════════════════════════════════════════╝${NC}"
echo ""
# Check 1: systemd service
echo -e "${CYAN}[1/5] Checking systemd service...${NC}"
if systemctl list-units --type=service --all | grep -q "gitea-runner.service"; then
echo -e "${GREEN}✓ systemd service found${NC}"
systemctl status gitea-runner --no-pager -l || true
else
echo -e "${YELLOW}⚠ systemd service not found (runner might be running differently)${NC}"
fi
echo ""
# Check 2: Running processes
echo -e "${CYAN}[2/5] Checking for running runner processes...${NC}"
RUNNER_PROCESSES=$(ps aux | grep -E "(gitea|act_runner|woodpecker)" | grep -v grep || echo "")
if [ ! -z "$RUNNER_PROCESSES" ]; then
echo -e "${GREEN}✓ Found runner processes:${NC}"
echo "$RUNNER_PROCESSES" | while read line; do
echo " $line"
done
else
echo -e "${RED}✗ No runner processes found${NC}"
fi
echo ""
# Check 3: Docker containers (if runner runs in Docker)
echo -e "${CYAN}[3/5] Checking for runner Docker containers...${NC}"
RUNNER_CONTAINERS=$(docker ps -a --filter "name=runner" --format "{{.Names}}\t{{.Status}}" 2>/dev/null || echo "")
if [ ! -z "$RUNNER_CONTAINERS" ]; then
echo -e "${GREEN}✓ Found runner containers:${NC}"
echo "$RUNNER_CONTAINERS" | while read line; do
echo " $line"
done
else
echo -e "${YELLOW}⚠ No runner containers found${NC}"
fi
echo ""
# Check 4: Common runner directories
echo -e "${CYAN}[4/5] Checking common runner directories...${NC}"
RUNNER_DIRS=(
"/tmp/gitea-runner"
"/opt/gitea-runner"
"/home/*/gitea-runner"
"~/.gitea-runner"
"/usr/local/gitea-runner"
)
FOUND_DIRS=0
for dir in "${RUNNER_DIRS[@]}"; do
# Expand ~ and wildcards
EXPANDED_DIR=$(eval echo "$dir" 2>/dev/null || echo "")
if [ -d "$EXPANDED_DIR" ]; then
echo -e "${GREEN}✓ Found runner directory: $EXPANDED_DIR${NC}"
FOUND_DIRS=$((FOUND_DIRS + 1))
# Check for config files
if [ -f "$EXPANDED_DIR/.runner" ] || [ -f "$EXPANDED_DIR/config.yml" ]; then
echo " → Contains configuration files"
fi
fi
done
if [ $FOUND_DIRS -eq 0 ]; then
echo -e "${YELLOW}⚠ No runner directories found in common locations${NC}"
fi
echo ""
# Check 5: Network connections (check if runner is connecting to Gitea)
echo -e "${CYAN}[5/5] Checking network connections to Gitea...${NC}"
GITEA_URL="${GITEA_URL:-https://git.dk0.dev}"
if command -v netstat >/dev/null 2>&1; then
CONNECTIONS=$(netstat -tn 2>/dev/null | grep -E "(git.dk0.dev|3000|3001)" || echo "")
elif command -v ss >/dev/null 2>&1; then
CONNECTIONS=$(ss -tn 2>/dev/null | grep -E "(git.dk0.dev|3000|3001)" || echo "")
fi
if [ ! -z "$CONNECTIONS" ]; then
echo -e "${GREEN}✓ Found connections to Gitea:${NC}"
echo "$CONNECTIONS" | head -5
else
echo -e "${YELLOW}⚠ No active connections to Gitea found${NC}"
fi
echo ""
# Summary
echo -e "${BLUE}━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━${NC}"
echo -e "${BLUE}Summary:${NC}"
echo ""
if [ ! -z "$RUNNER_PROCESSES" ] || [ ! -z "$RUNNER_CONTAINERS" ]; then
echo -e "${GREEN}✓ Runner appears to be running${NC}"
echo ""
echo "To check runner status in Gitea:"
echo " 1. Go to: https://git.dk0.dev/denshooter/portfolio/settings/actions/runners"
echo " 2. Check if runner-01 shows as 'online' or 'idle'"
echo ""
echo "To view runner logs:"
if [ ! -z "$RUNNER_PROCESSES" ]; then
echo " - Check process logs or journalctl"
fi
if [ ! -z "$RUNNER_CONTAINERS" ]; then
echo " - docker logs <container-name>"
fi
else
echo -e "${RED}✗ Runner does not appear to be running${NC}"
echo ""
echo "To start the runner:"
echo " 1. Find where the runner binary is located"
echo " 2. Check Gitea for registration token"
echo " 3. Run: ./act_runner register --config config.yml"
echo " 4. Run: ./act_runner daemon --config config.yml"
fi
echo ""
echo -e "${CYAN}For more information, check:${NC}"
echo " - Gitea Runner Docs: https://docs.gitea.com/usage/actions/act-runner"
echo " - Runner Status: https://git.dk0.dev/denshooter/portfolio/settings/actions/runners"
echo ""

View File

@@ -1,225 +0,0 @@
#!/bin/bash
# Simplified Gitea deployment script for testing
# This version doesn't require database dependencies
set -e
# Configuration
PROJECT_NAME="portfolio"
CONTAINER_NAME="portfolio-app-simple"
IMAGE_NAME="portfolio-app"
PORT=3000
BACKUP_PORT=3001
LOG_FILE="./logs/gitea-deploy-simple.log"
# Colors for output
RED='\033[0;31m'
GREEN='\033[0;32m'
YELLOW='\033[1;33m'
BLUE='\033[0;34m'
NC='\033[0m' # No Color
# Logging function
log() {
echo -e "${BLUE}[$(date +'%Y-%m-%d %H:%M:%S')]${NC} $1" | tee -a "$LOG_FILE"
}
error() {
echo -e "${RED}[ERROR]${NC} $1" | tee -a "$LOG_FILE"
}
success() {
echo -e "${GREEN}[SUCCESS]${NC} $1" | tee -a "$LOG_FILE"
}
warning() {
echo -e "${YELLOW}[WARNING]${NC} $1" | tee -a "$LOG_FILE"
}
# Check if running as root (skip in CI environments)
if [[ $EUID -eq 0 ]] && [[ -z "$CI" ]]; then
error "This script should not be run as root (use CI=true to override)"
exit 1
fi
# Check if Docker is running
if ! docker info > /dev/null 2>&1; then
error "Docker is not running. Please start Docker and try again."
exit 1
fi
# Check if we're in the right directory
if [ ! -f "package.json" ] || [ ! -f "Dockerfile" ]; then
error "Please run this script from the project root directory"
exit 1
fi
log "🚀 Starting simplified Gitea deployment for $PROJECT_NAME"
# Step 1: Build Application
log "🔨 Step 1: Building application..."
# Build Next.js application
log "📦 Building Next.js application..."
npm run build || {
error "Build failed"
exit 1
}
success "✅ Application built successfully"
# Step 2: Docker Operations
log "🐳 Step 2: Docker operations..."
# Build Docker image
log "🏗️ Building Docker image..."
docker build -t "$IMAGE_NAME:latest" . || {
error "Docker build failed"
exit 1
}
# Tag with timestamp
TIMESTAMP=$(date +%Y%m%d-%H%M%S)
docker tag "$IMAGE_NAME:latest" "$IMAGE_NAME:$TIMESTAMP"
success "✅ Docker image built successfully"
# Step 3: Deployment
log "🚀 Step 3: Deploying application..."
# Export environment variables for docker-compose compatibility
log "📝 Exporting environment variables..."
export NODE_ENV=${NODE_ENV:-production}
export NEXT_PUBLIC_BASE_URL=${NEXT_PUBLIC_BASE_URL:-https://dk0.dev}
export MY_EMAIL=${MY_EMAIL:-contact@dk0.dev}
export MY_INFO_EMAIL=${MY_INFO_EMAIL:-info@dk0.dev}
export MY_PASSWORD="${MY_PASSWORD}"
export MY_INFO_PASSWORD="${MY_INFO_PASSWORD}"
export ADMIN_BASIC_AUTH="${ADMIN_BASIC_AUTH}"
export LOG_LEVEL=${LOG_LEVEL:-info}
export PORT=${PORT:-3000}
# Log which variables are set (without revealing secrets)
log "Environment variables configured:"
log " - NODE_ENV: ${NODE_ENV}"
log " - NEXT_PUBLIC_BASE_URL: ${NEXT_PUBLIC_BASE_URL}"
log " - MY_EMAIL: ${MY_EMAIL}"
log " - MY_INFO_EMAIL: ${MY_INFO_EMAIL}"
log " - MY_PASSWORD: [SET]"
log " - MY_INFO_PASSWORD: [SET]"
log " - ADMIN_BASIC_AUTH: [SET]"
log " - LOG_LEVEL: ${LOG_LEVEL}"
log " - PORT: ${PORT}"
# Check if container is running
if [ "$(docker inspect -f '{{.State.Running}}' "$CONTAINER_NAME" 2>/dev/null)" = "true" ]; then
log "📦 Stopping existing container..."
docker stop "$CONTAINER_NAME" || true
docker rm "$CONTAINER_NAME" || true
fi
# Check if port is available
if lsof -Pi :$PORT -sTCP:LISTEN -t >/dev/null ; then
warning "Port $PORT is in use. Trying backup port $BACKUP_PORT"
DEPLOY_PORT=$BACKUP_PORT
else
DEPLOY_PORT=$PORT
fi
# Start new container with minimal environment variables
log "🚀 Starting new container on port $DEPLOY_PORT..."
docker run -d \
--name "$CONTAINER_NAME" \
--restart unless-stopped \
-p "$DEPLOY_PORT:3000" \
-e NODE_ENV=production \
-e NEXT_PUBLIC_BASE_URL=https://dk0.dev \
-e MY_EMAIL=contact@dk0.dev \
-e MY_INFO_EMAIL=info@dk0.dev \
-e MY_PASSWORD=test-password \
-e MY_INFO_PASSWORD=test-password \
-e ADMIN_BASIC_AUTH=admin:test123 \
-e LOG_LEVEL=info \
"$IMAGE_NAME:latest" || {
error "Failed to start container"
exit 1
}
# Wait for container to be ready
log "⏳ Waiting for container to be ready..."
sleep 20
# Check if container is actually running
if [ "$(docker inspect -f '{{.State.Running}}' "$CONTAINER_NAME" 2>/dev/null)" != "true" ]; then
error "Container failed to start or crashed"
log "Container logs:"
docker logs "$CONTAINER_NAME" --tail=50
exit 1
fi
# Health check
log "🏥 Performing health check..."
HEALTH_CHECK_TIMEOUT=180
HEALTH_CHECK_INTERVAL=5
ELAPSED=0
while [ $ELAPSED -lt $HEALTH_CHECK_TIMEOUT ]; do
# Check if container is still running
if [ "$(docker inspect -f '{{.State.Running}}' "$CONTAINER_NAME" 2>/dev/null)" != "true" ]; then
error "Container stopped during health check"
log "Container logs:"
docker logs "$CONTAINER_NAME" --tail=50
exit 1
fi
# Try health check endpoint
if curl -f "http://localhost:$DEPLOY_PORT/api/health" > /dev/null 2>&1; then
success "✅ Application is healthy!"
break
fi
sleep $HEALTH_CHECK_INTERVAL
ELAPSED=$((ELAPSED + HEALTH_CHECK_INTERVAL))
echo -n "."
done
if [ $ELAPSED -ge $HEALTH_CHECK_TIMEOUT ]; then
error "Health check timeout. Application may not be running properly."
log "Container status:"
docker inspect "$CONTAINER_NAME" --format='{{.State.Status}} - {{.State.Health.Status}}'
log "Container logs:"
docker logs "$CONTAINER_NAME" --tail=100
exit 1
fi
# Step 4: Verification
log "✅ Step 4: Verifying deployment..."
# Test main page
if curl -f "http://localhost:$DEPLOY_PORT/" > /dev/null 2>&1; then
success "✅ Main page is accessible"
else
error "❌ Main page is not accessible"
exit 1
fi
# Show container status
log "📊 Container status:"
docker ps --filter "name=$CONTAINER_NAME" --format "table {{.Names}}\t{{.Status}}\t{{.Ports}}"
# Show resource usage
log "📈 Resource usage:"
docker stats --no-stream --format "table {{.Container}}\t{{.CPUPerc}}\t{{.MemUsage}}" "$CONTAINER_NAME"
# Final success message
success "🎉 Simplified Gitea deployment completed successfully!"
log "🌐 Application is available at: http://localhost:$DEPLOY_PORT"
log "🏥 Health check endpoint: http://localhost:$DEPLOY_PORT/api/health"
log "📊 Container name: $CONTAINER_NAME"
log "📝 Logs: docker logs $CONTAINER_NAME"
# Update deployment log
echo "$(date): Simplified Gitea deployment successful - Port: $DEPLOY_PORT - Image: $IMAGE_NAME:$TIMESTAMP" >> "$LOG_FILE"
exit 0

View File

@@ -1,257 +0,0 @@
#!/bin/bash
# Gitea-specific deployment script
# Optimiert für lokalen Gitea Runner
set -e
# Configuration
PROJECT_NAME="portfolio"
CONTAINER_NAME="portfolio-app"
IMAGE_NAME="portfolio-app"
PORT=3000
BACKUP_PORT=3001
LOG_FILE="./logs/gitea-deploy.log"
# Colors for output
RED='\033[0;31m'
GREEN='\033[0;32m'
YELLOW='\033[1;33m'
BLUE='\033[0;34m'
NC='\033[0m' # No Color
# Logging function
log() {
echo -e "${BLUE}[$(date +'%Y-%m-%d %H:%M:%S')]${NC} $1" | tee -a "$LOG_FILE"
}
error() {
echo -e "${RED}[ERROR]${NC} $1" | tee -a "$LOG_FILE"
}
success() {
echo -e "${GREEN}[SUCCESS]${NC} $1" | tee -a "$LOG_FILE"
}
warning() {
echo -e "${YELLOW}[WARNING]${NC} $1" | tee -a "$LOG_FILE"
}
# Check if running as root (skip in CI environments)
if [[ $EUID -eq 0 ]] && [[ -z "$CI" ]]; then
error "This script should not be run as root (use CI=true to override)"
exit 1
fi
# Check if Docker is running
if ! docker info > /dev/null 2>&1; then
error "Docker is not running. Please start Docker and try again."
exit 1
fi
# Check if we're in the right directory
if [ ! -f "package.json" ] || [ ! -f "Dockerfile" ]; then
error "Please run this script from the project root directory"
exit 1
fi
log "🚀 Starting Gitea deployment for $PROJECT_NAME"
# Step 1: Code Quality Checks
log "📋 Step 1: Running code quality checks..."
# Run linting
log "🔍 Running ESLint..."
npm run lint || {
error "ESLint failed. Please fix the issues before deploying."
exit 1
}
# Run tests
log "🧪 Running tests..."
npm run test:production || {
error "Tests failed. Please fix the issues before deploying."
exit 1
}
success "✅ Code quality checks passed"
# Step 2: Build Application
log "🔨 Step 2: Building application..."
# Build Next.js application
log "📦 Building Next.js application..."
npm run build || {
error "Build failed"
exit 1
}
success "✅ Application built successfully"
# Step 3: Docker Operations
log "🐳 Step 3: Docker operations..."
# Build Docker image
log "🏗️ Building Docker image..."
docker build -t "$IMAGE_NAME:latest" . || {
error "Docker build failed"
exit 1
}
# Tag with timestamp
TIMESTAMP=$(date +%Y%m%d-%H%M%S)
docker tag "$IMAGE_NAME:latest" "$IMAGE_NAME:$TIMESTAMP"
success "✅ Docker image built successfully"
# Step 4: Deployment
log "🚀 Step 4: Deploying application..."
# Export environment variables for docker-compose compatibility
log "📝 Exporting environment variables..."
export NODE_ENV=${NODE_ENV:-production}
export NEXT_PUBLIC_BASE_URL=${NEXT_PUBLIC_BASE_URL:-https://dk0.dev}
export MY_EMAIL=${MY_EMAIL:-contact@dk0.dev}
export MY_INFO_EMAIL=${MY_INFO_EMAIL:-info@dk0.dev}
export MY_PASSWORD="${MY_PASSWORD}"
export MY_INFO_PASSWORD="${MY_INFO_PASSWORD}"
export ADMIN_BASIC_AUTH="${ADMIN_BASIC_AUTH}"
export LOG_LEVEL=${LOG_LEVEL:-info}
export PORT=${PORT:-3000}
# Log which variables are set (without revealing secrets)
log "Environment variables configured:"
log " - NODE_ENV: ${NODE_ENV}"
log " - NEXT_PUBLIC_BASE_URL: ${NEXT_PUBLIC_BASE_URL}"
log " - MY_EMAIL: ${MY_EMAIL}"
log " - MY_INFO_EMAIL: ${MY_INFO_EMAIL}"
log " - MY_PASSWORD: [SET]"
log " - MY_INFO_PASSWORD: [SET]"
log " - ADMIN_BASIC_AUTH: [SET]"
log " - LOG_LEVEL: ${LOG_LEVEL}"
log " - PORT: ${PORT}"
# Check if container is running
if [ "$(docker inspect -f '{{.State.Running}}' "$CONTAINER_NAME" 2>/dev/null)" = "true" ]; then
log "📦 Stopping existing container..."
docker stop "$CONTAINER_NAME" || true
docker rm "$CONTAINER_NAME" || true
fi
# Check if port is available
if lsof -Pi :$PORT -sTCP:LISTEN -t >/dev/null ; then
warning "Port $PORT is in use. Trying backup port $BACKUP_PORT"
DEPLOY_PORT=$BACKUP_PORT
else
DEPLOY_PORT=$PORT
fi
# Start new container with environment variables
log "🚀 Starting new container on port $DEPLOY_PORT..."
docker run -d \
--name "$CONTAINER_NAME" \
--restart unless-stopped \
-p "$DEPLOY_PORT:3000" \
-e NODE_ENV=production \
-e NEXT_PUBLIC_BASE_URL=https://dk0.dev \
-e MY_EMAIL=contact@dk0.dev \
-e MY_INFO_EMAIL=info@dk0.dev \
-e MY_PASSWORD="${MY_PASSWORD:-your-email-password}" \
-e MY_INFO_PASSWORD="${MY_INFO_PASSWORD:-your-info-email-password}" \
-e ADMIN_BASIC_AUTH="${ADMIN_BASIC_AUTH:-admin:your_secure_password_here}" \
-e LOG_LEVEL=info \
"$IMAGE_NAME:latest" || {
error "Failed to start container"
exit 1
}
# Wait for container to be ready
log "⏳ Waiting for container to be ready..."
sleep 15
# Check if container is actually running
if [ "$(docker inspect -f '{{.State.Running}}' "$CONTAINER_NAME" 2>/dev/null)" != "true" ]; then
error "Container failed to start or crashed"
log "Container logs:"
docker logs "$CONTAINER_NAME" --tail=50
exit 1
fi
# Health check
log "🏥 Performing health check..."
HEALTH_CHECK_TIMEOUT=120
HEALTH_CHECK_INTERVAL=3
ELAPSED=0
while [ $ELAPSED -lt $HEALTH_CHECK_TIMEOUT ]; do
# Check if container is still running
if [ "$(docker inspect -f '{{.State.Running}}' "$CONTAINER_NAME" 2>/dev/null)" != "true" ]; then
error "Container stopped during health check"
log "Container logs:"
docker logs "$CONTAINER_NAME" --tail=50
exit 1
fi
# Try health check endpoint
if curl -f "http://localhost:$DEPLOY_PORT/api/health" > /dev/null 2>&1; then
success "✅ Application is healthy!"
break
fi
sleep $HEALTH_CHECK_INTERVAL
ELAPSED=$((ELAPSED + HEALTH_CHECK_INTERVAL))
echo -n "."
done
if [ $ELAPSED -ge $HEALTH_CHECK_TIMEOUT ]; then
error "Health check timeout. Application may not be running properly."
log "Container status:"
docker inspect "$CONTAINER_NAME" --format='{{.State.Status}} - {{.State.Health.Status}}'
log "Container logs:"
docker logs "$CONTAINER_NAME" --tail=100
exit 1
fi
# Step 5: Verification
log "✅ Step 5: Verifying deployment..."
# Test main page
if curl -f "http://localhost:$DEPLOY_PORT/" > /dev/null 2>&1; then
success "✅ Main page is accessible"
else
error "❌ Main page is not accessible"
exit 1
fi
# Show container status
log "📊 Container status:"
docker ps --filter "name=$CONTAINER_NAME" --format "table {{.Names}}\t{{.Status}}\t{{.Ports}}"
# Show resource usage
log "📈 Resource usage:"
docker stats --no-stream --format "table {{.Container}}\t{{.CPUPerc}}\t{{.MemUsage}}" "$CONTAINER_NAME"
# Step 6: Cleanup
log "🧹 Step 6: Cleaning up old images..."
# Remove old images (keep last 3 versions)
docker images "$IMAGE_NAME" --format "table {{.Tag}}\t{{.ID}}" | tail -n +2 | head -n -3 | awk '{print $2}' | xargs -r docker rmi || {
warning "No old images to remove"
}
# Clean up unused Docker resources
docker system prune -f --volumes || {
warning "Failed to clean up Docker resources"
}
# Final success message
success "🎉 Gitea deployment completed successfully!"
log "🌐 Application is available at: http://localhost:$DEPLOY_PORT"
log "🏥 Health check endpoint: http://localhost:$DEPLOY_PORT/api/health"
log "📊 Container name: $CONTAINER_NAME"
log "📝 Logs: docker logs $CONTAINER_NAME"
# Update deployment log
echo "$(date): Gitea deployment successful - Port: $DEPLOY_PORT - Image: $IMAGE_NAME:$TIMESTAMP" >> "$LOG_FILE"
exit 0

View File

@@ -1,192 +0,0 @@
#!/bin/bash
# Gitea Runner Setup Script
# Installiert und konfiguriert einen lokalen Gitea Runner
set -e
# Configuration
GITEA_URL="${GITEA_URL:-http://localhost:3000}"
RUNNER_NAME="${RUNNER_NAME:-portfolio-runner}"
RUNNER_LABELS="${RUNNER_LABELS:-ubuntu-latest,self-hosted,portfolio}"
RUNNER_WORK_DIR="${RUNNER_WORK_DIR:-/tmp/gitea-runner}"
# Colors for output
RED='\033[0;31m'
GREEN='\033[0;32m'
YELLOW='\033[1;33m'
BLUE='\033[0;34m'
NC='\033[0m' # No Color
# Logging function
log() {
echo -e "${BLUE}[$(date +'%Y-%m-%d %H:%M:%S')]${NC} $1"
}
error() {
echo -e "${RED}[ERROR]${NC} $1"
}
success() {
echo -e "${GREEN}[SUCCESS]${NC} $1"
}
warning() {
echo -e "${YELLOW}[WARNING]${NC} $1"
}
# Check if running as root (skip in CI environments)
if [[ $EUID -eq 0 ]] && [[ -z "$CI" ]]; then
error "This script should not be run as root (use CI=true to override)"
exit 1
fi
log "🚀 Setting up Gitea Runner for Portfolio"
# Check if Gitea URL is accessible
log "🔍 Checking Gitea server accessibility..."
if ! curl -f "$GITEA_URL" > /dev/null 2>&1; then
error "Cannot access Gitea server at $GITEA_URL"
error "Please make sure Gitea is running and accessible"
exit 1
fi
success "✅ Gitea server is accessible"
# Create runner directory
log "📁 Creating runner directory..."
mkdir -p "$RUNNER_WORK_DIR"
cd "$RUNNER_WORK_DIR"
# Download Gitea Runner
log "📥 Downloading Gitea Runner..."
RUNNER_VERSION="latest"
RUNNER_ARCH="linux-amd64"
# Get latest version
if [ "$RUNNER_VERSION" = "latest" ]; then
RUNNER_VERSION=$(curl -s https://api.github.com/repos/woodpecker-ci/woodpecker/releases/latest | grep -o '"tag_name": "[^"]*' | grep -o '[^"]*$')
fi
RUNNER_URL="https://github.com/woodpecker-ci/woodpecker/releases/download/${RUNNER_VERSION}/woodpecker-agent_${RUNNER_VERSION}_${RUNNER_ARCH}.tar.gz"
log "Downloading from: $RUNNER_URL"
curl -L -o woodpecker-agent.tar.gz "$RUNNER_URL"
# Extract runner
log "📦 Extracting Gitea Runner..."
tar -xzf woodpecker-agent.tar.gz
chmod +x woodpecker-agent
success "✅ Gitea Runner downloaded and extracted"
# Create systemd service
log "⚙️ Creating systemd service..."
sudo tee /etc/systemd/system/gitea-runner.service > /dev/null <<EOF
[Unit]
Description=Gitea Runner for Portfolio
After=network.target
[Service]
Type=simple
User=$USER
WorkingDirectory=$RUNNER_WORK_DIR
ExecStart=$RUNNER_WORK_DIR/woodpecker-agent
Restart=always
RestartSec=5
Environment=WOODPECKER_SERVER=$GITEA_URL
Environment=WOODPECKER_AGENT_SECRET=
Environment=WOODPECKER_LOG_LEVEL=info
[Install]
WantedBy=multi-user.target
EOF
# Reload systemd
sudo systemctl daemon-reload
success "✅ Systemd service created"
# Instructions for manual registration
log "📋 Manual registration required:"
echo ""
echo "1. Go to your Gitea instance: $GITEA_URL"
echo "2. Navigate to: Settings → Actions → Runners"
echo "3. Click 'Create new Runner'"
echo "4. Copy the registration token"
echo "5. Run the following command:"
echo ""
echo " cd $RUNNER_WORK_DIR"
echo " ./woodpecker-agent register --server $GITEA_URL --token YOUR_TOKEN"
echo ""
echo "6. After registration, start the service:"
echo " sudo systemctl enable gitea-runner"
echo " sudo systemctl start gitea-runner"
echo ""
echo "7. Check status:"
echo " sudo systemctl status gitea-runner"
echo ""
# Create helper scripts
log "📝 Creating helper scripts..."
# Start script
cat > "$RUNNER_WORK_DIR/start-runner.sh" << 'EOF'
#!/bin/bash
echo "Starting Gitea Runner..."
sudo systemctl start gitea-runner
sudo systemctl status gitea-runner
EOF
# Stop script
cat > "$RUNNER_WORK_DIR/stop-runner.sh" << 'EOF'
#!/bin/bash
echo "Stopping Gitea Runner..."
sudo systemctl stop gitea-runner
EOF
# Status script
cat > "$RUNNER_WORK_DIR/status-runner.sh" << 'EOF'
#!/bin/bash
echo "Gitea Runner Status:"
sudo systemctl status gitea-runner
echo ""
echo "Logs (last 20 lines):"
sudo journalctl -u gitea-runner -n 20 --no-pager
EOF
# Logs script
cat > "$RUNNER_WORK_DIR/logs-runner.sh" << 'EOF'
#!/bin/bash
echo "Gitea Runner Logs:"
sudo journalctl -u gitea-runner -f
EOF
chmod +x "$RUNNER_WORK_DIR"/*.sh
success "✅ Helper scripts created"
# Create environment file
cat > "$RUNNER_WORK_DIR/.env" << EOF
# Gitea Runner Configuration
GITEA_URL=$GITEA_URL
RUNNER_NAME=$RUNNER_NAME
RUNNER_LABELS=$RUNNER_LABELS
RUNNER_WORK_DIR=$RUNNER_WORK_DIR
EOF
log "📋 Setup Summary:"
echo " • Runner Directory: $RUNNER_WORK_DIR"
echo " • Gitea URL: $GITEA_URL"
echo " • Runner Name: $RUNNER_NAME"
echo " • Labels: $RUNNER_LABELS"
echo " • Helper Scripts: $RUNNER_WORK_DIR/*.sh"
echo ""
log "🎯 Next Steps:"
echo "1. Register the runner in Gitea web interface"
echo "2. Enable and start the service"
echo "3. Test with a workflow run"
echo ""
success "🎉 Gitea Runner setup completed!"
log "📁 All files are in: $RUNNER_WORK_DIR"

View File

@@ -1,79 +0,0 @@
/* eslint-disable @typescript-eslint/no-require-imports */
const fetch = require('node-fetch');
require('dotenv').config();
const DIRECTUS_URL = process.env.DIRECTUS_URL || 'https://cms.dk0.dev';
const DIRECTUS_TOKEN = process.env.DIRECTUS_STATIC_TOKEN;
async function setupSnippets() {
console.log('📦 Setting up Snippets collection...');
// 1. Create Collection
try {
await fetch(`${DIRECTUS_URL}/collections`, {
method: 'POST',
headers: { 'Authorization': `Bearer ${DIRECTUS_TOKEN}`, 'Content-Type': 'application/json' },
body: JSON.stringify({
collection: 'snippets',
meta: { icon: 'terminal', display_template: '{{title}}' },
schema: { name: 'snippets' }
})
});
} catch (_e) {}
// 2. Add Fields
const fields = [
{ field: 'status', type: 'string', meta: { interface: 'select-dropdown' }, schema: { default_value: 'published' } },
{ field: 'title', type: 'string', meta: { interface: 'input' } },
{ field: 'category', type: 'string', meta: { interface: 'input' } },
{ field: 'code', type: 'text', meta: { interface: 'input-code' } },
{ field: 'description', type: 'text', meta: { interface: 'textarea' } },
{ field: 'language', type: 'string', meta: { interface: 'input' }, schema: { default_value: 'javascript' } },
{ field: 'featured', type: 'boolean', meta: { interface: 'boolean' }, schema: { default_value: false } }
];
for (const f of fields) {
try {
await fetch(`${DIRECTUS_URL}/fields/snippets`, {
method: 'POST',
headers: { 'Authorization': `Bearer ${DIRECTUS_TOKEN}`, 'Content-Type': 'application/json' },
body: JSON.stringify(f)
});
} catch (_e) {}
}
// 3. Add Example Data
const exampleSnippets = [
{
title: 'Traefik SSL Config',
category: 'Docker',
language: 'yaml',
featured: true,
description: "Meine Standard-Konfiguration für automatisches SSL via Let's Encrypt in Docker Swarm.",
code: "labels:\n - \"traefik.enable=true\"\n - \"traefik.http.routers.myapp.rule=Host(`example.com`)\"\n - \"traefik.http.routers.myapp.entrypoints=websecure\"\n - \"traefik.http.routers.myapp.tls.certresolver=myresolver\""
},
{
title: 'Docker Cleanup Alias',
category: 'ZSH',
language: 'bash',
featured: true,
description: 'Ein einfacher Alias, um ungenutzte Docker-Container, Images und Volumes schnell zu entfernen.',
code: "alias dclean='docker system prune -af --volumes'"
}
];
for (const s of exampleSnippets) {
try {
await fetch(`${DIRECTUS_URL}/items/snippets`, {
method: 'POST',
headers: { 'Authorization': `Bearer ${DIRECTUS_TOKEN}`, 'Content-Type': 'application/json' },
body: JSON.stringify(s)
});
} catch (_e) {}
}
console.log('✅ Snippets setup complete!');
}
setupSnippets();

View File

@@ -1,35 +0,0 @@
# Test 1: Eigenes Projekt (sollte hohen Coolness Score bekommen)
curl -X POST https://n8n.dk0.dev/webhook/docker-event `
-H "Content-Type: application/json" `
-d '{
"container": "portfolio-dev",
"image": "denshooter/portfolio:latest",
"timestamp": "2026-04-01T23:18:00Z"
}'
# Test 2: Bekanntes Self-Hosted Tool (mittlerer Score)
curl -X POST https://n8n.dk0.dev/webhook/docker-event `
-H "Content-Type: application/json" `
-d '{
"container": "plausible-analytics",
"image": "plausible/analytics:latest",
"timestamp": "2026-04-01T23:18:00Z"
}'
# Test 3: CI/CD Runner (sollte ignoriert werden)
curl -X POST https://n8n.dk0.dev/webhook/docker-event `
-H "Content-Type: application/json" `
-d '{
"container": "gitea-actions-task-351-workflow-ci-cd-job-test-build",
"image": "catthehacker/ubuntu:act-latest",
"timestamp": "2026-04-01T23:18:00Z"
}'
# Test 4: Spannendes Sicherheitstool (hoher Score)
curl -X POST https://n8n.dk0.dev/webhook/docker-event `
-H "Content-Type: application/json" `
-d '{
"container": "suricata-ids",
"image": "jasonish/suricata:latest",
"timestamp": "2026-04-01T23:18:00Z"
}'