User Panel: Phase-8-Abschluss, Titelbild/Lizenzen/Zeitzonen und KI-Pruef-Pipeline
Phase 8 (Rest) + Umbauten vom 10./11.06.: - Ein Titelbild pro PM (Cover 1280x580), SVG-Platzhalter-Set + Picker, PressReleaseCoverImage-Resolver - Lizenz-/Rechteformular nach "Lizenztyp Bildupload" (7 Lizenztypen, Personen-/Sachrechte-Status, bedingte Pflichtfelder, Risikohinweise) - Veroeffentlichungs-Box vereinfacht (Embargo aus der Form-UI entfernt), geplante Termine in Europe/Berlin (Speicherung UTC, DISPLAY_TIMEZONE) - Quota-Stub (users.press_release_quota) + monatlicher Reset-Command - Einreichungs-Modal einheitlich in Show/Create/Edit; Ghost-Buttons auf filled; PM-Editor-Layout responsive entkoppelt (.pr-editor-layout) KI-Pruef-Pipeline (Phasen 1-5 des Entwicklungsplans): - API-Haertung: status nicht mehr per API setzbar, eigene Submit-Route durch denselben Funnel (Blacklist, Quota, Status-Log) - Klassifikation Rot/Gelb/Gruen asynchron (Queue classification, OpenAI-Treiber + deterministischer Fallback), ki_audits-Audit-Log - Routing: Rot -> rejected + Mail, Gelb -> Review-Queue, Gruen -> Auto-Publish; Scheduler publiziert nur gruene faellige PMs - Content-Score 0-100 -> Stufe (Standard/Geprueft/Hochwertig) inkl. Editor-Panel und Badges; Re-Klassifikation/-Score bei Aenderung - Admin: KI-Badge + Filter, On-Demand-Pruefung mit Anbieter-Override Suite: 442 passed, 4 skipped. Co-Authored-By: Claude Fable 5 <noreply@anthropic.com>
This commit is contained in:
parent
0efabaf446
commit
a000238ca8
141 changed files with 5922 additions and 1001 deletions
|
|
@ -0,0 +1,115 @@
|
|||
<?php
|
||||
|
||||
namespace App\Services\PressRelease\Classification\Drivers;
|
||||
|
||||
use App\Enums\PressReleaseClassification;
|
||||
use App\Models\PressRelease;
|
||||
use App\Services\PressRelease\Classification\ClassificationResult;
|
||||
use App\Services\PressRelease\Classification\Contracts\ClassificationDriver;
|
||||
use Illuminate\Support\Facades\Http;
|
||||
use RuntimeException;
|
||||
|
||||
/**
|
||||
* Klassifikations-Treiber auf Basis der OpenAI Chat-Completions-API.
|
||||
*
|
||||
* Liest Zugangsdaten/Modell aus config/services.openai. Verlangt eine
|
||||
* strukturierte JSON-Antwort `{classification, reasons[]}`. Bei fehlendem
|
||||
* Key, Transport-/HTTP-Fehlern oder ungültiger Antwort wird eine Exception
|
||||
* geworfen, sodass der Job auf den deterministischen Treiber ausweicht.
|
||||
*/
|
||||
class OpenAiClassificationDriver implements ClassificationDriver
|
||||
{
|
||||
public function classify(PressRelease $pressRelease): ClassificationResult
|
||||
{
|
||||
$config = config('services.openai');
|
||||
$apiKey = $config['api_key'] ?? null;
|
||||
|
||||
if (blank($apiKey)) {
|
||||
throw new RuntimeException('OpenAI API-Key ist nicht konfiguriert.');
|
||||
}
|
||||
|
||||
$model = config('scoring.classification.model') ?: ($config['model'] ?? 'gpt-5.4-mini');
|
||||
$timeout = (int) (config('scoring.classification.timeout') ?: ($config['timeout'] ?? 30));
|
||||
|
||||
$response = Http::withToken($apiKey)
|
||||
->timeout($timeout)
|
||||
->acceptJson()
|
||||
->post($config['url'] ?? 'https://api.openai.com/v1/chat/completions', [
|
||||
'model' => $model,
|
||||
'response_format' => ['type' => 'json_object'],
|
||||
'messages' => [
|
||||
['role' => 'system', 'content' => $this->systemPrompt()],
|
||||
['role' => 'user', 'content' => $this->userPrompt($pressRelease)],
|
||||
],
|
||||
]);
|
||||
|
||||
if ($response->failed()) {
|
||||
throw new RuntimeException("OpenAI-Klassifikation fehlgeschlagen (HTTP {$response->status()}).");
|
||||
}
|
||||
|
||||
$payload = $response->json();
|
||||
$content = data_get($payload, 'choices.0.message.content');
|
||||
|
||||
if (! is_string($content) || trim($content) === '') {
|
||||
throw new RuntimeException('OpenAI-Antwort enthielt keinen Inhalt.');
|
||||
}
|
||||
|
||||
$parsed = json_decode($content, true);
|
||||
|
||||
if (! is_array($parsed) || ! isset($parsed['classification'])) {
|
||||
throw new RuntimeException('OpenAI-Antwort war kein gültiges Klassifikations-JSON.');
|
||||
}
|
||||
|
||||
$classification = PressReleaseClassification::tryFrom((string) $parsed['classification']);
|
||||
|
||||
if ($classification === null) {
|
||||
throw new RuntimeException('OpenAI lieferte einen unbekannten Klassifikationswert.');
|
||||
}
|
||||
|
||||
$reasons = array_values(array_filter(array_map(
|
||||
static fn ($reason): string => (string) $reason,
|
||||
is_array($parsed['reasons'] ?? null) ? $parsed['reasons'] : [],
|
||||
)));
|
||||
|
||||
return new ClassificationResult(
|
||||
classification: $classification,
|
||||
reasons: $reasons,
|
||||
provider: 'openai',
|
||||
model: $model,
|
||||
rawResponse: is_array($payload) ? $payload : [],
|
||||
);
|
||||
}
|
||||
|
||||
private function systemPrompt(): string
|
||||
{
|
||||
return <<<'PROMPT'
|
||||
Du bist ein redaktioneller Prüf-Assistent für ein deutsches Presseportal.
|
||||
Bewerte, ob eine eingereichte Pressemitteilung veröffentlicht werden darf.
|
||||
|
||||
Klassifiziere genau eine der drei Stufen:
|
||||
- "green": unauffällig, kann veröffentlicht werden.
|
||||
- "yellow": grenzwertig/unklar, sollte manuell geprüft werden.
|
||||
- "red": unzulässig, darf nicht veröffentlicht werden.
|
||||
|
||||
Prüfe insbesondere diese Faktoren (Red Flags):
|
||||
- reine Werbung statt journalistischer Pressemitteilung
|
||||
- beleidigende, diskriminierende oder hetzerische Inhalte
|
||||
- rechtlich heikle Aussagen (z. B. unbelegte Heil-/Gewinnversprechen,
|
||||
Verleumdung, Aufruf zu Straftaten)
|
||||
- Spam-Muster (sinnlose Keyword-Wiederholung, irreführende Links)
|
||||
- unseriöse oder offensichtlich falsche Versprechen
|
||||
|
||||
Antworte AUSSCHLIESSLICH als JSON-Objekt in genau diesem Schema:
|
||||
{"classification": "green|yellow|red", "reasons": ["kurze Begründung", ...]}
|
||||
Bei "green" darf "reasons" leer sein. Schreibe Begründungen auf Deutsch.
|
||||
PROMPT;
|
||||
}
|
||||
|
||||
private function userPrompt(PressRelease $pressRelease): string
|
||||
{
|
||||
$title = (string) $pressRelease->title;
|
||||
$text = trim(strip_tags((string) $pressRelease->text));
|
||||
|
||||
return "Titel:\n{$title}\n\nText:\n{$text}";
|
||||
}
|
||||
}
|
||||
Loading…
Add table
Add a link
Reference in a new issue