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:
@@ -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();
|
||||
});
|
||||
Reference in New Issue
Block a user