Upload beautify
This commit is contained in:
104
tests/Feature/Admin/FeedPerformanceReportTest.php
Normal file
104
tests/Feature/Admin/FeedPerformanceReportTest.php
Normal file
@@ -0,0 +1,104 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
use App\Models\Artwork;
|
||||
use App\Models\User;
|
||||
use Illuminate\Foundation\Testing\RefreshDatabase;
|
||||
use Illuminate\Support\Facades\DB;
|
||||
|
||||
uses(RefreshDatabase::class);
|
||||
|
||||
it('admin report returns feed performance breakdown and top clicked artworks', function () {
|
||||
$admin = User::factory()->create(['role' => 'admin']);
|
||||
|
||||
$artworkA = Artwork::factory()->create(['title' => 'Feed Artwork A']);
|
||||
$artworkB = Artwork::factory()->create(['title' => 'Feed Artwork B']);
|
||||
|
||||
$metricDate = now()->subDay()->toDateString();
|
||||
|
||||
DB::table('feed_daily_metrics')->insert([
|
||||
[
|
||||
'metric_date' => $metricDate,
|
||||
'algo_version' => 'clip-cosine-v1',
|
||||
'source' => 'personalized',
|
||||
'impressions' => 10,
|
||||
'clicks' => 4,
|
||||
'saves' => 2,
|
||||
'ctr' => 0.4,
|
||||
'save_rate' => 0.5,
|
||||
'dwell_0_5' => 1,
|
||||
'dwell_5_30' => 1,
|
||||
'dwell_30_120' => 1,
|
||||
'dwell_120_plus' => 1,
|
||||
'created_at' => now(),
|
||||
'updated_at' => now(),
|
||||
],
|
||||
]);
|
||||
|
||||
DB::table('feed_events')->insert([
|
||||
[
|
||||
'event_date' => $metricDate,
|
||||
'event_type' => 'feed_impression',
|
||||
'user_id' => $admin->id,
|
||||
'artwork_id' => $artworkA->id,
|
||||
'position' => 1,
|
||||
'algo_version' => 'clip-cosine-v1',
|
||||
'source' => 'personalized',
|
||||
'dwell_seconds' => null,
|
||||
'occurred_at' => now()->subDay(),
|
||||
'created_at' => now(),
|
||||
'updated_at' => now(),
|
||||
],
|
||||
[
|
||||
'event_date' => $metricDate,
|
||||
'event_type' => 'feed_click',
|
||||
'user_id' => $admin->id,
|
||||
'artwork_id' => $artworkA->id,
|
||||
'position' => 1,
|
||||
'algo_version' => 'clip-cosine-v1',
|
||||
'source' => 'personalized',
|
||||
'dwell_seconds' => 12,
|
||||
'occurred_at' => now()->subDay(),
|
||||
'created_at' => now(),
|
||||
'updated_at' => now(),
|
||||
],
|
||||
[
|
||||
'event_date' => $metricDate,
|
||||
'event_type' => 'feed_click',
|
||||
'user_id' => $admin->id,
|
||||
'artwork_id' => $artworkB->id,
|
||||
'position' => 2,
|
||||
'algo_version' => 'clip-cosine-v1',
|
||||
'source' => 'personalized',
|
||||
'dwell_seconds' => 7,
|
||||
'occurred_at' => now()->subDay(),
|
||||
'created_at' => now(),
|
||||
'updated_at' => now(),
|
||||
],
|
||||
]);
|
||||
|
||||
$response = $this->actingAs($admin)->getJson('/api/admin/reports/feed-performance?from=' . $metricDate . '&to=' . $metricDate);
|
||||
|
||||
$response->assertOk();
|
||||
$response->assertJsonPath('meta.from', $metricDate);
|
||||
$response->assertJsonPath('meta.to', $metricDate);
|
||||
|
||||
$rows = collect($response->json('by_algo_source'));
|
||||
expect($rows->count())->toBe(1);
|
||||
expect($rows->first()['algo_version'])->toBe('clip-cosine-v1');
|
||||
expect($rows->first()['source'])->toBe('personalized');
|
||||
expect((float) $rows->first()['ctr'])->toBe(0.4);
|
||||
|
||||
$top = collect($response->json('top_clicked_artworks'));
|
||||
expect($top->isNotEmpty())->toBeTrue();
|
||||
expect((int) $top->first()['artwork_id'])->toBe($artworkA->id);
|
||||
});
|
||||
|
||||
it('non-admin is denied feed performance report endpoint', function () {
|
||||
$user = User::factory()->create(['role' => 'user']);
|
||||
|
||||
$response = $this->actingAs($user)->getJson('/api/admin/reports/feed-performance');
|
||||
|
||||
$response->assertStatus(403);
|
||||
});
|
||||
99
tests/Feature/Admin/SimilarArtworkReportTest.php
Normal file
99
tests/Feature/Admin/SimilarArtworkReportTest.php
Normal file
@@ -0,0 +1,99 @@
|
||||
<?php
|
||||
|
||||
use App\Models\Artwork;
|
||||
use App\Models\User;
|
||||
use Illuminate\Foundation\Testing\RefreshDatabase;
|
||||
use Illuminate\Support\Facades\DB;
|
||||
|
||||
uses(RefreshDatabase::class);
|
||||
|
||||
it('admin report returns date-filtered breakdown by algo and top similarities with ctr', function () {
|
||||
$admin = User::factory()->create(['role' => 'admin']);
|
||||
|
||||
$sourceA = Artwork::factory()->create(['title' => 'Source A']);
|
||||
$similarA = Artwork::factory()->create(['title' => 'Similar A']);
|
||||
$sourceB = Artwork::factory()->create(['title' => 'Source B']);
|
||||
$similarB = Artwork::factory()->create(['title' => 'Similar B']);
|
||||
|
||||
$inRangeDate = now()->subDay()->toDateString();
|
||||
$outRangeDate = now()->subDays(5)->toDateString();
|
||||
|
||||
DB::table('similar_artwork_events')->insert([
|
||||
[
|
||||
'event_date' => $inRangeDate,
|
||||
'event_type' => 'impression',
|
||||
'algo_version' => 'clip-cosine-v1',
|
||||
'source_artwork_id' => $sourceA->id,
|
||||
'similar_artwork_id' => $similarA->id,
|
||||
'position' => 1,
|
||||
'items_count' => 4,
|
||||
'occurred_at' => now()->subDay(),
|
||||
'created_at' => now(),
|
||||
'updated_at' => now(),
|
||||
],
|
||||
[
|
||||
'event_date' => $inRangeDate,
|
||||
'event_type' => 'click',
|
||||
'algo_version' => 'clip-cosine-v1',
|
||||
'source_artwork_id' => $sourceA->id,
|
||||
'similar_artwork_id' => $similarA->id,
|
||||
'position' => 1,
|
||||
'items_count' => null,
|
||||
'occurred_at' => now()->subDay(),
|
||||
'created_at' => now(),
|
||||
'updated_at' => now(),
|
||||
],
|
||||
[
|
||||
'event_date' => $inRangeDate,
|
||||
'event_type' => 'impression',
|
||||
'algo_version' => 'clip-cosine-v2',
|
||||
'source_artwork_id' => $sourceB->id,
|
||||
'similar_artwork_id' => $similarB->id,
|
||||
'position' => 2,
|
||||
'items_count' => 4,
|
||||
'occurred_at' => now()->subDay(),
|
||||
'created_at' => now(),
|
||||
'updated_at' => now(),
|
||||
],
|
||||
[
|
||||
'event_date' => $outRangeDate,
|
||||
'event_type' => 'click',
|
||||
'algo_version' => 'clip-cosine-v1',
|
||||
'source_artwork_id' => $sourceA->id,
|
||||
'similar_artwork_id' => $similarA->id,
|
||||
'position' => 1,
|
||||
'items_count' => null,
|
||||
'occurred_at' => now()->subDays(5),
|
||||
'created_at' => now(),
|
||||
'updated_at' => now(),
|
||||
],
|
||||
]);
|
||||
|
||||
$response = $this->actingAs($admin)->getJson('/api/admin/reports/similar-artworks?from=' . $inRangeDate . '&to=' . $inRangeDate);
|
||||
|
||||
$response->assertOk();
|
||||
$response->assertJsonPath('meta.from', $inRangeDate);
|
||||
$response->assertJsonPath('meta.to', $inRangeDate);
|
||||
|
||||
$byAlgo = collect($response->json('by_algo_version'));
|
||||
expect($byAlgo->count())->toBe(2);
|
||||
|
||||
$v1 = $byAlgo->firstWhere('algo_version', 'clip-cosine-v1');
|
||||
expect($v1['impressions'])->toBe(1);
|
||||
expect($v1['clicks'])->toBe(1);
|
||||
expect((float) $v1['ctr'])->toBe(1.0);
|
||||
|
||||
$top = collect($response->json('top_similarities'));
|
||||
expect($top->isNotEmpty())->toBeTrue();
|
||||
expect($top->first()['source_artwork_id'])->toBe($sourceA->id);
|
||||
expect($top->first()['similar_artwork_id'])->toBe($similarA->id);
|
||||
expect((float) $top->first()['ctr'])->toBe(1.0);
|
||||
});
|
||||
|
||||
it('non-admin is denied similar artwork report endpoint', function () {
|
||||
$user = User::factory()->create(['role' => 'user']);
|
||||
|
||||
$response = $this->actingAs($user)->getJson('/api/admin/reports/similar-artworks');
|
||||
|
||||
$response->assertStatus(403);
|
||||
});
|
||||
168
tests/Feature/Admin/UploadModerationTest.php
Normal file
168
tests/Feature/Admin/UploadModerationTest.php
Normal file
@@ -0,0 +1,168 @@
|
||||
<?php
|
||||
|
||||
use App\Models\User;
|
||||
use Illuminate\Foundation\Testing\RefreshDatabase;
|
||||
use Illuminate\Support\Facades\DB;
|
||||
use Illuminate\Support\Facades\Storage;
|
||||
use Illuminate\Support\Str;
|
||||
|
||||
uses(RefreshDatabase::class);
|
||||
|
||||
function createModerationCategory(): int
|
||||
{
|
||||
$contentTypeId = DB::table('content_types')->insertGetId([
|
||||
'name' => 'Photography',
|
||||
'slug' => 'photography-' . Str::lower(Str::random(6)),
|
||||
'description' => null,
|
||||
'created_at' => now(),
|
||||
'updated_at' => now(),
|
||||
]);
|
||||
|
||||
return DB::table('categories')->insertGetId([
|
||||
'content_type_id' => $contentTypeId,
|
||||
'parent_id' => null,
|
||||
'name' => 'Moderation',
|
||||
'slug' => 'moderation-' . Str::lower(Str::random(6)),
|
||||
'description' => null,
|
||||
'image' => null,
|
||||
'is_active' => true,
|
||||
'sort_order' => 0,
|
||||
'created_at' => now(),
|
||||
'updated_at' => now(),
|
||||
]);
|
||||
}
|
||||
|
||||
function createModerationDraft(int $userId, int $categoryId, array $overrides = []): string
|
||||
{
|
||||
$uploadId = (string) Str::uuid();
|
||||
|
||||
DB::table('uploads')->insert(array_merge([
|
||||
'id' => $uploadId,
|
||||
'user_id' => $userId,
|
||||
'type' => 'image',
|
||||
'status' => 'draft',
|
||||
'processing_state' => 'ready',
|
||||
'moderation_status' => 'pending',
|
||||
'title' => 'Pending Moderation Upload',
|
||||
'category_id' => $categoryId,
|
||||
'is_scanned' => true,
|
||||
'has_tags' => true,
|
||||
'preview_path' => "tmp/drafts/{$uploadId}/preview.webp",
|
||||
'created_at' => now(),
|
||||
'updated_at' => now(),
|
||||
], $overrides));
|
||||
|
||||
return $uploadId;
|
||||
}
|
||||
|
||||
function addReadyMainFile(string $uploadId, string $hash = 'aabbccddeeff00112233'): void
|
||||
{
|
||||
Storage::disk('local')->put("tmp/drafts/{$uploadId}/main/main.jpg", 'jpg');
|
||||
Storage::disk('local')->put("tmp/drafts/{$uploadId}/preview.webp", 'preview');
|
||||
|
||||
DB::table('upload_files')->insert([
|
||||
'upload_id' => $uploadId,
|
||||
'path' => "tmp/drafts/{$uploadId}/main/main.jpg",
|
||||
'type' => 'main',
|
||||
'hash' => $hash,
|
||||
'size' => 3,
|
||||
'mime' => 'image/jpeg',
|
||||
'created_at' => now(),
|
||||
]);
|
||||
}
|
||||
|
||||
it('admin sees pending uploads', function () {
|
||||
$admin = User::factory()->create(['role' => 'admin']);
|
||||
$owner = User::factory()->create();
|
||||
$categoryId = createModerationCategory();
|
||||
|
||||
createModerationDraft($owner->id, $categoryId, ['title' => 'First Pending']);
|
||||
createModerationDraft($owner->id, $categoryId, ['title' => 'Second Pending']);
|
||||
|
||||
$response = $this->actingAs($admin)->getJson('/api/admin/uploads/pending');
|
||||
|
||||
$response->assertOk();
|
||||
$response->assertJsonCount(2, 'data');
|
||||
});
|
||||
|
||||
it('non-admin is denied moderation API access', function () {
|
||||
$user = User::factory()->create(['role' => 'user']);
|
||||
|
||||
$response = $this->actingAs($user)->getJson('/api/admin/uploads/pending');
|
||||
|
||||
$response->assertStatus(403);
|
||||
});
|
||||
|
||||
it('approve works', function () {
|
||||
$admin = User::factory()->create(['role' => 'moderator']);
|
||||
$owner = User::factory()->create();
|
||||
$categoryId = createModerationCategory();
|
||||
$uploadId = createModerationDraft($owner->id, $categoryId);
|
||||
|
||||
$response = $this->actingAs($admin)->postJson("/api/admin/uploads/{$uploadId}/approve", [
|
||||
'note' => 'Looks good.',
|
||||
]);
|
||||
|
||||
$response->assertOk();
|
||||
|
||||
$row = DB::table('uploads')->where('id', $uploadId)->first([
|
||||
'moderation_status',
|
||||
'moderation_note',
|
||||
'moderated_by',
|
||||
'moderated_at',
|
||||
]);
|
||||
|
||||
expect($row->moderation_status)->toBe('approved');
|
||||
expect($row->moderation_note)->toBe('Looks good.');
|
||||
expect((int) $row->moderated_by)->toBe((int) $admin->id);
|
||||
expect($row->moderated_at)->not->toBeNull();
|
||||
});
|
||||
|
||||
it('reject works', function () {
|
||||
$admin = User::factory()->create(['role' => 'admin']);
|
||||
$owner = User::factory()->create();
|
||||
$categoryId = createModerationCategory();
|
||||
$uploadId = createModerationDraft($owner->id, $categoryId);
|
||||
|
||||
$response = $this->actingAs($admin)->postJson("/api/admin/uploads/{$uploadId}/reject", [
|
||||
'note' => 'Policy violation.',
|
||||
]);
|
||||
|
||||
$response->assertOk();
|
||||
|
||||
$row = DB::table('uploads')->where('id', $uploadId)->first([
|
||||
'status',
|
||||
'processing_state',
|
||||
'moderation_status',
|
||||
'moderation_note',
|
||||
'moderated_by',
|
||||
'moderated_at',
|
||||
]);
|
||||
|
||||
expect($row->status)->toBe('rejected');
|
||||
expect($row->processing_state)->toBe('rejected');
|
||||
expect($row->moderation_status)->toBe('rejected');
|
||||
expect($row->moderation_note)->toBe('Policy violation.');
|
||||
expect((int) $row->moderated_by)->toBe((int) $admin->id);
|
||||
expect($row->moderated_at)->not->toBeNull();
|
||||
});
|
||||
|
||||
it('user cannot publish without approval', function () {
|
||||
Storage::fake('local');
|
||||
|
||||
$owner = User::factory()->create(['role' => 'user']);
|
||||
$categoryId = createModerationCategory();
|
||||
$uploadId = createModerationDraft($owner->id, $categoryId, [
|
||||
'moderation_status' => 'pending',
|
||||
'title' => 'Blocked Publish',
|
||||
]);
|
||||
|
||||
addReadyMainFile($uploadId);
|
||||
|
||||
$response = $this->actingAs($owner)->postJson("/api/uploads/{$uploadId}/publish");
|
||||
|
||||
$response->assertStatus(422);
|
||||
$response->assertJsonFragment([
|
||||
'message' => 'Upload requires moderation approval before publish.',
|
||||
]);
|
||||
});
|
||||
Reference in New Issue
Block a user