Files
portal/app/tests/Feature/Services/Project/ProjectServiceAppliesFromTest.php
T
Дмитрий 19644a1d36 feat(slepok): Task 2.8 — ProjectService exposes applies_from after slepok-sensitive update
ProjectService::update() теперь возвращает Project с dynamic applies_from
attribute (CarbonImmutable | null), который ProjectResource подхватит для UI
(«изменения вступят в силу с DD.MM 21:00»).

Логика: для каждого изменённого поля из SupplierSnapshotGuard::SLEPOK_SENSITIVE_FIELDS
вычисляется максимум appliesFrom() — slepok-инвариант (до 18:00 МСК = today 21:00,
после = tomorrow 21:00). NULL = применяется немедленно (none changed / no supplier links).

Plan: docs/superpowers/plans/2026-05-26-slepok-routing-protection.md §Task 2.8
Spec: docs/superpowers/specs/2026-05-26-slepok-routing-protection-design.md §4.2.5

Tests: tests/Feature/Services/Project/ProjectServiceAppliesFromTest.php — 4/4 PASS.
ProjectService regression — 7/7 PASS.
2026-05-28 06:49:43 +03:00

74 lines
2.5 KiB
PHP

<?php
declare(strict_types=1);
use App\Models\Project;
use App\Models\SupplierProject;
use App\Models\Tenant;
use App\Services\Project\ProjectService;
use Carbon\Carbon;
use Carbon\CarbonImmutable;
use Illuminate\Foundation\Testing\DatabaseTransactions;
uses(DatabaseTransactions::class);
it('returns applies_from when changing daily_limit_target before 18:00 MSK', function (): void {
Carbon::setTestNow('2026-05-28 14:00:00', 'Europe/Moscow');
$tenant = Tenant::factory()->create();
$project = Project::factory()->for($tenant)->create(['daily_limit_target' => 10]);
$sp = SupplierProject::factory()->create();
linkProjectToSupplier($project, $sp);
$result = app(ProjectService::class)->update($project, ['daily_limit_target' => 5]);
expect($result->applies_from)->toBeInstanceOf(CarbonImmutable::class);
expect($result->applies_from->format('Y-m-d H:i'))->toBe('2026-05-28 21:00');
Carbon::setTestNow();
});
it('returns applies_from = tomorrow 21:00 MSK when edit after 18:00 MSK', function (): void {
Carbon::setTestNow('2026-05-28 19:30:00', 'Europe/Moscow');
$tenant = Tenant::factory()->create();
$project = Project::factory()->for($tenant)->create(['daily_limit_target' => 10]);
$sp = SupplierProject::factory()->create();
linkProjectToSupplier($project, $sp);
$result = app(ProjectService::class)->update($project, ['daily_limit_target' => 7]);
expect($result->applies_from->format('Y-m-d H:i'))->toBe('2026-05-29 21:00');
Carbon::setTestNow();
});
it('returns applies_from = null when only non-slepok fields changed (e.g. name)', function (): void {
Carbon::setTestNow('2026-05-28 14:00:00', 'Europe/Moscow');
$tenant = Tenant::factory()->create();
$project = Project::factory()->for($tenant)->create();
$sp = SupplierProject::factory()->create();
linkProjectToSupplier($project, $sp);
$result = app(ProjectService::class)->update($project, ['name' => 'Renamed project']);
expect($result->applies_from)->toBeNull();
Carbon::setTestNow();
});
it('returns applies_from = null when project has no supplier links', function (): void {
Carbon::setTestNow('2026-05-28 14:00:00', 'Europe/Moscow');
$tenant = Tenant::factory()->create();
$project = Project::factory()->for($tenant)->create(['daily_limit_target' => 10]);
// нет linkProjectToSupplier — нет slepok-риска
$result = app(ProjectService::class)->update($project, ['daily_limit_target' => 5]);
expect($result->applies_from)->toBeNull();
Carbon::setTestNow();
});