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>
300 lines
12 KiB
PHP
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);
|
|
});
|