Files
portal/app/tests/Unit/Services/Project/PausedAtWriteSideTest.php
T

64 lines
2.7 KiB
PHP
Raw Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
<?php
declare(strict_types=1);
namespace Tests\Unit\Services\Project;
use App\Http\Controllers\Api\ProjectController;
use App\Services\Project\ProjectService;
use ReflectionMethod;
use Tests\TestCase;
/**
* Гарантирует, что переключение `is_active` всегда сопровождается записью `paused_at`:
* - is_active = false → paused_at := NOW()
* - is_active = true → paused_at := null
*
* Без этого SupplierSnapshotGuard для bulk-paused проектов начнёт считать их
* "защищёнными навсегда" (paused_at NULL trait), и удаление никогда не разблокируется.
*
* Тест читает исходник методов и проверяет наличие явной записи `paused_at` рядом
* с записью `is_active`. Это структурный smoke — поведенческие тесты (через БД)
* пишутся отдельно (Task 14 final regression).
*
* Spec: docs/superpowers/plans/2026-05-26-supplier-snapshot-guard.md (Task 11).
*/
class PausedAtWriteSideTest extends TestCase
{
public function test_project_service_bulk_pause_resume_writes_paused_at(): void
{
$body = $this->methodBody(ProjectService::class, 'bulkPauseResume');
$this->assertStringContainsString('paused_at', $body,
'bulkPauseResume должен явно обновлять paused_at вместе с is_active');
$this->assertStringContainsString('is_active', $body);
}
public function test_project_controller_toggle_active_writes_paused_at(): void
{
$body = $this->methodBody(ProjectController::class, 'toggleActive');
$this->assertStringContainsString('paused_at', $body,
'toggleActive должен явно обновлять paused_at вместе с is_active');
$this->assertStringContainsString('is_active', $body);
}
public function test_bulk_delete_distinguishes_supplier_snapshot_lock_from_has_deals(): void
{
$body = $this->methodBody(ProjectService::class, 'bulkDelete');
$this->assertStringContainsString('supplier_snapshot_locked', $body,
'bulkDelete должен помечать пропущенные проекты reason="supplier_snapshot_locked" при guard-блоке');
$this->assertStringContainsString('has_deals', $body);
}
private function methodBody(string $class, string $method): string
{
$rm = new ReflectionMethod($class, $method);
$lines = file($rm->getFileName());
$body = array_slice($lines, $rm->getStartLine() - 1, $rm->getEndLine() - $rm->getStartLine() + 1);
return implode('', $body);
}
}