Files
portal/app/tests/Feature/PartitionsCreateMonthsTest.php
T
Дмитрий 1d4738dfa2 phase1(infra): partitions:create-months — Artisan-команда (замена pg_partman)
Закрыт пункт «pg_partman replacement» из project_phase1_strategy.md
(расширение pg_partman недоступно на native Windows-стеке без сборки
из исходников).

Реализация:
  - app/app/Console/Commands/PartitionsCreateMonths.php
    - signature: partitions:create-months {--ahead=2}
    - создаёт партиции для deals + supplier_lead_costs (обе по received_at)
    - идемпотентна (проверка через pg_class WHERE relkind='r' перед CREATE)
    - запускать ежесуточно через Windows Task Scheduler / cron

Smoke-test на dev: --ahead=8 создал 6 партиций (Nov 2026 - Jan 2027) +
12 skipped. После migrate:fresh партиции возвращаются к initial 6.

4 новых Pest-теста в PartitionsCreateMonthsTest:
  - создание партиций на 8 месяцев вперёд для обеих таблиц
  - идемпотентность (повторный --ahead=5 → 0 created, 12 skipped)
  - --ahead=0 создаёт только текущий месяц
  - INSERT в deals с received_at в новой партиции корректно роутится

Тесты используют beforeEach/afterEach для cleanup'а через
DROP TABLE ... CASCADE (FK webhook_dedup_keys на партицию propagates).

Pest 45/45 зелёные за 4.9 сек. Pint + Larastan чисто (phpstan-baseline
регенерирован для динамических свойств $this в Pest closure'ах).

CLAUDE.md v1.14 → v1.15. Реестр Открытые_вопросы v1.23 → v1.24.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-08 15:46:41 +03:00

130 lines
5.3 KiB
PHP
Raw Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
<?php
declare(strict_types=1);
use Illuminate\Support\Facades\Artisan;
use Illuminate\Support\Facades\DB;
/**
* Тесты Artisan-команды `partitions:create-months` (замена pg_partman).
*
* NB: команда создаёт партиции через DDL (CREATE TABLE) — это глобальное
* действие, не Rolled Back trait'ом DatabaseTransactions. Поэтому каждый
* тест чистит за собой созданные партиции вручную (DROP TABLE IF EXISTS).
*/
beforeEach(function () {
// Снимок партиций ДО теста для cleanup.
$this->partitionsBefore = collect(DB::select("
SELECT relname FROM pg_class
WHERE relkind = 'r'
AND relname ~ '^(deals|supplier_lead_costs)_[0-9]{4}_[0-9]{2}$'
"))->pluck('relname')->all();
});
afterEach(function () {
$partitionsAfter = collect(DB::select("
SELECT relname FROM pg_class
WHERE relkind = 'r'
AND relname ~ '^(deals|supplier_lead_costs)_[0-9]{4}_[0-9]{2}$'
"))->pluck('relname')->all();
// CASCADE: webhook_dedup_keys.FK на deals (partitioned) автоматически
// пропагируется на каждую партицию — нужно удалить связанные FK.
foreach (array_diff($partitionsAfter, $this->partitionsBefore) as $partition) {
DB::statement("DROP TABLE IF EXISTS {$partition} CASCADE");
}
});
test('создаёт партиции на N месяцев вперёд для deals и supplier_lead_costs', function () {
$exitCode = Artisan::call('partitions:create-months', ['--ahead' => 8]);
expect($exitCode)->toBe(0);
// Должны быть партиции до текущий+8 месяцев включительно.
$futureMonth = now()->startOfMonth()->addMonths(8);
$expectedDealName = 'deals_'.$futureMonth->format('Y_m');
$expectedCostName = 'supplier_lead_costs_'.$futureMonth->format('Y_m');
$row = DB::selectOne("SELECT 1 AS x FROM pg_class WHERE relname = ? AND relkind = 'r'", [$expectedDealName]);
expect($row)->not->toBeNull();
$row = DB::selectOne("SELECT 1 AS x FROM pg_class WHERE relname = ? AND relkind = 'r'", [$expectedCostName]);
expect($row)->not->toBeNull();
});
test('идемпотентность: повторный запуск не падает и не создаёт дубли', function () {
Artisan::call('partitions:create-months', ['--ahead' => 5]);
$afterFirst = collect(DB::select("
SELECT relname FROM pg_class
WHERE relkind = 'r'
AND relname ~ '^(deals|supplier_lead_costs)_[0-9]{4}_[0-9]{2}$'
"))->count();
// Повторный запуск — должен только skip'ать.
$exitCode = Artisan::call('partitions:create-months', ['--ahead' => 5]);
expect($exitCode)->toBe(0);
$afterSecond = collect(DB::select("
SELECT relname FROM pg_class
WHERE relkind = 'r'
AND relname ~ '^(deals|supplier_lead_costs)_[0-9]{4}_[0-9]{2}$'
"))->count();
expect($afterSecond)->toBe($afterFirst);
// Output второго запуска должен говорить «skipped» по всем 12 партициям (6 мес × 2 табл).
$output = Artisan::output();
expect($output)->toContain('0 created, 12 skipped');
});
test('--ahead=0 создаёт только текущий месяц', function () {
Artisan::call('partitions:create-months', ['--ahead' => 0]);
$currentMonth = now()->startOfMonth();
$name = 'deals_'.$currentMonth->format('Y_m');
$row = DB::selectOne("SELECT 1 AS x FROM pg_class WHERE relname = ? AND relkind = 'r'", [$name]);
expect($row)->not->toBeNull();
});
test('партиция корректно принимает INSERT в окно своего месяца', function () {
Artisan::call('partitions:create-months', ['--ahead' => 8]);
// Проверяем, что INSERT в deals с received_at в новой партиции работает.
$futureMonth = now()->startOfMonth()->addMonths(8);
$tenantId = DB::table('tenants')->insertGetId([
'subdomain' => 'partition-test-'.uniqid(),
'organization_name' => 'PartitionTest',
'contact_email' => 'pt@test.local',
'webhook_token' => str_repeat('p', 64),
'api_key_limit' => 5,
]);
$projectId = DB::table('projects')->insertGetId([
'tenant_id' => $tenantId,
'name' => 'PartitionProject',
'type' => 'webhook',
'is_active' => true,
'daily_limit_target' => 10,
'region_mask' => 255,
'region_mode' => 'include',
'delivery_days_mask' => 127,
'assignment_strategy' => 'manual',
'ttfr_target_minutes' => 15,
]);
$dealId = DB::table('deals')->insertGetId([
'tenant_id' => $tenantId,
'project_id' => $projectId,
'phone' => '79001234567',
'status' => 'new',
'received_at' => $futureMonth->copy()->addDays(5),
]);
expect($dealId)->toBeInt();
// Cleanup (за пределами trait DatabaseTransactions, т.к. INSERT через DB::table в новой партиции).
DB::table('deals')->where('id', $dealId)->delete();
DB::table('projects')->where('id', $projectId)->delete();
DB::table('tenants')->where('id', $tenantId)->delete();
});