feat: complete Epic 0 — foundation migration & infrastructure setup
Stories 0.2-0.5: rename folders→declarations (backend+frontend), configure Redis for cache/queue/sessions, add foundation database migrations (permissions, archived_at), replace DeclarationStatus enum with architecture lifecycle values, create DeclarationObserver for status transition validation and auto-archive, fix controller status transitions to respect observer rules. 93 tests pass (240 assertions). Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
This commit is contained in:
103
tests/Feature/Declaration/MediaDownloadTest.php
Normal file
103
tests/Feature/Declaration/MediaDownloadTest.php
Normal file
@@ -0,0 +1,103 @@
|
||||
<?php
|
||||
|
||||
use App\Models\Client;
|
||||
use App\Models\Declaration;
|
||||
use App\Models\MediaDownload;
|
||||
use App\Models\User;
|
||||
use App\Models\Workspace;
|
||||
use Illuminate\Http\UploadedFile;
|
||||
use Illuminate\Support\Facades\Storage;
|
||||
|
||||
function setupDeclarationWithMedia(): array
|
||||
{
|
||||
$user = User::factory()->create();
|
||||
$workspace = Workspace::factory()->create();
|
||||
$workspace->users()->attach($user, ['role' => 'owner']);
|
||||
$client = Client::factory()->create(['workspace_id' => $workspace->id]);
|
||||
$declaration = Declaration::factory()->create([
|
||||
'workspace_id' => $workspace->id,
|
||||
'client_id' => $client->id,
|
||||
]);
|
||||
|
||||
Storage::fake('public');
|
||||
$file = UploadedFile::fake()->create('document.pdf', 100, 'application/pdf');
|
||||
$media = $declaration->addMedia($file)->toMediaCollection('documents');
|
||||
|
||||
return [$user, $workspace, $declaration, $media];
|
||||
}
|
||||
|
||||
test('downloading creates a media download record', function () {
|
||||
[$user, $workspace, $declaration, $media] = setupDeclarationWithMedia();
|
||||
session(['current_workspace_id' => $workspace->id]);
|
||||
|
||||
$this->actingAs($user)->get(route('declarations.media.download', [
|
||||
'declaration' => $declaration,
|
||||
'mediaId' => $media->id,
|
||||
]));
|
||||
|
||||
$download = MediaDownload::query()
|
||||
->where('media_id', $media->id)
|
||||
->where('user_id', $user->id)
|
||||
->first();
|
||||
|
||||
expect($download)->not->toBeNull();
|
||||
expect($download->downloaded_at)->not->toBeNull();
|
||||
});
|
||||
|
||||
test('re-downloading updates timestamp without creating duplicates', function () {
|
||||
[$user, $workspace, $declaration, $media] = setupDeclarationWithMedia();
|
||||
session(['current_workspace_id' => $workspace->id]);
|
||||
|
||||
$this->actingAs($user)->get(route('declarations.media.download', [
|
||||
'declaration' => $declaration,
|
||||
'mediaId' => $media->id,
|
||||
]));
|
||||
|
||||
$firstDownload = MediaDownload::query()
|
||||
->where('media_id', $media->id)
|
||||
->where('user_id', $user->id)
|
||||
->first();
|
||||
$firstTimestamp = $firstDownload->downloaded_at;
|
||||
|
||||
$this->travel(5)->minutes();
|
||||
|
||||
$this->actingAs($user)->get(route('declarations.media.download', [
|
||||
'declaration' => $declaration,
|
||||
'mediaId' => $media->id,
|
||||
]));
|
||||
|
||||
$count = MediaDownload::query()
|
||||
->where('media_id', $media->id)
|
||||
->where('user_id', $user->id)
|
||||
->count();
|
||||
|
||||
expect($count)->toBe(1);
|
||||
|
||||
$firstDownload->refresh();
|
||||
expect($firstDownload->downloaded_at->gt($firstTimestamp))->toBeTrue();
|
||||
});
|
||||
|
||||
test('download status is per-user in show endpoint', function () {
|
||||
[$user, $workspace, $declaration, $media] = setupDeclarationWithMedia();
|
||||
$otherUser = User::factory()->create();
|
||||
$workspace->users()->attach($otherUser, ['role' => 'member']);
|
||||
session(['current_workspace_id' => $workspace->id]);
|
||||
|
||||
MediaDownload::query()->create([
|
||||
'media_id' => $media->id,
|
||||
'user_id' => $user->id,
|
||||
'downloaded_at' => now(),
|
||||
]);
|
||||
|
||||
$response = $this->actingAs($user)->get(route('declarations.show', $declaration));
|
||||
$response->assertOk();
|
||||
$documents = $response->original->getData()['page']['props']['documents'];
|
||||
$doc = collect($documents)->firstWhere('id', $media->id);
|
||||
expect($doc['is_downloaded'])->toBeTrue();
|
||||
|
||||
$response2 = $this->actingAs($otherUser)->get(route('declarations.show', $declaration));
|
||||
$response2->assertOk();
|
||||
$documents2 = $response2->original->getData()['page']['props']['documents'];
|
||||
$doc2 = collect($documents2)->firstWhere('id', $media->id);
|
||||
expect($doc2['is_downloaded'])->toBeFalse();
|
||||
});
|
||||
Reference in New Issue
Block a user