feat(projects): source-lock state для UI (guard.lockState + ProjectResource + анти-N+1)

SupplierSnapshotGuard::lockState (pure, без DB) + ProjectResource отдаёт source_locked/source_unlock_at/source_unlock_projected; ProjectController withCount(supplierProjects). Логика гарда не изменена.

Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
This commit is contained in:
Дмитрий
2026-06-22 20:52:18 +03:00
parent 9901e74f89
commit 08cf23893a
6 changed files with 160 additions and 16 deletions
@@ -0,0 +1,44 @@
<?php
declare(strict_types=1);
use App\Http\Resources\ProjectResource;
use App\Models\Project;
use App\Models\SupplierProject;
use App\Models\Tenant;
use Illuminate\Foundation\Testing\DatabaseTransactions;
use Illuminate\Support\Facades\DB;
uses(DatabaseTransactions::class);
function linkedProject(bool $active, ?string $pausedAt = null): Project
{
$tenant = Tenant::factory()->create();
$p = Project::factory()->for($tenant)->create(['is_active' => $active, 'paused_at' => $pausedAt]);
$sp = SupplierProject::factory()->create();
DB::table('project_supplier_links')->insert([
'project_id' => $p->id,
'supplier_project_id' => $sp->id,
'platform' => $sp->platform,
'subject_code' => null,
]);
return $p->loadCount('supplierProjects');
}
it('active linked project → source_locked true + projected', function (): void {
$res = (new ProjectResource(linkedProject(true)))->toArray(request());
expect($res)->toHaveKeys(['source_locked', 'source_unlock_at', 'source_unlock_projected']);
expect($res['source_locked'])->toBeTrue();
expect($res['source_unlock_projected'])->toBeTrue();
expect($res['source_unlock_at'])->not->toBeNull();
});
it('project with no supplier links → source_locked false', function (): void {
$tenant = Tenant::factory()->create();
$p = Project::factory()->for($tenant)->create(['is_active' => true])->loadCount('supplierProjects');
$res = (new ProjectResource($p))->toArray(request());
expect($res['source_locked'])->toBeFalse();
expect($res['source_unlock_at'])->toBeNull();
expect($res['source_unlock_projected'])->toBeFalse();
});