feat: L'Ami Fiduciaire V1.0.0 — full codebase with Story 0.1 complete

Initial commit of the L'Ami Fiduciaire SaaS platform built on Laravel 12,
Vue 3, Inertia.js 2, and Tailwind CSS 4.

Story 0.1 (rename folders to declarations in database) is implemented and
code-reviewed: migration, rollback, and 6 Pest tests all passing.

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
This commit is contained in:
2026-03-11 23:33:10 +00:00
commit 35545c2a8f
1517 changed files with 246774 additions and 0 deletions

View File

@@ -0,0 +1,134 @@
<?php
use App\Models\Client;
use App\Models\User;
use App\Models\Workspace;
test('can create vat_monthly folder requiring month', function () {
$user = User::factory()->create();
$workspace = Workspace::factory()->create();
$workspace->users()->attach($user, ['role' => 'owner']);
$client = Client::factory()->create(['workspace_id' => $workspace->id]);
session(['current_workspace_id' => $workspace->id]);
$response = $this->actingAs($user)->post(route('folders.store'), [
'client_id' => $client->id,
'title' => 'TVA Mensuelle Mars 2026',
'type' => 'vat_monthly',
'period_year' => 2026,
'period_month' => 3,
'status' => 'draft',
'priority' => 'medium',
]);
$response->assertRedirect();
$folder = $client->folders()->where('type', 'vat_monthly')->first();
expect($folder)->not->toBeNull();
expect($folder->period_month)->toBe(3);
expect($folder->period_quarter)->toBeNull();
});
test('vat_monthly validation fails without month', function () {
$user = User::factory()->create();
$workspace = Workspace::factory()->create();
$workspace->users()->attach($user, ['role' => 'owner']);
$client = Client::factory()->create(['workspace_id' => $workspace->id]);
session(['current_workspace_id' => $workspace->id]);
$response = $this->actingAs($user)->post(route('folders.store'), [
'client_id' => $client->id,
'title' => 'TVA Mensuelle Sans Mois',
'type' => 'vat_monthly',
'period_year' => 2026,
'status' => 'draft',
'priority' => 'medium',
]);
$response->assertSessionHasErrors('period_month');
});
test('can create vat_quarterly folder requiring quarter', function () {
$user = User::factory()->create();
$workspace = Workspace::factory()->create();
$workspace->users()->attach($user, ['role' => 'owner']);
$client = Client::factory()->create(['workspace_id' => $workspace->id]);
session(['current_workspace_id' => $workspace->id]);
$response = $this->actingAs($user)->post(route('folders.store'), [
'client_id' => $client->id,
'title' => 'TVA Trimestrielle T1 2026',
'type' => 'vat_quarterly',
'period_year' => 2026,
'period_quarter' => 1,
'status' => 'draft',
'priority' => 'medium',
]);
$response->assertRedirect();
$folder = $client->folders()->where('type', 'vat_quarterly')->first();
expect($folder)->not->toBeNull();
expect($folder->period_quarter)->toBe(1);
expect($folder->period_month)->toBeNull();
});
test('vat_quarterly validation fails without quarter', function () {
$user = User::factory()->create();
$workspace = Workspace::factory()->create();
$workspace->users()->attach($user, ['role' => 'owner']);
$client = Client::factory()->create(['workspace_id' => $workspace->id]);
session(['current_workspace_id' => $workspace->id]);
$response = $this->actingAs($user)->post(route('folders.store'), [
'client_id' => $client->id,
'title' => 'TVA Trimestrielle Sans Trimestre',
'type' => 'vat_quarterly',
'period_year' => 2026,
'status' => 'draft',
'priority' => 'medium',
]);
$response->assertSessionHasErrors('period_quarter');
});
test('server rejects old vat type', function () {
$user = User::factory()->create();
$workspace = Workspace::factory()->create();
$workspace->users()->attach($user, ['role' => 'owner']);
$client = Client::factory()->create(['workspace_id' => $workspace->id]);
session(['current_workspace_id' => $workspace->id]);
$response = $this->actingAs($user)->post(route('folders.store'), [
'client_id' => $client->id,
'title' => 'Old VAT',
'type' => 'vat',
'period_year' => 2026,
'status' => 'draft',
'priority' => 'medium',
]);
$response->assertSessionHasErrors('type');
});
test('vat_monthly nulls quarter field server-side', function () {
$user = User::factory()->create();
$workspace = Workspace::factory()->create();
$workspace->users()->attach($user, ['role' => 'owner']);
$client = Client::factory()->create(['workspace_id' => $workspace->id]);
session(['current_workspace_id' => $workspace->id]);
$response = $this->actingAs($user)->post(route('folders.store'), [
'client_id' => $client->id,
'title' => 'TVA Mensuelle With Quarter',
'type' => 'vat_monthly',
'period_year' => 2026,
'period_month' => 6,
'period_quarter' => 2,
'status' => 'draft',
'priority' => 'medium',
]);
$response->assertRedirect();
$folder = $client->folders()->where('title', 'TVA Mensuelle With Quarter')->first();
expect($folder->period_quarter)->toBeNull();
expect($folder->period_month)->toBe(6);
});

View File

@@ -0,0 +1,103 @@
<?php
use App\Models\Client;
use App\Models\Folder;
use App\Models\MediaDownload;
use App\Models\User;
use App\Models\Workspace;
use Illuminate\Http\UploadedFile;
use Illuminate\Support\Facades\Storage;
function setupFolderWithMedia(): array
{
$user = User::factory()->create();
$workspace = Workspace::factory()->create();
$workspace->users()->attach($user, ['role' => 'owner']);
$client = Client::factory()->create(['workspace_id' => $workspace->id]);
$folder = Folder::factory()->create([
'workspace_id' => $workspace->id,
'client_id' => $client->id,
]);
Storage::fake('public');
$file = UploadedFile::fake()->create('document.pdf', 100, 'application/pdf');
$media = $folder->addMedia($file)->toMediaCollection('documents');
return [$user, $workspace, $folder, $media];
}
test('downloading creates a media download record', function () {
[$user, $workspace, $folder, $media] = setupFolderWithMedia();
session(['current_workspace_id' => $workspace->id]);
$this->actingAs($user)->get(route('folders.media.download', [
'folder' => $folder,
'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, $folder, $media] = setupFolderWithMedia();
session(['current_workspace_id' => $workspace->id]);
$this->actingAs($user)->get(route('folders.media.download', [
'folder' => $folder,
'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('folders.media.download', [
'folder' => $folder,
'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, $folder, $media] = setupFolderWithMedia();
$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('folders.show', $folder));
$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('folders.show', $folder));
$response2->assertOk();
$documents2 = $response2->original->getData()['page']['props']['documents'];
$doc2 = collect($documents2)->firstWhere('id', $media->id);
expect($doc2['is_downloaded'])->toBeFalse();
});