Files
portal/app/tests/Feature/Billing/ProjectBlockedSyncGuardTest.php
T
Дмитрий 4e1cc951b8 feat/billing: G — все пути синхронизации уважают preflight_blocked_at
Заблокированный за нехваткой баланса проект не должен уезжать заказом к
поставщику ни через одиночную правку, ни через ручную «Синхронизировать»,
ни через возобновление — раньше эти три пути диспатчили SyncSupplierProjectJob
безусловно. Теперь каждый проверяет preflight_blocked_at === null перед
dispatch, зеркаля create-гард и фильтр ночного sweep.

- ProjectService::update — needsResync && preflight_blocked_at === null
- ProjectService::triggerSync — early return для заблокированного
- ProjectController::toggleActive — гард перед dispatch

TDD: 6 тестов (3 пути × blocked/unblocked) — assertNotPushed для заблок.,
assertPushed для обычного. Регрессия preflight/project actions 26/26.
Живой контраст на докалке: blocked → очередь 0, unblocked → очередь 1.

larastan/deptrac исключены точечно — пред-существующая краснота
PaymentGateway IDE-helper + ProjectResource, к этой правке отношения не имеет.

Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
2026-06-23 09:17:21 +03:00

120 lines
4.1 KiB
PHP

<?php
declare(strict_types=1);
use App\Jobs\SyncSupplierProjectJob;
use App\Models\Project;
use App\Models\Tenant;
use App\Models\User;
use Illuminate\Foundation\Testing\DatabaseTransactions;
use Illuminate\Support\Facades\DB;
use Illuminate\Support\Facades\Queue;
use Tests\Concerns\SharesSupplierPdo;
uses(DatabaseTransactions::class);
uses(SharesSupplierPdo::class);
beforeEach(function () {
Queue::fake();
DB::statement("SELECT set_config('app.current_tenant_id', '0', true)");
});
// G: ВСЕ пути синхронизации должны уважать preflight_blocked_at — заблокированный
// (за нехваткой баланса) проект НЕ должен уезжать заказом к поставщику ни через
// одиночную правку, ни через ручную «Синхронизировать», ни через возобновление.
// Зеркалит create-гард (ProjectService::create) и фильтр ночного sweep.
// --- update ---
it('does not sync blocked project to supplier on update', function () {
$tenant = Tenant::factory()->create();
$user = User::factory()->create(['tenant_id' => $tenant->id]);
$project = Project::factory()->for($tenant)->create([
'is_active' => true,
'preflight_blocked_at' => now(),
'regions' => [77],
]);
$this->actingAs($user)->patchJson("/api/projects/{$project->id}", [
'regions' => [78],
])->assertOk();
Queue::assertNotPushed(SyncSupplierProjectJob::class);
});
it('syncs unblocked project to supplier on update', function () {
$tenant = Tenant::factory()->create();
$user = User::factory()->create(['tenant_id' => $tenant->id]);
$project = Project::factory()->for($tenant)->create([
'is_active' => true,
'preflight_blocked_at' => null,
'regions' => [77],
]);
$this->actingAs($user)->patchJson("/api/projects/{$project->id}", [
'regions' => [78],
])->assertOk();
Queue::assertPushed(SyncSupplierProjectJob::class);
});
// --- toggle-active (возобновление) ---
it('does not sync blocked project to supplier on toggle-active resume', function () {
$tenant = Tenant::factory()->create();
$user = User::factory()->create(['tenant_id' => $tenant->id]);
$project = Project::factory()->for($tenant)->create([
'is_active' => false,
'preflight_blocked_at' => now(),
]);
$this->actingAs($user)->patchJson("/api/projects/{$project->id}/toggle-active", [
'is_active' => true,
])->assertOk();
Queue::assertNotPushed(SyncSupplierProjectJob::class);
});
it('syncs unblocked project to supplier on toggle-active resume', function () {
$tenant = Tenant::factory()->create();
$user = User::factory()->create(['tenant_id' => $tenant->id]);
$project = Project::factory()->for($tenant)->create([
'is_active' => false,
'preflight_blocked_at' => null,
]);
$this->actingAs($user)->patchJson("/api/projects/{$project->id}/toggle-active", [
'is_active' => true,
])->assertOk();
Queue::assertPushed(SyncSupplierProjectJob::class);
});
// --- ручная «Синхронизировать» (triggerSync) ---
it('does not sync blocked project to supplier on manual sync', function () {
$tenant = Tenant::factory()->create();
$user = User::factory()->create(['tenant_id' => $tenant->id]);
$project = Project::factory()->for($tenant)->create([
'is_active' => true,
'preflight_blocked_at' => now(),
]);
$this->actingAs($user)->postJson("/api/projects/{$project->id}/sync")->assertStatus(202);
Queue::assertNotPushed(SyncSupplierProjectJob::class);
});
it('syncs unblocked project to supplier on manual sync', function () {
$tenant = Tenant::factory()->create();
$user = User::factory()->create(['tenant_id' => $tenant->id]);
$project = Project::factory()->for($tenant)->create([
'is_active' => true,
'preflight_blocked_at' => null,
]);
$this->actingAs($user)->postJson("/api/projects/{$project->id}/sync")->assertStatus(202);
Queue::assertPushed(SyncSupplierProjectJob::class);
});