Files
portal/app/tests/Feature/Autopodbor/AutopodborCompetitorCrudApiTest.php
T
Дмитрий 4387333118 feat(Конкурентное поле): рабочее место конкуренты→источники→проекты (поверх автоподбора)
Фича «Конкурентное поле» на dev до уровня прототипа 2026-06-29-konkurentnoe-pole-proto.html.

Данные: box (proposal|field) на competitors+sources; phone_type city/mobile/tollfree рядом
с phone_kind (вариант C). 3 миграции, дефолты тарифов 300/50.

API (AutopodborController): GET /field (+счётчики), GET /proposals, PATCH/DELETE competitors
и sources с гвардами активного проекта, переключение box, POST /competitors/manual (+directory_urls),
competitor(id) обогащён box+project-статусом; projectStatus отдаёт limit/delivered/days/regions.
Смена источника проекта = PATCH /api/projects/{id} (реальный гвард слепка §14.10).

Фронт: FieldWorkspaceScreen/FieldCompetitorScreen/FieldProposalsScreen/FieldManualCompetitorScreen
+ field-shared.css (Forest) + AutopodborServicesPanel в Биллинге. Дословно по прототипу: подзаголовки,
баннер предложений, баннер правил времени 18:00 МСК, Справочник 2ГИС·Яндекс, статус проекта
5/день·заявки, окна сбора с ценами 300/50 + «что известно», полные формы. Пункт меню «Конкурентное поле».

Тесты: backend автоподбор 80/80, фронт автоподбор 49/49. Движок шага 2 = заглушка FakeCompetitorAgent.
OmegaDemoFieldSeeder — только для визуальной проверки (НЕ на прод).

Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
2026-06-30 04:18:46 +03:00

96 lines
3.9 KiB
PHP

<?php
declare(strict_types=1);
use App\Models\AutopodborCompetitor;
use App\Models\AutopodborRun;
use App\Models\AutopodborSource;
use App\Models\Project;
use App\Models\Tenant;
use App\Models\User;
use Illuminate\Foundation\Testing\DatabaseTransactions;
use Illuminate\Support\Facades\DB;
use Tests\Concerns\SharesSupplierPdo;
uses(DatabaseTransactions::class, SharesSupplierPdo::class);
/** @return array{0: Tenant, 1: User, 2: AutopodborRun, 3: AutopodborCompetitor} */
function compCrudSetup(): array
{
$tenant = Tenant::factory()->create();
$user = User::factory()->create(['tenant_id' => $tenant->id]);
DB::statement('SET app.current_tenant_id = '.$tenant->id);
$run = AutopodborRun::create(['tenant_id' => $tenant->id, 'kind' => 'search', 'status' => 'done', 'region_code' => 16, 'params' => []]);
$comp = AutopodborCompetitor::create([
'tenant_id' => $tenant->id, 'search_run_id' => $run->id,
'name' => 'Старое имя', 'dedup_key' => 'old', 'relevance_pct' => 50, 'box' => 'proposal',
]);
return [$tenant, $user, $run, $comp];
}
it('PATCH competitors/{id} — правит поля карточки', function () {
[$tenant, $user, $run, $comp] = compCrudSetup();
$this->actingAs($user)->patchJson("/api/autopodbor/competitors/{$comp->id}", [
'name' => 'Окна Премиум',
'description' => 'Премиальные окна',
'is_federal' => true,
'relevance_pct' => 88,
'site_url' => 'okna-premium.ru',
])->assertOk()
->assertJsonPath('data.name', 'Окна Премиум')
->assertJsonPath('data.relevance_pct', 88)
->assertJsonPath('data.is_federal', true);
$fresh = $comp->fresh();
expect($fresh->name)->toBe('Окна Премиум')
->and($fresh->site_url)->toBe('okna-premium.ru');
});
it('PATCH competitors/{id} — отвергает похожесть вне 0..100 (422)', function () {
[$tenant, $user, $run, $comp] = compCrudSetup();
$this->actingAs($user)->patchJson("/api/autopodbor/competitors/{$comp->id}", [
'relevance_pct' => 150,
])->assertStatus(422);
});
it('PATCH competitors/{id} — чужой тенант не правит (404)', function () {
[$tenant, $user, $run, $comp] = compCrudSetup();
$other = User::factory()->create(['tenant_id' => Tenant::factory()->create()->id]);
$this->actingAs($other)->patchJson("/api/autopodbor/competitors/{$comp->id}", ['name' => 'Взлом'])
->assertStatus(404);
});
it('DELETE competitors/{id} — удаляет конкурента и его источники', function () {
[$tenant, $user, $run, $comp] = compCrudSetup();
$src = AutopodborSource::create([
'tenant_id' => $tenant->id, 'competitor_id' => $comp->id, 'study_run_id' => $run->id,
'signal_type' => 'site', 'identifier' => 'okna.ru', 'dedup_key' => 'site:okna.ru',
]);
$this->actingAs($user)->deleteJson("/api/autopodbor/competitors/{$comp->id}")
->assertStatus(204);
expect(AutopodborCompetitor::find($comp->id))->toBeNull()
->and(AutopodborSource::find($src->id))->toBeNull();
});
it('DELETE competitors/{id} — блок, если у источника активный проект (409)', function () {
[$tenant, $user, $run, $comp] = compCrudSetup();
$project = Project::factory()->create(['tenant_id' => $tenant->id, 'is_active' => true, 'preflight_blocked_at' => null]);
AutopodborSource::create([
'tenant_id' => $tenant->id, 'competitor_id' => $comp->id, 'study_run_id' => $run->id,
'signal_type' => 'site', 'identifier' => 'okna.ru', 'dedup_key' => 'site:okna.ru',
'created_project_id' => $project->id,
]);
$this->actingAs($user)->deleteJson("/api/autopodbor/competitors/{$comp->id}")
->assertStatus(409)
->assertJsonPath('error', 'has_active_projects');
expect(AutopodborCompetitor::find($comp->id))->not->toBeNull();
});