Files
portal/app/tests/Feature/LeadRouter/SnapshotRoutingTest.php
T
Дмитрий 2ec70b338f
Accessibility (Pa11y live) / a11y (push) Has been cancelled
test: оздоровление тест-стенда — изоляция протекателей плюс фикстуры, партиции, видимость supplier-коннекта
Закрыто 36 из 55 пре-существующих падений backend-набора (55 to 19), всё тест-сторона,
код продукта не тронут. Группы:
- incident-показ/РКН: добавлен SharesSupplierPdo + синхрон уровня транзакции в трейте
  (вложенный transaction на общем PDO теперь делает SAVEPOINT, не повторный BEGIN).
- auto-pause и lead-delivery: тесты создают project_routing_snapshots, от которого
  зависит выбор кандидатов в LeadRouter (slepok-инвариант).
- изоляция 16 протекающих тестов: добавлен DatabaseTransactions (где нужно плюс
  SharesSupplierPdo) — перестали оставлять committed-строки, отравлявшие глобально
  сканирующие тесты (snapshot, verify-audit, size-N).
- partition time-bombs: ensureRange месячных партиций для тестов на дату 2026-05.
- устаревшие ассерты: SchemaDelta метрики v8.35 to v8.52, ProjectsStore телефон 8 to 7
  нормализуется, incidents-watch фильтр активного admin, register captcha_token,
  impersonation активный юзер тенанта, activity_log.deal_id, ProjectUpdateDedup пауза.

Остаток 19 (отдельно): verify-audit-chains и size-N (протекатели audit-строк),
webhook B-префикс (решение владельца), пара env/каскадных.

Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
2026-06-25 07:39:51 +03:00

118 lines
5.0 KiB
PHP

<?php
declare(strict_types=1);
use App\Models\Project;
use App\Models\SupplierProject;
use App\Models\Tenant;
use App\Services\LeadRouter;
use Carbon\Carbon;
uses(\Illuminate\Foundation\Testing\DatabaseTransactions::class);
uses(\Tests\Concerns\SharesSupplierPdo::class);
it('uses snapshot before 21:00 MSK, snapshot_date = today', function () {
Carbon::setTestNow('2026-05-28 12:00:00', 'Europe/Moscow');
$tenant = Tenant::factory()->create(['balance_rub' => '500.00', 'frozen_by_balance_at' => null]);
$project = Project::factory()->for($tenant)->create([
'is_active' => false, // ЖИВОЕ состояние — paused
'delivery_days_mask' => 127,
'daily_limit_target' => 100,
'delivered_today' => 0,
]);
$sp = SupplierProject::factory()->create();
DB::table('project_supplier_links')->insert([
'project_id' => $project->id, 'supplier_project_id' => $sp->id,
'platform' => $sp->platform, 'subject_code' => null,
]);
// SNAPSHOT за сегодня имеет проект → роутер должен вернуть, несмотря на is_active=false
DB::table('project_routing_snapshots')->insert([
'snapshot_date' => '2026-05-28', 'project_id' => $project->id, 'tenant_id' => $tenant->id,
'daily_limit' => 10, 'delivery_days_mask' => 127, 'regions' => '{}',
'signal_type' => 'call', 'expected_volume' => 10, 'delivered_count' => 0,
'created_at' => now(),
]);
$matched = app(LeadRouter::class)->matchEligibleProjects($sp);
expect($matched)->toHaveCount(1); // ← это R-01 closure
Carbon::setTestNow();
});
it('uses snapshot after 21:00 MSK, snapshot_date = tomorrow', function () {
Carbon::setTestNow('2026-05-28 22:00:00', 'Europe/Moscow');
$tenant = Tenant::factory()->create(['balance_rub' => '500.00', 'frozen_by_balance_at' => null]);
$project = Project::factory()->for($tenant)->create([
'is_active' => true, 'delivery_days_mask' => 127,
'daily_limit_target' => 100, 'delivered_today' => 0,
]);
$sp = SupplierProject::factory()->create();
DB::table('project_supplier_links')->insert([
'project_id' => $project->id, 'supplier_project_id' => $sp->id,
'platform' => $sp->platform, 'subject_code' => null,
]);
// Snapshot за СЕГОДНЯ (2026-05-28) НЕТ.
// Snapshot за ЗАВТРА (2026-05-29) есть.
DB::table('project_routing_snapshots')->insert([
'snapshot_date' => '2026-05-29', 'project_id' => $project->id, 'tenant_id' => $tenant->id,
'daily_limit' => 10, 'delivery_days_mask' => 127, 'regions' => '{}',
'signal_type' => 'call', 'expected_volume' => 10, 'delivered_count' => 0,
'created_at' => now(),
]);
$matched = app(LeadRouter::class)->matchEligibleProjects($sp);
expect($matched)->toHaveCount(1); // после 21:00 МСК активен завтрашний snapshot
Carbon::setTestNow();
});
it('returns 0 if no snapshot exists for active date', function () {
Carbon::setTestNow('2026-05-28 12:00:00', 'Europe/Moscow');
$tenant = Tenant::factory()->create(['balance_rub' => '500.00']);
$project = Project::factory()->for($tenant)->create([
'is_active' => true, 'delivery_days_mask' => 127, 'daily_limit_target' => 10,
]);
$sp = SupplierProject::factory()->create();
DB::table('project_supplier_links')->insert([
'project_id' => $project->id, 'supplier_project_id' => $sp->id,
'platform' => $sp->platform, 'subject_code' => null,
]);
// НЕТ snapshot за 2026-05-28.
$matched = app(LeadRouter::class)->matchEligibleProjects($sp);
expect($matched)->toHaveCount(0); // fail-loud, не fallback на live
Carbon::setTestNow();
});
it('limit comes from snapshot, not live projects.daily_limit_target', function () {
Carbon::setTestNow('2026-05-28 12:00:00', 'Europe/Moscow');
$tenant = Tenant::factory()->create(['balance_rub' => '500.00']);
$project = Project::factory()->for($tenant)->create([
'is_active' => true, 'delivery_days_mask' => 127,
'daily_limit_target' => 100, // живой лимит
'delivered_today' => 7,
]);
$sp = SupplierProject::factory()->create();
DB::table('project_supplier_links')->insert([
'project_id' => $project->id, 'supplier_project_id' => $sp->id,
'platform' => $sp->platform, 'subject_code' => null,
]);
DB::table('project_routing_snapshots')->insert([
'snapshot_date' => '2026-05-28', 'project_id' => $project->id, 'tenant_id' => $tenant->id,
'daily_limit' => 5, // ← snapshot лимит МЕНЬШЕ чем delivered_today=7
'delivery_days_mask' => 127, 'regions' => '{}',
'signal_type' => 'call', 'expected_volume' => 5, 'delivered_count' => 0,
'created_at' => now(),
]);
$matched = app(LeadRouter::class)->matchEligibleProjects($sp);
expect($matched)->toHaveCount(0); // delivered_today=7 >= snapshot.daily_limit=5
Carbon::setTestNow();
});