presseportale/tests/Feature/PressReleaseImageLicenseTest.php
Kevin Adametz a000238ca8 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>
2026-06-12 08:30:13 +00:00

300 lines
12 KiB
PHP

<?php
use App\Enums\ImageLicenseType;
use App\Models\Category;
use App\Models\Company;
use App\Models\PressRelease;
use App\Models\User;
use Database\Seeders\RolesAndPermissionsSeeder;
use Illuminate\Http\UploadedFile;
use Illuminate\Support\Facades\Storage;
use Livewire\Volt\Volt as LivewireVolt;
use Tests\TestCase;
beforeEach(function (): void {
/** @var TestCase $this */
$this->seed(RolesAndPermissionsSeeder::class);
Storage::fake('public');
});
function makeImageDraftOwner(): array
{
$owner = User::factory()->create(['is_active' => true]);
$owner->assignRole('customer');
$company = Company::factory()->presseecho()->create();
$owner->companies()->attach($company->id, ['role' => 'owner']);
$pr = PressRelease::factory()->create([
'user_id' => $owner->id,
'company_id' => $company->id,
'category_id' => Category::factory()->create()->id,
'portal' => $company->portal->value,
'status' => 'draft',
]);
return compact('owner', 'pr');
}
test('image upload requires author, license type and rights confirmation', function () {
/** @var TestCase $this */
['owner' => $owner, 'pr' => $pr] = makeImageDraftOwner();
$this->actingAs($owner);
LivewireVolt::test('components.press-release-images-manager', ['pressReleaseId' => $pr->id])
->set('newImage', UploadedFile::fake()->image('foto.jpg', 1200, 800))
->call('saveImage')
->assertHasErrors(['newAuthor', 'newLicenseType', 'newPeopleRightsStatus', 'newPropertyRightsStatus', 'newRightsConfirmed']);
expect($pr->images()->count())->toBe(0);
});
test('license type starts with an explicit placeholder before own photo option', function () {
/** @var TestCase $this */
['owner' => $owner, 'pr' => $pr] = makeImageDraftOwner();
$this->actingAs($owner);
LivewireVolt::test('components.press-release-images-manager', ['pressReleaseId' => $pr->id])
->call('openUploadForm')
->assertSet('newLicenseType', '')
->assertSee('Bitte wählen')
->assertSee('Eigene Aufnahme');
});
test('title image upload form is collapsed until requested', function () {
/** @var TestCase $this */
['owner' => $owner, 'pr' => $pr] = makeImageDraftOwner();
$this->actingAs($owner);
LivewireVolt::test('components.press-release-images-manager', ['pressReleaseId' => $pr->id])
->assertSee('Hier fehlt ein Titelbild')
->assertSee('Eigenes Titelbild hochladen')
->assertDontSee('Bild hierher ziehen oder klicken')
->call('openUploadForm')
->assertSee('Titelbild hochladen')
->assertSee('Bild hierher ziehen oder klicken')
->assertDontSee('Als Vorschaubild verwenden')
->assertDontSee('Unsicher');
});
test('unclear rights selections are not accepted', function () {
/** @var TestCase $this */
['owner' => $owner, 'pr' => $pr] = makeImageDraftOwner();
$this->actingAs($owner);
LivewireVolt::test('components.press-release-images-manager', ['pressReleaseId' => $pr->id])
->set('newImage', UploadedFile::fake()->image('foto.jpg', 1200, 800))
->set('newAuthor', 'Jane Doe')
->set('newLicenseType', ImageLicenseType::Own->value)
->set('newPeopleRightsStatus', 'unsure')
->set('newPropertyRightsStatus', 'unsure')
->set('newRightsConfirmed', true)
->call('saveImage')
->assertHasErrors(['newPeopleRightsStatus', 'newPropertyRightsStatus']);
expect($pr->images()->count())->toBe(0);
});
test('cc license requires a license url', function () {
/** @var TestCase $this */
['owner' => $owner, 'pr' => $pr] = makeImageDraftOwner();
$this->actingAs($owner);
LivewireVolt::test('components.press-release-images-manager', ['pressReleaseId' => $pr->id])
->set('newImage', UploadedFile::fake()->image('foto.jpg', 1200, 800))
->set('newAuthor', 'Jane Doe')
->set('newLicenseType', ImageLicenseType::CreativeCommons->value)
->set('newLicenseDetail', 'cc_by')
->set('newPeopleRightsStatus', 'none')
->set('newPropertyRightsStatus', 'none')
->set('newRightsConfirmed', true)
->call('saveImage')
->assertHasErrors(['newLicenseUrl']);
expect($pr->images()->count())->toBe(0);
});
test('cc license requires a concrete cc variant', function () {
/** @var TestCase $this */
['owner' => $owner, 'pr' => $pr] = makeImageDraftOwner();
$this->actingAs($owner);
LivewireVolt::test('components.press-release-images-manager', ['pressReleaseId' => $pr->id])
->set('newImage', UploadedFile::fake()->image('foto.jpg', 1200, 800))
->set('newAuthor', 'Jane Doe')
->set('newLicenseType', ImageLicenseType::CreativeCommons->value)
->set('newLicenseUrl', 'https://creativecommons.org/licenses/by/4.0/')
->set('newPeopleRightsStatus', 'none')
->set('newPropertyRightsStatus', 'none')
->set('newRightsConfirmed', true)
->call('saveImage')
->assertHasErrors(['newLicenseDetail']);
expect($pr->images()->count())->toBe(0);
});
test('other license requires details', function () {
/** @var TestCase $this */
['owner' => $owner, 'pr' => $pr] = makeImageDraftOwner();
$this->actingAs($owner);
LivewireVolt::test('components.press-release-images-manager', ['pressReleaseId' => $pr->id])
->set('newImage', UploadedFile::fake()->image('foto.jpg', 1200, 800))
->set('newAuthor', 'Jane Doe')
->set('newLicenseType', ImageLicenseType::Other->value)
->set('newPeopleRightsStatus', 'none')
->set('newPropertyRightsStatus', 'none')
->set('newRightsConfirmed', true)
->call('saveImage')
->assertHasErrors(['newLicenseDetail']);
expect($pr->images()->count())->toBe(0);
});
test('non previewable image uploads fail validation without preview exception', function () {
/** @var TestCase $this */
['owner' => $owner, 'pr' => $pr] = makeImageDraftOwner();
$this->actingAs($owner);
LivewireVolt::test('components.press-release-images-manager', ['pressReleaseId' => $pr->id])
->set('newImage', UploadedFile::fake()->create('scan.tif', 100, 'image/tiff'))
->set('newAuthor', 'Jane Doe')
->set('newLicenseType', ImageLicenseType::Own->value)
->set('newPeopleRightsStatus', 'none')
->set('newPropertyRightsStatus', 'none')
->set('newRightsConfirmed', true)
->call('saveImage')
->assertHasErrors(['newImage']);
expect($pr->images()->count())->toBe(0);
});
test('valid upload stores license metadata and confirms rights', function () {
/** @var TestCase $this */
['owner' => $owner, 'pr' => $pr] = makeImageDraftOwner();
$this->actingAs($owner);
LivewireVolt::test('components.press-release-images-manager', ['pressReleaseId' => $pr->id])
->set('newImage', UploadedFile::fake()->image('foto.jpg', 1200, 800))
->set('newTitle', 'Maschine im Einsatz')
->set('newCopyright', 'Foto: Jane Doe / Beispiel GmbH')
->set('newAuthor', 'Jane Doe')
->set('newLicenseType', ImageLicenseType::Own->value)
->set('newSourceUrl', 'https://example.com/source')
->set('newPeopleRightsStatus', 'consent')
->set('newPropertyRightsStatus', 'cleared')
->set('newRightsNotes', 'Freigabe liegt intern vor.')
->set('newRightsConfirmed', true)
->call('saveImage')
->assertHasNoErrors();
$image = $pr->images()->first();
expect($image)->not->toBeNull();
expect($image->title)->toBe('Maschine im Einsatz');
expect($image->copyright)->toBe('Foto: Jane Doe / Beispiel GmbH');
expect($image->author)->toBe('Jane Doe');
expect($image->license_type)->toBe(ImageLicenseType::Own);
expect($image->source_url)->toBe('https://example.com/source');
expect($image->people_rights_status)->toBe('consent');
expect($image->property_rights_status)->toBe('cleared');
expect($image->rights_notes)->toBe('Freigabe liegt intern vor.');
expect($image->persons_consent)->toBeTrue();
expect($image->rights_confirmed_at)->not->toBeNull();
expect($image->is_preview)->toBeTrue();
expect($image->path)->toBe($image->variants['cover']);
expect($image->width)->toBe(1280);
expect($image->height)->toBe(580);
Storage::disk('public')->assertExists($image->path);
$originalPath = preg_replace(
'#/variants/([^/]+)-cover(\.[^.]+)$#',
'/$1$2',
$image->path,
);
expect($originalPath)->toBeString()->not->toBe($image->path);
Storage::disk('public')->assertMissing($originalPath);
});
test('valid cc upload stores license detail and license url', function () {
/** @var TestCase $this */
['owner' => $owner, 'pr' => $pr] = makeImageDraftOwner();
$this->actingAs($owner);
LivewireVolt::test('components.press-release-images-manager', ['pressReleaseId' => $pr->id])
->set('newImage', UploadedFile::fake()->image('foto.jpg', 1200, 800))
->set('newAuthor', 'Jane Doe')
->set('newLicenseType', ImageLicenseType::CreativeCommons->value)
->set('newLicenseDetail', 'cc_by')
->set('newLicenseUrl', 'https://creativecommons.org/licenses/by/4.0/')
->set('newPeopleRightsStatus', 'none')
->set('newPropertyRightsStatus', 'none')
->set('newRightsConfirmed', true)
->call('saveImage')
->assertHasNoErrors();
$image = $pr->images()->first();
expect($image)->not->toBeNull();
expect($image->license_type)->toBe(ImageLicenseType::CreativeCommons);
expect($image->license_detail)->toBe('cc_by');
expect($image->license_url)->toBe('https://creativecommons.org/licenses/by/4.0/');
});
test('existing title image hides upload form and can be removed', function () {
/** @var TestCase $this */
['owner' => $owner, 'pr' => $pr] = makeImageDraftOwner();
$this->actingAs($owner);
$image = $pr->images()->create([
'disk' => 'public',
'path' => 'press-releases/'.$pr->id.'/images/title.jpg',
'variants' => ['cover' => 'press-releases/'.$pr->id.'/images/title-cover.jpg'],
'title' => 'Messefoto',
'copyright' => 'Pressefoto GmbH',
'author' => 'Jane Doe',
'license_type' => ImageLicenseType::Own->value,
'rights_confirmed_at' => now(),
'is_preview' => true,
'sort_order' => 1,
]);
LivewireVolt::test('components.press-release-images-manager', ['pressReleaseId' => $pr->id])
->assertSee('Messefoto')
->assertSee('Bildnachweis: Pressefoto GmbH')
->assertSee('Titelbild löschen')
->assertDontSee('Titelbild hochladen')
->call('remove', $image->id)
->assertSee('Eigenes Titelbild hochladen');
expect($pr->images()->count())->toBe(0);
});
test('second title image upload is blocked while one exists', function () {
/** @var TestCase $this */
['owner' => $owner, 'pr' => $pr] = makeImageDraftOwner();
$this->actingAs($owner);
$pr->images()->create([
'disk' => 'public',
'path' => 'press-releases/'.$pr->id.'/images/title.jpg',
'variants' => ['cover' => 'press-releases/'.$pr->id.'/images/title-cover.jpg'],
'author' => 'Jane Doe',
'license_type' => ImageLicenseType::Own->value,
'rights_confirmed_at' => now(),
'is_preview' => true,
'sort_order' => 1,
]);
LivewireVolt::test('components.press-release-images-manager', ['pressReleaseId' => $pr->id])
->set('newImage', UploadedFile::fake()->image('zweites.jpg', 1200, 800))
->set('newAuthor', 'Jane Doe')
->set('newLicenseType', ImageLicenseType::Own->value)
->set('newRightsConfirmed', true)
->call('saveImage')
->assertHasErrors(['newImage']);
expect($pr->images()->count())->toBe(1);
});