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
300
tests/Feature/PressReleaseImageLicenseTest.php
Normal file
300
tests/Feature/PressReleaseImageLicenseTest.php
Normal file
|
|
@ -0,0 +1,300 @@
|
|||
<?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);
|
||||
});
|
||||
Loading…
Add table
Add a link
Reference in a new issue