2026-05-28 06:49:43 +03:00
|
|
|
<?php
|
|
|
|
|
|
|
|
|
|
declare(strict_types=1);
|
|
|
|
|
|
|
|
|
|
use App\Models\Project;
|
|
|
|
|
use App\Models\SupplierProject;
|
|
|
|
|
use App\Models\Tenant;
|
2026-06-25 07:39:51 +03:00
|
|
|
use App\Services\MonthlyPartitionManager;
|
2026-05-28 06:49:43 +03:00
|
|
|
use App\Services\Project\ProjectService;
|
|
|
|
|
use Carbon\Carbon;
|
|
|
|
|
use Carbon\CarbonImmutable;
|
|
|
|
|
use Illuminate\Foundation\Testing\DatabaseTransactions;
|
2026-06-25 07:39:51 +03:00
|
|
|
use Tests\Concerns\SharesSupplierPdo;
|
2026-05-28 06:49:43 +03:00
|
|
|
|
2026-06-25 07:39:51 +03:00
|
|
|
// ProjectService::update() пишет в tenant_operations_log (partition by created_at)
|
|
|
|
|
// через pgsql_supplier. Тесты фиксируют Carbon на 2026-05 → партиция мая нужна;
|
|
|
|
|
// SharesSupplierPdo держит DDL-коннект в той же откатываемой транзакции.
|
|
|
|
|
uses(DatabaseTransactions::class, SharesSupplierPdo::class);
|
|
|
|
|
|
|
|
|
|
beforeEach(function () {
|
|
|
|
|
app(MonthlyPartitionManager::class)->ensureRange(
|
|
|
|
|
'tenant_operations_log',
|
|
|
|
|
Carbon::parse('2026-05-01'),
|
|
|
|
|
Carbon::parse('2026-05-31'),
|
|
|
|
|
);
|
|
|
|
|
});
|
2026-05-28 06:49:43 +03:00
|
|
|
|
|
|
|
|
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();
|
|
|
|
|
});
|