e213f9b01c
deals:backfill-region-city fills deals.city from the lead resolved_subject_code (deals -> supplier_lead_deliveries -> supplier_leads) for deals where city is still empty, idempotently and across all tenants (BYPASSRLS). --dry-run reports the count without writing. Whitelisted in artisan-run.yml (dry-run read-only; real run requires confirm_apply). TDD: +4 tests GREEN. Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
103 lines
3.3 KiB
PHP
103 lines
3.3 KiB
PHP
<?php
|
||
|
||
declare(strict_types=1);
|
||
|
||
use App\Models\Deal;
|
||
use App\Models\Project;
|
||
use App\Models\SupplierLead;
|
||
use App\Models\Tenant;
|
||
use Illuminate\Foundation\Testing\DatabaseTransactions;
|
||
use Illuminate\Support\Facades\DB;
|
||
use Tests\Concerns\SharesSupplierPdo;
|
||
|
||
uses(DatabaseTransactions::class);
|
||
uses(SharesSupplierPdo::class);
|
||
|
||
beforeEach(function (): void {
|
||
DB::statement("SELECT set_config('app.current_tenant_id', '0', true)");
|
||
});
|
||
|
||
/**
|
||
* Сеет сделку (city=NULL по умолчанию) + лид с resolved_subject_code + связь
|
||
* supplier_lead_deliveries. Возвращает [tenantId, dealId].
|
||
*
|
||
* @return array{0: int, 1: int}
|
||
*/
|
||
function seedDealWithResolvedLead(?int $resolvedCode, ?string $city = null): array
|
||
{
|
||
$tenant = Tenant::factory()->create(['balance_rub' => '100000.00']);
|
||
$project = Project::factory()->create([
|
||
'tenant_id' => $tenant->id,
|
||
'signal_type' => 'site',
|
||
'signal_identifier' => 'backfill-city.ru',
|
||
'is_active' => true,
|
||
]);
|
||
|
||
DB::statement("SET LOCAL app.current_tenant_id = '{$tenant->id}'");
|
||
$deal = Deal::create([
|
||
'tenant_id' => $tenant->id,
|
||
'project_id' => $project->id,
|
||
'phone' => '79161234567',
|
||
'phones' => ['79161234567'],
|
||
'status' => 'new',
|
||
'received_at' => now(),
|
||
'subject_code' => $resolvedCode,
|
||
'city' => $city,
|
||
]);
|
||
DB::statement("SELECT set_config('app.current_tenant_id', '0', true)");
|
||
|
||
$lead = SupplierLead::factory()->create([
|
||
'platform' => 'B1',
|
||
'phone' => '79161234567',
|
||
'resolved_subject_code' => $resolvedCode,
|
||
'region_source' => $resolvedCode !== null ? 'dadata' : 'unknown',
|
||
]);
|
||
|
||
DB::connection('pgsql_supplier')->table('supplier_lead_deliveries')->insert([
|
||
'supplier_lead_id' => $lead->id,
|
||
'tenant_id' => $tenant->id,
|
||
'deal_id' => $deal->id,
|
||
'created_at' => now(),
|
||
]);
|
||
|
||
return [$tenant->id, $deal->id];
|
||
}
|
||
|
||
function dealCity(int $dealId): ?string
|
||
{
|
||
// BYPASSRLS чтение (как и сам бэкфилл) — без tenant-контекста.
|
||
return DB::connection('pgsql_supplier')->table('deals')->where('id', $dealId)->value('city');
|
||
}
|
||
|
||
it('backfills deal city from the lead resolved region code', function (): void {
|
||
[, $dealId] = seedDealWithResolvedLead(29); // 29 → Красноярский край
|
||
|
||
$this->artisan('deals:backfill-region-city')->assertSuccessful();
|
||
|
||
expect(dealCity($dealId))->toBe('Красноярский край');
|
||
});
|
||
|
||
it('does not touch deals that already have a city', function (): void {
|
||
[, $dealId] = seedDealWithResolvedLead(29, city: 'Уже стоит');
|
||
|
||
$this->artisan('deals:backfill-region-city')->assertSuccessful();
|
||
|
||
expect(dealCity($dealId))->toBe('Уже стоит');
|
||
});
|
||
|
||
it('dry-run reports candidates without writing', function (): void {
|
||
[, $dealId] = seedDealWithResolvedLead(29);
|
||
|
||
$this->artisan('deals:backfill-region-city', ['--dry-run' => true])->assertSuccessful();
|
||
|
||
expect(dealCity($dealId))->toBeNull();
|
||
});
|
||
|
||
it('leaves city null when the lead has no resolved region', function (): void {
|
||
[, $dealId] = seedDealWithResolvedLead(null);
|
||
|
||
$this->artisan('deals:backfill-region-city')->assertSuccessful();
|
||
|
||
expect(dealCity($dealId))->toBeNull();
|
||
});
|