4387333118
Фича «Конкурентное поле» на 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>
56 lines
2.3 KiB
PHP
56 lines
2.3 KiB
PHP
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
use App\Models\AutopodborCompetitor;
|
|
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);
|
|
|
|
it('POST competitors/manual — заводит конкурента сразу в поле без изучения', function () {
|
|
$tenant = Tenant::factory()->create();
|
|
$user = User::factory()->create(['tenant_id' => $tenant->id]);
|
|
DB::statement('SET app.current_tenant_id = '.$tenant->id);
|
|
|
|
$this->actingAs($user)->postJson('/api/autopodbor/competitors/manual', [
|
|
'name' => 'Окна Ромашка',
|
|
'site_url' => 'romashka.ru',
|
|
'description' => 'Местный конкурент',
|
|
])->assertCreated()
|
|
->assertJsonPath('data.name', 'Окна Ромашка')
|
|
->assertJsonPath('data.box', 'field')
|
|
->assertJsonPath('data.origin', 'manual');
|
|
|
|
$comp = AutopodborCompetitor::where('tenant_id', $tenant->id)->where('name', 'Окна Ромашка')->first();
|
|
expect($comp)->not->toBeNull()
|
|
->and($comp->box)->toBe('field')
|
|
->and($comp->origin)->toBe('manual')
|
|
->and($comp->search_run_id)->toBeNull()
|
|
->and($comp->study_run_id)->toBeNull(); // изучение НЕ запускалось
|
|
});
|
|
|
|
it('POST competitors/manual — имя обязательно (422)', function () {
|
|
$tenant = Tenant::factory()->create();
|
|
$user = User::factory()->create(['tenant_id' => $tenant->id]);
|
|
DB::statement('SET app.current_tenant_id = '.$tenant->id);
|
|
|
|
$this->actingAs($user)->postJson('/api/autopodbor/competitors/manual', [
|
|
'site_url' => 'romashka.ru',
|
|
])->assertStatus(422);
|
|
});
|
|
|
|
it('POST competitors/manual — конкурент привязан к своему тенанту', function () {
|
|
$tenant = Tenant::factory()->create();
|
|
$user = User::factory()->create(['tenant_id' => $tenant->id]);
|
|
DB::statement('SET app.current_tenant_id = '.$tenant->id);
|
|
|
|
$this->actingAs($user)->postJson('/api/autopodbor/competitors/manual', ['name' => 'Берёзка'])
|
|
->assertCreated();
|
|
|
|
expect(AutopodborCompetitor::where('name', 'Берёзка')->first()->tenant_id)->toBe($tenant->id);
|
|
});
|