Protokoll #2173
| ID | 2173 |
|---|---|
| Zeitstempel | 2025-12-21 10:40:12.615666 |
| Client | root |
| IP | Warning: Undefined array key "request_ip" in /var/www/dev.campus.systemische-tools.de/src/View/protokoll/show.php on line 13 Deprecated: htmlspecialchars(): Passing null to parameter #1 ($string) of type string is deprecated in /var/www/dev.campus.systemische-tools.de/src/View/protokoll/show.php on line 13 |
| Modell | claude-sonnet-4-20250514 |
| Status | completed |
| Tokens | 8,606 (Input: 0, Output: 0) |
| Dauer | 97 ms |
| Request-Zeit | Warning: Undefined array key "request_timestamp" in /var/www/dev.campus.systemische-tools.de/src/View/protokoll/show.php on line 18 |
| Response-Zeit | - |
Warning: Undefined array key "error_message" in /var/www/dev.campus.systemische-tools.de/src/View/protokoll/show.php on line 22
Request
{
"event": "PreToolUse",
"tool_name": "Edit",
"tool_input": {
"file_path": "\/var\/www\/dev.campus.systemische-tools.de\/docs\/plan-form-components-refactoring.md",
"old_string": "## 0j. Structure (Content Studio only)\n\n### 0j.1 Architektur-Entscheidung\n\n**Gewählt: Bestehende Tabelle `content_config` nutzen**\n\n- Structures sind in `content_config` mit `type = 'structure'`\n- Nur im Content Studio verwendet\n- Optional: kein Auto-Apply\n\n### 0j.2 Partial: structure.php\n\n```php\n\/\/ \/src\/View\/partials\/form\/structure.php\n<?php\n\/**\n * Benötigte Variablen (via Controller):\n * - $structures: array aus ContentConfigRepository::findByType('structure')\n * - $userPrefs: array aus UserPreferencesService\n *\n * Optionale Variablen:\n * - $selected: int (Order-Wert)\n * - $variant: 'default' | 'inline'\n * - $name: string (default: 'structure_id')\n *\/\n\n$structures = $structures ?? [];\n$selected = $selected \/\/ 1. Order-Wert\n ?? ($userPrefs['default_structure_id'] ?? null) \/\/ 2. User-Präferenz\n ?? null; \/\/ 3. Keine Struktur\n\n$name = $name ?? 'structure_id';\n$id = $id ?? 'structure_id';\n$variant = $variant ?? 'default';\n$class = $variant === 'inline' ? 'form-select--inline' : 'form-select';\n?>\n<select name=\"<?= $name ?>\" id=\"<?= $id ?>\" class=\"<?= $class ?>\">\n <option value=\"\">-- Keine Struktur --<\/option>\n <?php foreach ($structures as $structure): ?>\n <option value=\"<?= $structure['id'] ?>\" <?= (int)$selected === (int)$structure['id'] ? 'selected' : '' ?>>\n <?= htmlspecialchars($structure['name']) ?>\n <\/option>\n <?php endforeach; ?>\n<\/select>\n```\n\n### 0j.3 Werte-Hierarchie\n\n```\n1. Order-Wert → $order['structure_id']\n ↓ falls null\n2. User-Präferenz → $userPrefs['default_structure_id']\n ↓ falls null\n3. System-Default → null (keine Struktur)\n```\n\n### 0j.4 Verwendung\n\n| View | Kontext | Variante |\n|------|---------|----------|\n| Chat | - (nicht verwendet) | - |\n| Content New | Formular | default |\n| Content Edit | Formular | default |\n\n### 0j.5 Migrations-Schritte (Structure)\n\n1. [ ] Partial `structure.php` erstellen\n2. [ ] Content New auf Partial umstellen\n3. [ ] Content Edit auf Partial umstellen\n4. [ ] Testen: Default aus User-Präferenzen",
"new_string": "## 0j. Structures (Chat + Content Studio, Multi-Select)\n\n### 0j.1 Architektur-Entscheidung\n\n**Gewählt: Multi-Select in beiden Systemen**\n\n- Structures sind in `content_config` mit `type = 'structure'`\n- **Verwendung in BEIDEN Systemen:** Chat + Content Studio\n- **Multi-Select:** Mehrere Strukturen können kombiniert werden\n- Speicherung als JSON-Array (wie Collections und Contracts)\n- Optional: kein Auto-Apply (leere Auswahl erlaubt)\n\n### 0j.2 Use Cases\n\n| System | Verwendungszweck |\n|--------|------------------|\n| **Chat** | Ausgabeformat für Antworten (z.B. \"Markdown\", \"Bullet Points\", \"Tabelle\") |\n| **Content Studio** | Inhaltsstruktur für generierten Content (z.B. \"Blog-Artikel\", \"Whitepaper\") |\n\n### 0j.3 Datenbank-Änderungen\n\n**chat_sessions (ki_content):**\n```sql\nALTER TABLE chat_sessions\nADD COLUMN structures JSON DEFAULT '[]' AFTER contracts;\n```\n\n**content_orders (ki_content):**\n```sql\n-- Von INT zu JSON ändern\nALTER TABLE content_orders\nCHANGE COLUMN structure_id structures JSON DEFAULT '[]';\n```\n\n**user_preferences (ki_dev):**\n```sql\n-- Von INT zu JSON ändern\nALTER TABLE user_preferences\nCHANGE COLUMN default_structure_id default_structures JSON DEFAULT '[]';\n```\n\n### 0j.4 Partial: structures-select.php\n\n```php\n\/\/ \/src\/View\/partials\/form\/structures-select.php\n<?php\n\/**\n * Benötigte Variablen (via Controller):\n * - $structures: array aus ContentConfigRepository::findByType('structure')\n * - $userPrefs: array aus UserPreferencesService\n *\n * Optionale Variablen:\n * - $selected: array|string (Session\/Order-Wert als JSON oder Array)\n * - $variant: 'default' | 'inline'\n * - $name: string (default: 'structures[]')\n *\/\n\n$structures = $structures ?? [];\n\n\/\/ Selected normalisieren (JSON → Array)\n$selected = $selected ?? null;\nif (is_string($selected)) {\n $selected = json_decode($selected, true) ?: [];\n}\n\n\/\/ Werte-Hierarchie\nif ($selected === null || $selected === []) {\n $userDefault = $userPrefs['default_structures'] ?? '[]';\n if (is_string($userDefault)) {\n $selected = json_decode($userDefault, true) ?: [];\n } else {\n $selected = $userDefault ?: [];\n }\n}\n\n$name = $name ?? 'structures[]';\n$id = $id ?? 'structures';\n$variant = $variant ?? 'default';\n$class = $variant === 'inline' ? 'form-select--inline' : 'form-select';\n?>\n<select name=\"<?= $name ?>\" id=\"<?= $id ?>\" class=\"<?= $class ?> structures-select\" multiple\n title=\"Strukturvorlagen auswählen\">\n <?php foreach ($structures as $structure): ?>\n <option value=\"<?= $structure['id'] ?>\"\n <?= in_array($structure['id'], $selected, false) ? 'selected' : '' ?>\n title=\"<?= htmlspecialchars($structure['description'] ?? '') ?>\">\n <?= htmlspecialchars($structure['name']) ?>\n <\/option>\n <?php endforeach; ?>\n<\/select>\n```\n\n### 0j.5 Werte-Hierarchie\n\n```\n1. Session\/Order-Wert → $session['structures'] oder $order['structures'] (JSON)\n ↓ falls leer\n2. User-Präferenz → $userPrefs['default_structures'] (JSON)\n ↓ falls leer\n3. System-Default → [] (leer, keine Struktur erforderlich)\n```\n\n### 0j.6 Verwendung\n\n| View | Kontext | Variante |\n|------|---------|----------|\n| Chat | Session-Einstellungen | default |\n| Content New | Formular | default |\n| Content Edit | Formular | default |\n| Content Show | Inline-Aktionen | inline |\n\n### 0j.7 Controller-Anpassungen\n\n**ChatController:**\n```php\n\/\/ createSession()\n$structures = json_decode($prefs['default_structures'] ?? '[]', true);\n\n$stmt->execute([\n \/\/ ...\n 'structures' => json_encode($structures),\n]);\n\n\/\/ show()\n$this->view('chat.index', [\n \/\/ ...\n 'structures' => $this->contentConfigRepository->findByType('structure'),\n]);\n```\n\n**ContentController:**\n```php\n\/\/ store()\n$structures = $_POST['structures'] ?? [];\nif (!is_array($structures)) {\n $structures = [$structures];\n}\n\n\/\/ In DB speichern\n'structures' => json_encode($structures),\n```\n\n### 0j.8 Python-Pipeline Anpassung\n\n```python\ndef get_order(order_id):\n # Structures jetzt als JSON-Array\n structures_json = row['structures'] or '[]'\n structures_ids = json.loads(structures_json)\n\n # Alle Structures laden\n structures = []\n for sid in structures_ids:\n structure = get_config_item(sid, 'structure')\n if structure:\n structures.append(structure)\n\n return {\n # ...\n 'structures': structures, # Liste statt einzelner Wert\n }\n```\n\n### 0j.9 Migrations-Schritte (Structures)\n\n#### Phase 1: DB-Schema\n1. [ ] `chat_sessions`: Spalte `structures JSON` hinzufügen\n2. [ ] `content_orders`: Spalte `structure_id` → `structures JSON` migrieren\n3. [ ] `user_preferences`: Spalte `default_structure_id` → `default_structures JSON` migrieren\n\n#### Phase 2: Partial\n4. [ ] Partial `structures-select.php` erstellen (Multi-Select)\n\n#### Phase 3: Chat-Integration\n5. [ ] `ChatController::createSession()` anpassen\n6. [ ] `ChatController::show()` Structures laden\n7. [ ] `chat\/index.php` Partial einbinden\n\n#### Phase 4: Content-Integration\n8. [ ] `ContentController::store()` anpassen (JSON)\n9. [ ] `ContentController::contentNew()` Structures laden\n10. [ ] `content\/new.php` auf Partial umstellen\n11. [ ] `content\/show.php` auf Partial umstellen\n\n#### Phase 5: Pipeline\n12. [ ] `generate.py` auf JSON-Array umstellen\n\n#### Phase 6: Testing\n13. [ ] Chat: Multi-Select funktioniert\n14. [ ] Content: Multi-Select funktioniert\n15. [ ] Default aus User-Präferenzen\n16. [ ] Pipeline verarbeitet mehrere Structures"
}
}
Response
{
"tool_response": {
"filePath": "\/var\/www\/dev.campus.systemische-tools.de\/docs\/plan-form-components-refactoring.md",
"oldString": "## 0j. Structure (Content Studio only)\n\n### 0j.1 Architektur-Entscheidung\n\n**Gewählt: Bestehende Tabelle `content_config` nutzen**\n\n- Structures sind in `content_config` mit `type = 'structure'`\n- Nur im Content Studio verwendet\n- Optional: kein Auto-Apply\n\n### 0j.2 Partial: structure.php\n\n```php\n\/\/ \/src\/View\/partials\/form\/structure.php\n<?php\n\/**\n * Benötigte Variablen (via Controller):\n * - $structures: array aus ContentConfigRepository::findByType('structure')\n * - $userPrefs: array aus UserPreferencesService\n *\n * Optionale Variablen:\n * - $selected: int (Order-Wert)\n * - $variant: 'default' | 'inline'\n * - $name: string (default: 'structure_id')\n *\/\n\n$structures = $structures ?? [];\n$selected = $selected \/\/ 1. Order-Wert\n ?? ($userPrefs['default_structure_id'] ?? null) \/\/ 2. User-Präferenz\n ?? null; \/\/ 3. Keine Struktur\n\n$name = $name ?? 'structure_id';\n$id = $id ?? 'structure_id';\n$variant = $variant ?? 'default';\n$class = $variant === 'inline' ? 'form-select--inline' : 'form-select';\n?>\n<select name=\"<?= $name ?>\" id=\"<?= $id ?>\" class=\"<?= $class ?>\">\n <option value=\"\">-- Keine Struktur --<\/option>\n <?php foreach ($structures as $structure): ?>\n <option value=\"<?= $structure['id'] ?>\" <?= (int)$selected === (int)$structure['id'] ? 'selected' : '' ?>>\n <?= htmlspecialchars($structure['name']) ?>\n <\/option>\n <?php endforeach; ?>\n<\/select>\n```\n\n### 0j.3 Werte-Hierarchie\n\n```\n1. Order-Wert → $order['structure_id']\n ↓ falls null\n2. User-Präferenz → $userPrefs['default_structure_id']\n ↓ falls null\n3. System-Default → null (keine Struktur)\n```\n\n### 0j.4 Verwendung\n\n| View | Kontext | Variante |\n|------|---------|----------|\n| Chat | - (nicht verwendet) | - |\n| Content New | Formular | default |\n| Content Edit | Formular | default |\n\n### 0j.5 Migrations-Schritte (Structure)\n\n1. [ ] Partial `structure.php` erstellen\n2. [ ] Content New auf Partial umstellen\n3. [ ] Content Edit auf Partial umstellen\n4. [ ] Testen: Default aus User-Präferenzen",
"newString": "## 0j. Structures (Chat + Content Studio, Multi-Select)\n\n### 0j.1 Architektur-Entscheidung\n\n**Gewählt: Multi-Select in beiden Systemen**\n\n- Structures sind in `content_config` mit `type = 'structure'`\n- **Verwendung in BEIDEN Systemen:** Chat + Content Studio\n- **Multi-Select:** Mehrere Strukturen können kombiniert werden\n- Speicherung als JSON-Array (wie Collections und Contracts)\n- Optional: kein Auto-Apply (leere Auswahl erlaubt)\n\n### 0j.2 Use Cases\n\n| System | Verwendungszweck |\n|--------|------------------|\n| **Chat** | Ausgabeformat für Antworten (z.B. \"Markdown\", \"Bullet Points\", \"Tabelle\") |\n| **Content Studio** | Inhaltsstruktur für generierten Content (z.B. \"Blog-Artikel\", \"Whitepaper\") |\n\n### 0j.3 Datenbank-Änderungen\n\n**chat_sessions (ki_content):**\n```sql\nALTER TABLE chat_sessions\nADD COLUMN structures JSON DEFAULT '[]' AFTER contracts;\n```\n\n**content_orders (ki_content):**\n```sql\n-- Von INT zu JSON ändern\nALTER TABLE content_orders\nCHANGE COLUMN structure_id structures JSON DEFAULT '[]';\n```\n\n**user_preferences (ki_dev):**\n```sql\n-- Von INT zu JSON ändern\nALTER TABLE user_preferences\nCHANGE COLUMN default_structure_id default_structures JSON DEFAULT '[]';\n```\n\n### 0j.4 Partial: structures-select.php\n\n```php\n\/\/ \/src\/View\/partials\/form\/structures-select.php\n<?php\n\/**\n * Benötigte Variablen (via Controller):\n * - $structures: array aus ContentConfigRepository::findByType('structure')\n * - $userPrefs: array aus UserPreferencesService\n *\n * Optionale Variablen:\n * - $selected: array|string (Session\/Order-Wert als JSON oder Array)\n * - $variant: 'default' | 'inline'\n * - $name: string (default: 'structures[]')\n *\/\n\n$structures = $structures ?? [];\n\n\/\/ Selected normalisieren (JSON → Array)\n$selected = $selected ?? null;\nif (is_string($selected)) {\n $selected = json_decode($selected, true) ?: [];\n}\n\n\/\/ Werte-Hierarchie\nif ($selected === null || $selected === []) {\n $userDefault = $userPrefs['default_structures'] ?? '[]';\n if (is_string($userDefault)) {\n $selected = json_decode($userDefault, true) ?: [];\n } else {\n $selected = $userDefault ?: [];\n }\n}\n\n$name = $name ?? 'structures[]';\n$id = $id ?? 'structures';\n$variant = $variant ?? 'default';\n$class = $variant === 'inline' ? 'form-select--inline' : 'form-select';\n?>\n<select name=\"<?= $name ?>\" id=\"<?= $id ?>\" class=\"<?= $class ?> structures-select\" multiple\n title=\"Strukturvorlagen auswählen\">\n <?php foreach ($structures as $structure): ?>\n <option value=\"<?= $structure['id'] ?>\"\n <?= in_array($structure['id'], $selected, false) ? 'selected' : '' ?>\n title=\"<?= htmlspecialchars($structure['description'] ?? '') ?>\">\n <?= htmlspecialchars($structure['name']) ?>\n <\/option>\n <?php endforeach; ?>\n<\/select>\n```\n\n### 0j.5 Werte-Hierarchie\n\n```\n1. Session\/Order-Wert → $session['structures'] oder $order['structures'] (JSON)\n ↓ falls leer\n2. User-Präferenz → $userPrefs['default_structures'] (JSON)\n ↓ falls leer\n3. System-Default → [] (leer, keine Struktur erforderlich)\n```\n\n### 0j.6 Verwendung\n\n| View | Kontext | Variante |\n|------|---------|----------|\n| Chat | Session-Einstellungen | default |\n| Content New | Formular | default |\n| Content Edit | Formular | default |\n| Content Show | Inline-Aktionen | inline |\n\n### 0j.7 Controller-Anpassungen\n\n**ChatController:**\n```php\n\/\/ createSession()\n$structures = json_decode($prefs['default_structures'] ?? '[]', true);\n\n$stmt->execute([\n \/\/ ...\n 'structures' => json_encode($structures),\n]);\n\n\/\/ show()\n$this->view('chat.index', [\n \/\/ ...\n 'structures' => $this->contentConfigRepository->findByType('structure'),\n]);\n```\n\n**ContentController:**\n```php\n\/\/ store()\n$structures = $_POST['structures'] ?? [];\nif (!is_array($structures)) {\n $structures = [$structures];\n}\n\n\/\/ In DB speichern\n'structures' => json_encode($structures),\n```\n\n### 0j.8 Python-Pipeline Anpassung\n\n```python\ndef get_order(order_id):\n # Structures jetzt als JSON-Array\n structures_json = row['structures'] or '[]'\n structures_ids = json.loads(structures_json)\n\n # Alle Structures laden\n structures = []\n for sid in structures_ids:\n structure = get_config_item(sid, 'structure')\n if structure:\n structures.append(structure)\n\n return {\n # ...\n 'structures': structures, # Liste statt einzelner Wert\n }\n```\n\n### 0j.9 Migrations-Schritte (Structures)\n\n#### Phase 1: DB-Schema\n1. [ ] `chat_sessions`: Spalte `structures JSON` hinzufügen\n2. [ ] `content_orders`: Spalte `structure_id` → `structures JSON` migrieren\n3. [ ] `user_preferences`: Spalte `default_structure_id` → `default_structures JSON` migrieren\n\n#### Phase 2: Partial\n4. [ ] Partial `structures-select.php` erstellen (Multi-Select)\n\n#### Phase 3: Chat-Integration\n5. [ ] `ChatController::createSession()` anpassen\n6. [ ] `ChatController::show()` Structures laden\n7. [ ] `chat\/index.php` Partial einbinden\n\n#### Phase 4: Content-Integration\n8. [ ] `ContentController::store()` anpassen (JSON)\n9. [ ] `ContentController::contentNew()` Structures laden\n10. [ ] `content\/new.php` auf Partial umstellen\n11. [ ] `content\/show.php` auf Partial umstellen\n\n#### Phase 5: Pipeline\n12. [ ] `generate.py` auf JSON-Array umstellen\n\n#### Phase 6: Testing\n13. [ ] Chat: Multi-Select funktioniert\n14. [ ] Content: Multi-Select funktioniert\n15. [ ] Default aus User-Präferenzen\n16. [ ] Pipeline verarbeitet mehrere Structures",
"originalFile": "# Plan: Einheitliche Form-Komponenten (Chat & Content Studio)\n\n## Status: PLANUNG\n**Erstellt:** 2025-12-21\n**Ziel:** DRY, KISS, SRP - Shared Partials für alle wiederverwendbaren Form-Elemente\n\n---\n\n## 0. LLM-Verwaltung (Grundlage für Modell-Dropdown)\n\n### 0.1 Architektur-Entscheidung\n\n**Gewählt: Variante B - Dedizierte LLM-Verwaltung**\n\nStatt dynamischer API-Abfrage bei jedem Seitenaufruf:\n- Zentrale Admin-Seite `\/ressourcen\/llm`\n- Datenbank-Tabelle als Single Source of Truth\n- Sync-Buttons für Provider (Anthropic, Ollama, weitere)\n- Sprechende Namen und Zusatzmetadaten\n\n### 0.2 Datenbank-Tabelle `llm_models`\n\n```sql\nCREATE TABLE llm_models (\n id INT AUTO_INCREMENT PRIMARY KEY,\n provider ENUM('anthropic', 'ollama', 'openai', 'google', 'mistral', 'custom') NOT NULL,\n model_id VARCHAR(100) NOT NULL, -- API-ID: \"claude-opus-4-5-20251101\"\n display_name VARCHAR(100) NOT NULL, -- Anzeige: \"Claude Opus 4.5\"\n description TEXT, -- Kurzbeschreibung\n context_window INT, -- Max. Tokens Input: 200000\n max_output_tokens INT, -- Max. Tokens Output: 8192\n input_price_per_mtok DECIMAL(10,4), -- Preis Input $\/MTok\n output_price_per_mtok DECIMAL(10,4), -- Preis Output $\/MTok\n capabilities JSON, -- {\"vision\": true, \"function_calling\": true}\n is_local BOOLEAN DEFAULT FALSE, -- Lokal (Ollama) vs. Cloud\n is_active BOOLEAN DEFAULT TRUE, -- In Dropdowns anzeigen?\n sort_order INT DEFAULT 0, -- Reihenfolge in Dropdowns\n last_synced_at DATETIME, -- Letzte Synchronisation\n created_at DATETIME DEFAULT CURRENT_TIMESTAMP,\n updated_at DATETIME DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,\n UNIQUE KEY unique_provider_model (provider, model_id)\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;\n```\n\n### 0.3 Admin-Seite `\/ressourcen\/llm`\n\n**URL:** `\/ressourcen\/llm`\n**Controller:** `RessourcenController::llmIndex()`\n\n**Funktionen:**\n\n| Button | Aktion |\n|--------|--------|\n| \"Anthropic synchronisieren\" | `GET api.anthropic.com\/v1\/models` → DB aktualisieren |\n| \"Ollama synchronisieren\" | `ollama list` → DB aktualisieren |\n| \"Neuer Provider\" | Manuell weiteren Provider hinzufügen |\n\n**Tabellen-Ansicht:**\n\n| Provider | Model-ID | Display-Name | Context | Preis In\/Out | Lokal | Aktiv | Aktionen |\n|----------|----------|--------------|---------|--------------|-------|-------|----------|\n| Anthropic | claude-opus-4-5-20251101 | Claude Opus 4.5 | 200k | $15\/$75 | - | ✓ | Bearbeiten |\n| Ollama | mistral:latest | Mistral 7B | 32k | - | ✓ | ✓ | Bearbeiten |\n\n**Bearbeiten-Dialog:**\n- Display-Name ändern\n- Beschreibung hinzufügen\n- Context-Window \/ Max-Output korrigieren\n- Preise eintragen (für Kostenberechnung)\n- Aktivieren\/Deaktivieren\n- Sortierung ändern\n\n### 0.4 Sync-Logik\n\n**Anthropic Sync:**\n```php\n\/\/ GET https:\/\/api.anthropic.com\/v1\/models\n\/\/ Response: {\"data\": [{\"id\": \"claude-opus-4-5-20251101\", ...}]}\n\nforeach ($apiModels as $model) {\n \/\/ INSERT ... ON DUPLICATE KEY UPDATE\n \/\/ Neue Modelle: is_active = true, display_name = model_id (initial)\n \/\/ Existierende: last_synced_at aktualisieren\n \/\/ Fehlende: NICHT löschen, nur last_synced_at bleibt alt\n}\n```\n\n**Ollama Sync:**\n```php\n\/\/ $ ollama list\n\/\/ NAME ID SIZE MODIFIED\n\/\/ mistral:latest abc123... 4.1 GB 2 days ago\n\n$output = shell_exec('ollama list');\n\/\/ Parsen und in DB einfügen\n```\n\n### 0.5 Dropdown-Query\n\n```php\n\/\/ ModelService::getActiveModels()\npublic function getActiveModels(): array\n{\n return $this->db->query(\"\n SELECT model_id, display_name, provider, is_local, context_window\n FROM llm_models\n WHERE is_active = 1\n ORDER BY is_local ASC, sort_order ASC, display_name ASC\n \")->fetchAll();\n}\n```\n\n### 0.6 Partial nutzt DB-Daten\n\n```php\n\/\/ \/src\/View\/partials\/form\/model-select.php\n<?php\n$models = $models ?? [];\n$selected = $selected ?? '';\n$variant = $variant ?? 'default';\n$class = $variant === 'inline' ? 'form-select--inline' : 'form-select';\n?>\n<select name=\"model\" id=\"model\" class=\"<?= $class ?>\">\n <optgroup label=\"Cloud\">\n <?php foreach ($models as $m): ?>\n <?php if (!$m['is_local']): ?>\n <option value=\"<?= $m['model_id'] ?>\" <?= $selected === $m['model_id'] ? 'selected' : '' ?>>\n <?= htmlspecialchars($m['display_name']) ?>\n <\/option>\n <?php endif; ?>\n <?php endforeach; ?>\n <\/optgroup>\n <optgroup label=\"Lokal\">\n <?php foreach ($models as $m): ?>\n <?php if ($m['is_local']): ?>\n <option value=\"<?= $m['model_id'] ?>\" <?= $selected === $m['model_id'] ? 'selected' : '' ?>>\n <?= htmlspecialchars($m['display_name']) ?>\n <\/option>\n <?php endif; ?>\n <?php endforeach; ?>\n <\/optgroup>\n<\/select>\n```\n\n### 0.7 Migrations-Schritte (LLM-Verwaltung)\n\n1. [ ] Tabelle `llm_models` in `ki_dev` erstellen\n2. [ ] `LlmModelRepository` erstellen\n3. [ ] `LlmSyncService` erstellen (Anthropic + Ollama)\n4. [ ] Route `\/ressourcen\/llm` anlegen\n5. [ ] `RessourcenController::llmIndex()` implementieren\n6. [ ] View `\/ressourcen\/llm\/index.php` erstellen\n7. [ ] Sync-Buttons implementieren\n8. [ ] Bearbeiten-Funktionalität\n9. [ ] Initial-Sync durchführen (bestehende Modelle importieren)\n10. [ ] `ModelConfig.php` durch `LlmModelRepository` ersetzen\n11. [ ] Partial `model-select.php` erstellen\n12. [ ] Chat und Content auf Partial umstellen\n\n---\n\n## 0b. Collection-Verwaltung (Grundlage für Collection-Dropdown)\n\n### 0b.1 Architektur-Entscheidung\n\n**Gewählt: DB-Tabelle analog zu LLMs**\n\n- Zentrale Admin-Seite `\/ressourcen\/collections`\n- Datenbank-Tabelle als Single Source of Truth\n- Sync mit Qdrant (Metadaten abrufen)\n- Sprechende Namen und Beschreibungen\n- **Multi-Select** als einheitliche Darstellung\n\n### 0b.2 Datenbank-Tabelle `rag_collections`\n\n```sql\nCREATE TABLE rag_collections (\n id INT AUTO_INCREMENT PRIMARY KEY,\n collection_id VARCHAR(100) NOT NULL UNIQUE, -- Qdrant-Name: \"documents\"\n display_name VARCHAR(100) NOT NULL, -- Anzeige: \"Dokumente\"\n description TEXT, -- \"PDF-Dokumente aus Nextcloud\"\n\n -- Qdrant-Metadaten (via Sync)\n vector_size INT, -- 1024\n distance_metric VARCHAR(20), -- \"Cosine\"\n points_count INT DEFAULT 0, -- Anzahl Vektoren\n\n -- Konfiguration\n embedding_model VARCHAR(100), -- \"mxbai-embed-large\"\n chunk_size INT, -- 2000\n chunk_overlap INT, -- 200\n\n -- Verwaltung\n source_type ENUM('nextcloud', 'mail', 'manual', 'system') DEFAULT 'manual',\n source_path VARCHAR(500), -- \"\/var\/www\/nextcloud\/data\/...\"\n is_active BOOLEAN DEFAULT TRUE, -- In Dropdowns anzeigen?\n is_searchable BOOLEAN DEFAULT TRUE, -- Für RAG verfügbar?\n sort_order INT DEFAULT 0,\n\n -- Timestamps\n last_synced_at DATETIME,\n created_at DATETIME DEFAULT CURRENT_TIMESTAMP,\n updated_at DATETIME DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;\n```\n\n### 0b.3 Impact-Analyse: Wer nutzt Collections?\n\n| System\/Seite | Aktueller Zugriff | Neuer Zugriff | Änderung |\n|--------------|-------------------|---------------|----------|\n| **Chat** | `$qdrantService->listCollections()` | `CollectionRepository::getActive()` | Query aus DB |\n| **Content Studio** | `$qdrantService->listCollections()` | `CollectionRepository::getActive()` | Query aus DB |\n| **Content Show** | `$qdrantService->listCollections()` | `CollectionRepository::getActive()` | Query aus DB |\n| **Pipeline (Python)** | `config.py QDRANT_COLLECTIONS` | DB-Query oder Config-Sync | Python liest DB |\n| **Semantic Explorer** | Direkt Qdrant | `CollectionRepository` + Qdrant | Metadaten aus DB |\n| **System Explorer** | - | Statistiken aus DB | Neu |\n| **API \/api\/v1\/search** | Qdrant direkt | Validierung gegen DB | Nur aktive Collections |\n| **Embedding-Service** | `config.py` | DB oder Sync | Konsistenz |\n\n### 0b.4 Admin-Seite `\/ressourcen\/collections`\n\n**URL:** `\/ressourcen\/collections`\n**Controller:** `RessourcenController::collectionsIndex()`\n\n**Funktionen:**\n\n| Button | Aktion |\n|--------|--------|\n| \"Qdrant synchronisieren\" | Collections + Metadaten von Qdrant abrufen |\n| \"Neue Collection\" | Manuell Collection registrieren |\n\n**Tabellen-Ansicht:**\n\n| Collection-ID | Display-Name | Vektoren | Größe | Quelle | Aktiv | Suchbar | Aktionen |\n|---------------|--------------|----------|-------|--------|-------|---------|----------|\n| documents | Dokumente | 1.247 | 1024d | Nextcloud | ✓ | ✓ | Bearbeiten |\n| dokumentation_chunks | Doku-Chunks | 892 | 1024d | System | ✓ | ✓ | Bearbeiten |\n| mail | E-Mails | 3.421 | 1024d | Mail | ✓ | ✓ | Bearbeiten |\n| entities | Entitäten | 156 | 1024d | System | ✓ | - | Bearbeiten |\n\n**Bearbeiten-Dialog:**\n- Display-Name ändern\n- Beschreibung hinzufügen\n- Quelle\/Pfad dokumentieren\n- Aktivieren\/Deaktivieren (Dropdown)\n- Suchbar ja\/nein (RAG-Verfügbarkeit)\n- Sortierung ändern\n\n### 0b.5 Sync-Logik\n\n**Qdrant Sync:**\n```php\n\/\/ QdrantService::getCollectionsInfo()\npublic function syncCollections(): array\n{\n $qdrantCollections = $this->client->getCollections();\n\n foreach ($qdrantCollections as $col) {\n $info = $this->client->getCollectionInfo($col['name']);\n\n \/\/ INSERT ... ON DUPLICATE KEY UPDATE\n $this->repository->upsert([\n 'collection_id' => $col['name'],\n 'display_name' => $col['name'], \/\/ Initial = ID\n 'vector_size' => $info['config']['params']['size'],\n 'distance_metric' => $info['config']['params']['distance'],\n 'points_count' => $info['points_count'],\n 'last_synced_at' => now(),\n ]);\n }\n}\n```\n\n### 0b.6 Repository-Interface\n\n```php\n\/\/ CollectionRepository.php\ninterface CollectionRepositoryInterfac... [TRUNCATED-fa3622ac56864344]",
"structuredPatch": [
{
"oldStart": 1622,
"oldLines": 20,
"newStart": 1622,
"newLines": 51,
"lines": [
" ",
" ---",
" ",
"-## 0j. Structure (Content Studio only)",
"+## 0j. Structures (Chat + Content Studio, Multi-Select)",
" ",
" ### 0j.1 Architektur-Entscheidung",
" ",
"-**Gewählt: Bestehende Tabelle `content_config` nutzen**",
"+**Gewählt: Multi-Select in beiden Systemen**",
" ",
" - Structures sind in `content_config` mit `type = 'structure'`",
"-- Nur im Content Studio verwendet",
"-- Optional: kein Auto-Apply",
"+- **Verwendung in BEIDEN Systemen:** Chat + Content Studio",
"+- **Multi-Select:** Mehrere Strukturen können kombiniert werden",
"+- Speicherung als JSON-Array (wie Collections und Contracts)",
"+- Optional: kein Auto-Apply (leere Auswahl erlaubt)",
" ",
"-### 0j.2 Partial: structure.php",
"+### 0j.2 Use Cases",
" ",
"+| System | Verwendungszweck |",
"+|--------|------------------|",
"+| **Chat** | Ausgabeformat für Antworten (z.B. \"Markdown\", \"Bullet Points\", \"Tabelle\") |",
"+| **Content Studio** | Inhaltsstruktur für generierten Content (z.B. \"Blog-Artikel\", \"Whitepaper\") |",
"+",
"+### 0j.3 Datenbank-Änderungen",
"+",
"+**chat_sessions (ki_content):**",
"+```sql",
"+ALTER TABLE chat_sessions",
"+ADD COLUMN structures JSON DEFAULT '[]' AFTER contracts;",
"+```",
"+",
"+**content_orders (ki_content):**",
"+```sql",
"+-- Von INT zu JSON ändern",
"+ALTER TABLE content_orders",
"+CHANGE COLUMN structure_id structures JSON DEFAULT '[]';",
"+```",
"+",
"+**user_preferences (ki_dev):**",
"+```sql",
"+-- Von INT zu JSON ändern",
"+ALTER TABLE user_preferences",
"+CHANGE COLUMN default_structure_id default_structures JSON DEFAULT '[]';",
"+```",
"+",
"+### 0j.4 Partial: structures-select.php",
"+",
" ```php",
"-\/\/ \/src\/View\/partials\/form\/structure.php",
"+\/\/ \/src\/View\/partials\/form\/structures-select.php",
" <?php",
" \/**",
" * Benötigte Variablen (via Controller):"
]
},
{
"oldStart": 1643,
"oldLines": 56,
"newStart": 1674,
"newLines": 147,
"lines": [
" * - $userPrefs: array aus UserPreferencesService",
" *",
" * Optionale Variablen:",
"- * - $selected: int (Order-Wert)",
"+ * - $selected: array|string (Session\/Order-Wert als JSON oder Array)",
" * - $variant: 'default' | 'inline'",
"- * - $name: string (default: 'structure_id')",
"+ * - $name: string (default: 'structures[]')",
" *\/",
" ",
" $structures = $structures ?? [];",
"-$selected = $selected \/\/ 1. Order-Wert",
"- ?? ($userPrefs['default_structure_id'] ?? null) \/\/ 2. User-Präferenz",
"- ?? null; \/\/ 3. Keine Struktur",
" ",
"-$name = $name ?? 'structure_id';",
"-$id = $id ?? 'structure_id';",
"+\/\/ Selected normalisieren (JSON → Array)",
"+$selected = $selected ?? null;",
"+if (is_string($selected)) {",
"+ $selected = json_decode($selected, true) ?: [];",
"+}",
"+",
"+\/\/ Werte-Hierarchie",
"+if ($selected === null || $selected === []) {",
"+ $userDefault = $userPrefs['default_structures'] ?? '[]';",
"+ if (is_string($userDefault)) {",
"+ $selected = json_decode($userDefault, true) ?: [];",
"+ } else {",
"+ $selected = $userDefault ?: [];",
"+ }",
"+}",
"+",
"+$name = $name ?? 'structures[]';",
"+$id = $id ?? 'structures';",
" $variant = $variant ?? 'default';",
" $class = $variant === 'inline' ? 'form-select--inline' : 'form-select';",
" ?>",
"-<select name=\"<?= $name ?>\" id=\"<?= $id ?>\" class=\"<?= $class ?>\">",
"- <option value=\"\">-- Keine Struktur --<\/option>",
"+<select name=\"<?= $name ?>\" id=\"<?= $id ?>\" class=\"<?= $class ?> structures-select\" multiple",
"+ title=\"Strukturvorlagen auswählen\">",
" <?php foreach ($structures as $structure): ?>",
"- <option value=\"<?= $structure['id'] ?>\" <?= (int)$selected === (int)$structure['id'] ? 'selected' : '' ?>>",
"+ <option value=\"<?= $structure['id'] ?>\"",
"+ <?= in_array($structure['id'], $selected, false) ? 'selected' : '' ?>",
"+ title=\"<?= htmlspecialchars($structure['description'] ?? '') ?>\">",
" <?= htmlspecialchars($structure['name']) ?>",
" <\/option>",
" <?php endforeach; ?>",
" <\/select>",
" ```",
" ",
"-### 0j.3 Werte-Hierarchie",
"+### 0j.5 Werte-Hierarchie",
" ",
" ```",
"-1. Order-Wert → $order['structure_id']",
"- ↓ falls null",
"-2. User-Präferenz → $userPrefs['default_structure_id']",
"- ↓ falls null",
"-3. System-Default → null (keine Struktur)",
"+1. Session\/Order-Wert → $session['structures'] oder $order['structures'] (JSON)",
"+ ↓ falls leer",
"+2. User-Präferenz → $userPrefs['default_structures'] (JSON)",
"+ ↓ falls leer",
"+3. System-Default → [] (leer, keine Struktur erforderlich)",
" ```",
" ",
"-### 0j.4 Verwendung",
"+### 0j.6 Verwendung",
" ",
" | View | Kontext | Variante |",
" |------|---------|----------|",
"-| Chat | - (nicht verwendet) | - |",
"+| Chat | Session-Einstellungen | default |",
" | Content New | Formular | default |",
" | Content Edit | Formular | default |",
"+| Content Show | Inline-Aktionen | inline |",
" ",
"-### 0j.5 Migrations-Schritte (Structure)",
"+### 0j.7 Controller-Anpassungen",
" ",
"-1. [ ] Partial `structure.php` erstellen",
"-2. [ ] Content New auf Partial umstellen",
"-3. [ ] Content Edit auf Partial umstellen",
"-4. [ ] Testen: Default aus User-Präferenzen",
"+**ChatController:**",
"+```php",
"+\/\/ createSession()",
"+$structures = json_decode($prefs['default_structures'] ?? '[]', true);",
" ",
"+$stmt->execute([",
"+ \/\/ ...",
"+ 'structures' => json_encode($structures),",
"+]);",
"+",
"+\/\/ show()",
"+$this->view('chat.index', [",
"+ \/\/ ...",
"+ 'structures' => $this->contentConfigRepository->findByType('structure'),",
"+]);",
"+```",
"+",
"+**ContentController:**",
"+```php",
"+\/\/ store()",
"+$structures = $_POST['structures'] ?? [];",
"+if (!is_array($structures)) {",
"+ $structures = [$structures];",
"+}",
"+",
"+\/\/ In DB speichern",
"+'structures' => json_encode($structures),",
"+```",
"+",
"+### 0j.8 Python-Pipeline Anpassung",
"+",
"+```python",
"+def get_order(order_id):",
"+ # Structures jetzt als JSON-Array",
"+ structures_json = row['structures'] or '[]'",
"+ structures_ids = json.loads(structures_json)",
"+",
"+ # Alle Structures laden",
"+ structures = []",
"+ for sid in structures_ids:",
"+ structure = get_config_item(sid, 'structure')",
"+ if structure:",
"+ structures.append(structure)",
"+",
"+ return {",
"+ # ...",
"+ 'structures': structures, # Liste statt einzelner Wert",
"+ }",
"+```",
"+",
"+### 0j.9 Migrations-Schritte (Structures)",
"+",
"+#### Phase 1: DB-Schema",
"+1. [ ] `chat_sessions`: Spalte `structures JSON` hinzufügen",
"+2. [ ] `content_orders`: Spalte `structure_id` → `structures JSON` migrieren",
"+3. [ ] `user_preferences`: Spalte `default_structure_id` → `default_structures JSON` migrieren",
"+",
"+#### Phase 2: Partial",
"+4. [ ] Partial `structures-select.php` erstellen (Multi-Select)",
"+",
"+#### Phase 3: Chat-Integration",
"+5. [ ] `ChatController::createSession()` anpassen",
"+6. [ ] `ChatController::show()` Structures laden",
"+7. [ ] `chat\/index.php` Partial einbinden",
"+",
"+#### Phase 4: Content-Integration",
"+8. [ ] `ContentController::store()` anpassen (JSON)",
"+9. [ ] `ContentController::contentNew()` Structures laden",
"+10. [ ] `content\/new.php` auf Partial umstellen",
"+11. [ ] `content\/show.php` auf Partial umstellen",
"+",
"+#### Phase 5: Pipeline",
"+12. [ ] `generate.py` auf JSON-Array umstellen",
"+",
"+#### Phase 6: Testing",
"+13. [ ] Chat: Multi-Select funktioniert",
"+14. [ ] Content: Multi-Select funktioniert",
"+15. [ ] Default aus User-Präferenzen",
"+16. [ ] Pipeline verarbeitet mehrere Structures",
"+",
" ---",
" ",
" ## 0k. Ressourcen-Übersicht"
]
}
],
"userModified": false,
"replaceAll": false
}
}