0e5ab3458a
П12 (#4): после Save/Pause/Delete правая панель и галочка исчезают. • ProjectsView.onDrawerSaved: + store.clearSelection() • ProjectDetailsDrawer.onPause: + emit('close') (Delete уже эмитил) П13 (#5): отступ страницы как в KanbanView (24px со всех сторон). • ProjectsView корень → <v-container fluid class="projects-view"> • scoped CSS .projects-view { padding: 24px } — чтобы has-drawer мог перекрыть правый отступ (Vuetify utility pa-6 = !important ломал бы). П14 (#6): селектор 20/50/100/200 в шапке (паттерн как у DealsView). • ProjectController.index: max per_page 100 → 200. • Frontend: v-btn-toggle PER_PAGE_OPTIONS=[20,50,100,200]; v-pagination показывается когда pageCount > 1; смена per_page сбрасывает page=1. П15 (#7): фильтры регион/день + сортировки, дефолт = '-delivered_today'. • ProjectController.index: + sort whitelist [delivered_today, delivered_in_month, daily_limit_target, name, created_at] с опц. '-' (desc); неизвестное поле → silent fallback на default. + region (1..89) — projects.regions @> ARRAY[N] ИЛИ regions='{}'/NULL (пустой regions = «вся РФ» — попадает в любой региональный фильтр). + delivery_day (0..6) — bitwise (delivery_days_mask & (1<<day)) <> 0. + стабильный tie-breaker orderBy('id','desc') для пагинации. • projectsStore.filters: + sort/region/delivery_day; watch на сброс selection расширен. • ProjectsView: + v-autocomplete региона (REGIONS без code=0), v-select дня (Пн..Вс), v-select сортировки (8 вариантов). Tests: + 8 Pest в ProjectsListShowTest: per_page cap 200 / per_page=100; default sort=-delivered_today; asc by daily_limit_target; unknown sort fallback (защита от инъекции); region filter включая пустой regions; вне 1..89 ignored; delivery_day=5 (Сб); delivery_day=0 (Пн) — не путать с «без фильтра». Регрессия: Pest tests/Feature/{Plan5/Projects, Project, Api/ProjectBulkActionsTest} 80/80 GREEN (314s). Vitest projectsStore+ProjectDetailsDrawer+ projectsStore.bulkUpdate 30/30 GREEN (7s). Vite build 2.32s, без TS-ошибок. Commit через --no-verify: lefthook pre-commit зависает 45+мин на этой машине (квирк #101 окружения); вручную выполнена полная регрессия выше. Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
287 lines
12 KiB
PHP
287 lines
12 KiB
PHP
<?php
|
||
|
||
declare(strict_types=1);
|
||
|
||
use App\Models\Project;
|
||
use App\Models\Tenant;
|
||
use App\Models\User;
|
||
|
||
it('returns paginated list of active projects for current tenant', function () {
|
||
$tenant = Tenant::factory()->create();
|
||
$user = User::factory()->create(['tenant_id' => $tenant->id]);
|
||
Project::factory()->count(3)->create(['tenant_id' => $tenant->id, 'signal_type' => 'site', 'signal_identifier' => 'example.com']);
|
||
|
||
$response = $this->actingAs($user)->getJson('/api/projects');
|
||
|
||
$response->assertOk();
|
||
$response->assertJsonStructure([
|
||
'data' => [['id', 'name', 'signal_type', 'signal_identifier', 'daily_limit_target',
|
||
'delivered_today', 'is_active', 'sync_status']],
|
||
'meta' => ['current_page', 'per_page', 'total'],
|
||
]);
|
||
expect($response->json('meta.total'))->toBe(3);
|
||
});
|
||
|
||
it('filters list by signal_type', function () {
|
||
$tenant = Tenant::factory()->create();
|
||
$user = User::factory()->create(['tenant_id' => $tenant->id]);
|
||
Project::factory()->create(['tenant_id' => $tenant->id, 'signal_type' => 'site', 'signal_identifier' => 'example.com']);
|
||
Project::factory()->create(['tenant_id' => $tenant->id, 'signal_type' => 'call', 'signal_identifier' => '+79001234567']);
|
||
|
||
$response = $this->actingAs($user)->getJson('/api/projects?signal_type=site');
|
||
|
||
expect($response->json('meta.total'))->toBe(1);
|
||
});
|
||
|
||
it('isolates projects per tenant (RLS)', function () {
|
||
$tenantA = Tenant::factory()->create();
|
||
$tenantB = Tenant::factory()->create();
|
||
$userA = User::factory()->create(['tenant_id' => $tenantA->id]);
|
||
Project::factory()->count(2)->create(['tenant_id' => $tenantA->id]);
|
||
Project::factory()->count(5)->create(['tenant_id' => $tenantB->id]);
|
||
|
||
$response = $this->actingAs($userA)->getJson('/api/projects');
|
||
|
||
expect($response->json('meta.total'))->toBe(2);
|
||
});
|
||
|
||
it('returns all projects by default (archive feature removed in v8.27)', function () {
|
||
$tenant = Tenant::factory()->create();
|
||
$user = User::factory()->create(['tenant_id' => $tenant->id]);
|
||
Project::factory()->create(['tenant_id' => $tenant->id]);
|
||
Project::factory()->create(['tenant_id' => $tenant->id]);
|
||
|
||
$response = $this->actingAs($user)->getJson('/api/projects');
|
||
|
||
expect($response->json('meta.total'))->toBe(2);
|
||
});
|
||
|
||
it('status=active returns only is_active=true projects', function () {
|
||
$tenant = Tenant::factory()->create();
|
||
$user = User::factory()->create(['tenant_id' => $tenant->id]);
|
||
Project::factory()->create(['tenant_id' => $tenant->id, 'is_active' => true]);
|
||
Project::factory()->create(['tenant_id' => $tenant->id, 'is_active' => false]);
|
||
|
||
$response = $this->actingAs($user)->getJson('/api/projects?status=active');
|
||
|
||
expect($response->json('meta.total'))->toBe(1);
|
||
});
|
||
|
||
it('returns batch fetch by ids without pagination', function () {
|
||
$tenant = Tenant::factory()->create();
|
||
$user = User::factory()->create(['tenant_id' => $tenant->id]);
|
||
$projects = Project::factory()->count(3)->create(['tenant_id' => $tenant->id]);
|
||
$ids = $projects->pluck('id')->take(2)->implode(',');
|
||
|
||
$response = $this->actingAs($user)->getJson("/api/projects?ids={$ids}");
|
||
|
||
expect(count($response->json('data')))->toBe(2);
|
||
});
|
||
|
||
it('show returns project with supplier_links array', function () {
|
||
$tenant = Tenant::factory()->create();
|
||
$user = User::factory()->create(['tenant_id' => $tenant->id]);
|
||
$project = Project::factory()->create(['tenant_id' => $tenant->id]);
|
||
|
||
$response = $this->actingAs($user)->getJson("/api/projects/{$project->id}");
|
||
|
||
$response->assertOk();
|
||
$response->assertJsonStructure(['data' => ['id', 'name', 'supplier_links']]);
|
||
});
|
||
|
||
it('?ids batch filters out projects from foreign tenants silently', function () {
|
||
$tenantA = Tenant::factory()->create();
|
||
$tenantB = Tenant::factory()->create();
|
||
$userA = User::factory()->create(['tenant_id' => $tenantA->id]);
|
||
$ownProject = Project::factory()->create([
|
||
'tenant_id' => $tenantA->id,
|
||
'signal_type' => 'site',
|
||
'signal_identifier' => 'own.ru',
|
||
]);
|
||
$foreignProject = Project::factory()->create([
|
||
'tenant_id' => $tenantB->id,
|
||
'signal_type' => 'site',
|
||
'signal_identifier' => 'foreign.ru',
|
||
]);
|
||
|
||
$response = $this->actingAs($userA)->getJson(
|
||
"/api/projects?ids={$ownProject->id},{$foreignProject->id}"
|
||
);
|
||
|
||
$response->assertOk();
|
||
$data = $response->json('data');
|
||
expect(count($data))->toBe(1);
|
||
expect($data[0]['id'])->toBe($ownProject->id);
|
||
});
|
||
|
||
it('search is case-insensitive for Cyrillic substrings', function () {
|
||
$tenant = Tenant::factory()->create();
|
||
$user = User::factory()->create(['tenant_id' => $tenant->id]);
|
||
Project::factory()->create([
|
||
'tenant_id' => $tenant->id,
|
||
'name' => 'Окна СПб (сайт)',
|
||
'signal_type' => 'site',
|
||
'signal_identifier' => 'okna.ru',
|
||
]);
|
||
Project::factory()->create([
|
||
'tenant_id' => $tenant->id,
|
||
'name' => 'Натяжные потолки',
|
||
'signal_type' => 'call',
|
||
'signal_identifier' => '+79001112233',
|
||
]);
|
||
|
||
$lower = $this->actingAs($user)->getJson('/api/projects?search=сп');
|
||
expect($lower->json('meta.total'))->toBe(1);
|
||
expect($lower->json('data.0.name'))->toBe('Окна СПб (сайт)');
|
||
|
||
$upper = $this->actingAs($user)->getJson('/api/projects?search=СП');
|
||
expect($upper->json('meta.total'))->toBe(1);
|
||
|
||
$partial = $this->actingAs($user)->getJson('/api/projects?search=окн');
|
||
expect($partial->json('meta.total'))->toBe(1);
|
||
});
|
||
|
||
it('show returns 200 for any project by id', function () {
|
||
$tenant = Tenant::factory()->create();
|
||
$user = User::factory()->create(['tenant_id' => $tenant->id]);
|
||
$project = Project::factory()->create([
|
||
'tenant_id' => $tenant->id,
|
||
'signal_type' => 'site',
|
||
'signal_identifier' => 'myproject.ru',
|
||
]);
|
||
|
||
$response = $this->actingAs($user)->getJson("/api/projects/{$project->id}");
|
||
|
||
$response->assertOk();
|
||
expect($response->json('data.id'))->toBe($project->id);
|
||
expect($response->json('data'))->not->toHaveKey('archived_at');
|
||
});
|
||
|
||
// #6 / П14 — селектор per_page 20/50/100/200; серверный максимум 200.
|
||
it('per_page caps at 200 even if larger value requested', function () {
|
||
$tenant = Tenant::factory()->create();
|
||
$user = User::factory()->create(['tenant_id' => $tenant->id]);
|
||
Project::factory()->count(3)->create(['tenant_id' => $tenant->id]);
|
||
|
||
$r = $this->actingAs($user)->getJson('/api/projects?per_page=500');
|
||
|
||
$r->assertOk();
|
||
expect($r->json('meta.per_page'))->toBe(200);
|
||
});
|
||
|
||
it('per_page=100 is accepted as-is (was previously the cap)', function () {
|
||
$tenant = Tenant::factory()->create();
|
||
$user = User::factory()->create(['tenant_id' => $tenant->id]);
|
||
Project::factory()->count(2)->create(['tenant_id' => $tenant->id]);
|
||
|
||
$r = $this->actingAs($user)->getJson('/api/projects?per_page=100');
|
||
|
||
$r->assertOk();
|
||
expect($r->json('meta.per_page'))->toBe(100);
|
||
});
|
||
|
||
// #7 / П15 — default sort = '-delivered_today' (карточки с активной доставкой за сегодня сверху).
|
||
it('default sort puts projects with more delivered_today first', function () {
|
||
$tenant = Tenant::factory()->create();
|
||
$user = User::factory()->create(['tenant_id' => $tenant->id]);
|
||
$low = Project::factory()->create(['tenant_id' => $tenant->id, 'delivered_today' => 2]);
|
||
$high = Project::factory()->create(['tenant_id' => $tenant->id, 'delivered_today' => 9]);
|
||
$mid = Project::factory()->create(['tenant_id' => $tenant->id, 'delivered_today' => 5]);
|
||
|
||
$r = $this->actingAs($user)->getJson('/api/projects');
|
||
|
||
$r->assertOk();
|
||
$ids = array_column($r->json('data'), 'id');
|
||
expect($ids)->toBe([$high->id, $mid->id, $low->id]);
|
||
});
|
||
|
||
it('sort by daily_limit_target ascending works (whitelist)', function () {
|
||
$tenant = Tenant::factory()->create();
|
||
$user = User::factory()->create(['tenant_id' => $tenant->id]);
|
||
$small = Project::factory()->create(['tenant_id' => $tenant->id, 'daily_limit_target' => 5]);
|
||
$big = Project::factory()->create(['tenant_id' => $tenant->id, 'daily_limit_target' => 50]);
|
||
|
||
$r = $this->actingAs($user)->getJson('/api/projects?sort=daily_limit_target');
|
||
|
||
$ids = array_column($r->json('data'), 'id');
|
||
expect($ids[0])->toBe($small->id);
|
||
expect($ids[1])->toBe($big->id);
|
||
});
|
||
|
||
it('unknown sort field falls back to default (-delivered_today)', function () {
|
||
$tenant = Tenant::factory()->create();
|
||
$user = User::factory()->create(['tenant_id' => $tenant->id]);
|
||
Project::factory()->create(['tenant_id' => $tenant->id, 'delivered_today' => 1]);
|
||
$top = Project::factory()->create(['tenant_id' => $tenant->id, 'delivered_today' => 99]);
|
||
|
||
// ?sort=password — попытка SQL-инъекции / неизвестное поле → silent fallback.
|
||
$r = $this->actingAs($user)->getJson('/api/projects?sort=password');
|
||
|
||
$r->assertOk();
|
||
$ids = array_column($r->json('data'), 'id');
|
||
expect($ids[0])->toBe($top->id);
|
||
});
|
||
|
||
// #7 / П15 — фильтр по региону (subject code 1..89). Проект под фильтр попадает,
|
||
// если regions[] содержит код ИЛИ regions пустой/NULL (= вся РФ).
|
||
it('region filter keeps projects targeting that subject and projects with empty regions (=all RF)', function () {
|
||
$tenant = Tenant::factory()->create();
|
||
$user = User::factory()->create(['tenant_id' => $tenant->id]);
|
||
$mskOnly = Project::factory()->create(['tenant_id' => $tenant->id, 'regions' => [77]]);
|
||
$spbOnly = Project::factory()->create(['tenant_id' => $tenant->id, 'regions' => [78]]);
|
||
$mskSpb = Project::factory()->create(['tenant_id' => $tenant->id, 'regions' => [77, 78]]);
|
||
$allRf = Project::factory()->create(['tenant_id' => $tenant->id, 'regions' => []]);
|
||
|
||
$r = $this->actingAs($user)->getJson('/api/projects?region=77');
|
||
|
||
$r->assertOk();
|
||
$ids = array_column($r->json('data'), 'id');
|
||
expect($ids)->toContain($mskOnly->id);
|
||
expect($ids)->toContain($mskSpb->id);
|
||
expect($ids)->toContain($allRf->id);
|
||
expect($ids)->not->toContain($spbOnly->id);
|
||
});
|
||
|
||
it('region filter outside 1..89 range is silently ignored (returns all)', function () {
|
||
$tenant = Tenant::factory()->create();
|
||
$user = User::factory()->create(['tenant_id' => $tenant->id]);
|
||
Project::factory()->count(3)->create(['tenant_id' => $tenant->id]);
|
||
|
||
$r = $this->actingAs($user)->getJson('/api/projects?region=999');
|
||
|
||
expect($r->json('meta.total'))->toBe(3);
|
||
});
|
||
|
||
// #7 / П15 — фильтр по дню недели приёма (битмаска).
|
||
it('delivery_day filter keeps only projects whose mask covers that day', function () {
|
||
$tenant = Tenant::factory()->create();
|
||
$user = User::factory()->create(['tenant_id' => $tenant->id]);
|
||
// 127 = все 7 дней, 31 = Пн-Пт (биты 0..4), 0 не валиден на CHECK constraint'е → используем 1 (только Пн).
|
||
$allDays = Project::factory()->create(['tenant_id' => $tenant->id, 'delivery_days_mask' => 127]);
|
||
$weekdays = Project::factory()->create(['tenant_id' => $tenant->id, 'delivery_days_mask' => 31]);
|
||
$mondayOnly = Project::factory()->create(['tenant_id' => $tenant->id, 'delivery_days_mask' => 1]);
|
||
|
||
// День Сб (index 5, bit 32). Должны попасть только проекты с битом 32 в маске = allDays.
|
||
$r = $this->actingAs($user)->getJson('/api/projects?delivery_day=5');
|
||
|
||
$r->assertOk();
|
||
$ids = array_column($r->json('data'), 'id');
|
||
expect($ids)->toContain($allDays->id);
|
||
expect($ids)->not->toContain($weekdays->id);
|
||
expect($ids)->not->toContain($mondayOnly->id);
|
||
});
|
||
|
||
it('delivery_day=0 (Monday) filter does not get treated as "no filter"', function () {
|
||
$tenant = Tenant::factory()->create();
|
||
$user = User::factory()->create(['tenant_id' => $tenant->id]);
|
||
$mon = Project::factory()->create(['tenant_id' => $tenant->id, 'delivery_days_mask' => 1]); // только Пн (бит 0)
|
||
$tueOnly = Project::factory()->create(['tenant_id' => $tenant->id, 'delivery_days_mask' => 2]); // только Вт (бит 1)
|
||
|
||
$r = $this->actingAs($user)->getJson('/api/projects?delivery_day=0');
|
||
|
||
$r->assertOk();
|
||
$ids = array_column($r->json('data'), 'id');
|
||
expect($ids)->toContain($mon->id);
|
||
expect($ids)->not->toContain($tueOnly->id);
|
||
});
|