e8db184e99
LeadRouter SQL переписан на JOIN с project_routing_snapshots по active_slepok_date:
до 21:00 МСК = today, после 21:00 МСК = today+1. is_active / delivery_days_mask /
daily_limit / regions / signal_type / signal_identifier берутся из snapshot.
Из live projects — только delivered_today (счётчик остатка лимита). Из tenants —
balance_rub (live auto-pause при нулевом балансе).
Active snapshot date вычисляется в PHP (метод activeSnapshotDate()) и
передаётся в SQL как параметр — тестируемо через Carbon::setTestNow,
исключает дрейф между PHP- и DB-часами.
Fail-loud: Log::error('lead_router.no_snapshot_for_active_date', ...) если
по активной дате слепка вообще нет ни одной строки snapshot'а (cron не отработал).
Closes R-01, R-04, R-06, R-07, R-08, R-15.
Partial: R-02 (через шеринг), R-09 (race), R-10 (editable identifier) — закрываются Task 2.6+.
Plan: docs/superpowers/plans/2026-05-26-slepok-routing-protection.md §Task 2.5
Spec: docs/superpowers/specs/2026-05-26-slepok-routing-protection-design.md §4.2.3
Tests added:
- tests/Feature/LeadRouter/SnapshotRoutingTest.php (4 tests, all GREEN locally)
Tests patched (downstream — добавлен createRoutingSnapshotFromProject() helper):
- tests/Pest.php — global helper createRoutingSnapshotFromProject()
- tests/Feature/LeadRouter/BalanceFilterTest.php (2/2 GREEN)
- tests/Feature/Services/LeadRouterTest.php (10/10 GREEN)
- tests/Feature/Jobs/RouteSupplierLeadJobTest.php (14/14 GREEN)
- tests/Feature/Supplier/DirectPlatformTest.php (6/6 GREEN)
- tests/Feature/Supplier/RouteSupplierLeadJobBillingTest.php (3/3 GREEN)
- tests/Feature/Supplier/SupplierConnectionTest.php (5/5 GREEN)
- tests/Feature/Integration/SupplierLeadFlowTest.php (2/2 GREEN)
- tests/Feature/Pd/DealCreatePdLogTest.php (2/2 GREEN)
Each test file isolated regression: GREEN. Combined run 49/50 with 1 flake on
quirk #77 (Faker unique domainName + cross-connection pgsql/pgsql_supplier
DatabaseTransactions scope mismatch) — pre-existing, NOT regression от Task 2.5.
Patched via 7 parallel Sonnet subagents per Pravila §15.1; controller-verified
isolated + combined regression (latter caught 1 subagent over-application:
paused project in SupplierLeadFlowTest получил snapshot, что нарушило логику
теста — fixed inline, по semantic match with SnapshotBackfillCommand SQL
WHERE p.is_active = true).
115 lines
4.5 KiB
PHP
115 lines
4.5 KiB
PHP
<?php
|
||
|
||
use App\Models\Project;
|
||
use App\Models\SupplierProject;
|
||
use Illuminate\Foundation\Testing\RefreshDatabase;
|
||
use Illuminate\Support\Facades\DB;
|
||
use Tests\TestCase;
|
||
|
||
/*
|
||
|--------------------------------------------------------------------------
|
||
| Test Case
|
||
|--------------------------------------------------------------------------
|
||
|
|
||
| The closure you provide to your test functions is always bound to a specific PHPUnit test
|
||
| case class. By default, that class is "PHPUnit\Framework\TestCase". Of course, you may
|
||
| need to change it using the "pest()" function to bind different classes or traits.
|
||
|
|
||
*/
|
||
|
||
pest()->extend(TestCase::class)
|
||
// ->use(RefreshDatabase::class)
|
||
->in('Feature');
|
||
|
||
pest()->extend(TestCase::class)->in('Browser');
|
||
|
||
/*
|
||
|--------------------------------------------------------------------------
|
||
| Expectations
|
||
|--------------------------------------------------------------------------
|
||
|
|
||
| When you're writing tests, you often need to check that values meet certain conditions. The
|
||
| "expect()" function gives you access to a set of "expectations" methods that you can use
|
||
| to assert different things. Of course, you may extend the Expectation API at any time.
|
||
|
|
||
*/
|
||
|
||
expect()->extend('toBeOne', function () {
|
||
return $this->toBe(1);
|
||
});
|
||
|
||
/*
|
||
|--------------------------------------------------------------------------
|
||
| Functions
|
||
|--------------------------------------------------------------------------
|
||
|
|
||
| While Pest is very powerful out-of-the-box, you may have some testing code specific to your
|
||
| project that you don't want to repeat in every file. Here you can also expose helpers as
|
||
| global functions to help you to reduce the number of lines of code in your test files.
|
||
|
|
||
*/
|
||
|
||
function something()
|
||
{
|
||
// ..
|
||
}
|
||
|
||
/**
|
||
* Link a Лидерра-project to a supplier_project via the M:N pivot
|
||
* (Plan 1 model). Post-Plan-2 LeadRouter eligibility queries the pivot
|
||
* only; legacy supplier_b{1,2,3}_project_id FK is ignored for routing.
|
||
*
|
||
* Single source — replaces previous duplicated declarations in
|
||
* LeadRouterTest.php / RouteSupplierLeadJobTest.php (Plan 2 cleanup).
|
||
* pivot created_at has DEFAULT NOW(); supplier->subject_code may be null.
|
||
*/
|
||
function linkProjectToSupplier(Project $project, SupplierProject $supplier): void
|
||
{
|
||
DB::table('project_supplier_links')->insert([
|
||
'project_id' => $project->id,
|
||
'supplier_project_id' => $supplier->id,
|
||
'platform' => $supplier->platform,
|
||
// @phpstan-ignore-next-line property.notFound — subject_code is in $fillable/casts, IDE stubs lag
|
||
'subject_code' => $supplier->subject_code,
|
||
]);
|
||
}
|
||
|
||
/**
|
||
* Pest helper для slepok-routing тестов (Task 2.5).
|
||
*
|
||
* Создаёт строку в `project_routing_snapshots` за активную дату слепка,
|
||
* отражающую "идеальное" состояние live-проекта. Используется тестами,
|
||
* которым нужен только факт «маршрутизация возможна», а не сам snapshot
|
||
* mechanism.
|
||
*
|
||
* Активная дата по умолчанию — сегодняшняя МСК (до 21:00 МСК). Передайте
|
||
* `$date` явно, если тест использует `Carbon::setTestNow` с другой датой.
|
||
*
|
||
* NB: signal_type/signal_identifier берутся ЯВНО из аргументов, а не из
|
||
* `$project->signal_type` — на Windows-native PG факториальный override
|
||
* этого поля не персистится (см. memory project_slepok_protection.md).
|
||
*/
|
||
function createRoutingSnapshotFromProject(
|
||
Project $project,
|
||
?string $date = null,
|
||
string $signalType = 'call',
|
||
?string $signalIdentifier = null,
|
||
?int $dailyLimit = null,
|
||
): void {
|
||
DB::table('project_routing_snapshots')->insert([
|
||
'snapshot_date' => $date ?? \Carbon\Carbon::today('Europe/Moscow')->toDateString(),
|
||
'project_id' => $project->id,
|
||
'tenant_id' => $project->tenant_id,
|
||
'daily_limit' => $dailyLimit ?? (int) ($project->effective_daily_limit_today ?? $project->daily_limit_target),
|
||
'delivery_days_mask' => (int) ($project->delivery_days_mask ?? 127),
|
||
'regions' => '{}',
|
||
'signal_type' => $signalType,
|
||
'signal_identifier' => $signalIdentifier,
|
||
'sms_senders' => null,
|
||
'sms_keyword' => null,
|
||
'expected_volume' => $dailyLimit ?? (int) ($project->effective_daily_limit_today ?? $project->daily_limit_target),
|
||
'delivered_count' => 0,
|
||
'created_at' => \Illuminate\Support\Facades\Date::now(),
|
||
]);
|
||
}
|