2ec70b338f
Accessibility (Pa11y live) / a11y (push) Has been cancelled
Закрыто 36 из 55 пре-существующих падений backend-набора (55 to 19), всё тест-сторона, код продукта не тронут. Группы: - incident-показ/РКН: добавлен SharesSupplierPdo + синхрон уровня транзакции в трейте (вложенный transaction на общем PDO теперь делает SAVEPOINT, не повторный BEGIN). - auto-pause и lead-delivery: тесты создают project_routing_snapshots, от которого зависит выбор кандидатов в LeadRouter (slepok-инвариант). - изоляция 16 протекающих тестов: добавлен DatabaseTransactions (где нужно плюс SharesSupplierPdo) — перестали оставлять committed-строки, отравлявшие глобально сканирующие тесты (snapshot, verify-audit, size-N). - partition time-bombs: ensureRange месячных партиций для тестов на дату 2026-05. - устаревшие ассерты: SchemaDelta метрики v8.35 to v8.52, ProjectsStore телефон 8 to 7 нормализуется, incidents-watch фильтр активного admin, register captcha_token, impersonation активный юзер тенанта, activity_log.deal_id, ProjectUpdateDedup пауза. Остаток 19 (отдельно): verify-audit-chains и size-N (протекатели audit-строк), webhook B-префикс (решение владельца), пара env/каскадных. Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
110 lines
3.9 KiB
PHP
110 lines
3.9 KiB
PHP
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
use Illuminate\Foundation\Testing\DatabaseTransactions;
|
|
use Illuminate\Support\Carbon;
|
|
use Illuminate\Support\Facades\DB;
|
|
use Tests\Concerns\SharesSupplierPdo;
|
|
|
|
uses(DatabaseTransactions::class);
|
|
uses(SharesSupplierPdo::class);
|
|
|
|
// Helper: insert a failed_webhook_jobs row
|
|
function makeFailedWebhookJob(string $exception, ?DateTimeInterface $at = null): void
|
|
{
|
|
DB::table('failed_webhook_jobs')->insert([
|
|
'failed_at' => $at ?? now(),
|
|
'exception' => $exception,
|
|
'raw_payload' => '{}',
|
|
'retry_count' => 0,
|
|
]);
|
|
}
|
|
|
|
// Helper: ensure at least one saas_admin_users row exists for FK
|
|
function ensureSystemAdmin(): int
|
|
{
|
|
// ВАЖНО: фильтруем по контракту команды (IncidentsWatchFailures ищет
|
|
// is_active=true AND deleted_at IS NULL). Без фильтра протёкшие неактивные
|
|
// admin'ы от других тестов (PdErasureServiceTest вставляет is_active=false
|
|
// committed через pgsql_supplier) удовлетворяли бы value('id') → ранний
|
|
// return → активный admin не создавался → команда warn-only → 0 инцидентов.
|
|
$id = DB::table('saas_admin_users')->where('is_active', true)->whereNull('deleted_at')->value('id');
|
|
if ($id !== null) {
|
|
return (int) $id;
|
|
}
|
|
|
|
return (int) DB::table('saas_admin_users')->insertGetId([
|
|
'email' => 'system-cron@liderra.ru',
|
|
'full_name' => 'System Cron',
|
|
'password_hash' => '$2y$12$placeholder',
|
|
'role' => 'dev_oncall',
|
|
'is_active' => true,
|
|
'created_at' => now(),
|
|
]);
|
|
}
|
|
|
|
beforeEach(function () {
|
|
ensureSystemAdmin();
|
|
});
|
|
|
|
test('does not create incident when failures are below threshold', function () {
|
|
$now = Carbon::now();
|
|
// Insert 5 failures — below default threshold of 200
|
|
for ($i = 0; $i < 5; $i++) {
|
|
makeFailedWebhookJob('App\\Exceptions\\WebhookException: connection refused', $now);
|
|
}
|
|
|
|
$this->artisan('incidents:watch-failures')->assertSuccessful();
|
|
|
|
expect(DB::table('incidents_log')->count())->toBe(0);
|
|
});
|
|
|
|
test('creates incident when failures exceed threshold', function () {
|
|
$now = Carbon::now();
|
|
// Insert 201 failures with same exception signature
|
|
for ($i = 0; $i < 201; $i++) {
|
|
makeFailedWebhookJob('App\\Exceptions\\WebhookException: connection refused', $now);
|
|
}
|
|
|
|
$this->artisan('incidents:watch-failures')->assertSuccessful();
|
|
|
|
expect(DB::table('incidents_log')->count())->toBe(1);
|
|
|
|
$incident = DB::table('incidents_log')->first();
|
|
expect($incident->type)->toBe('other');
|
|
expect($incident->severity)->toBe('high');
|
|
expect($incident->summary)->toContain('201');
|
|
});
|
|
|
|
test('deduplicates: does not create second incident for same ongoing storm', function () {
|
|
$now = Carbon::now();
|
|
for ($i = 0; $i < 201; $i++) {
|
|
makeFailedWebhookJob('App\\Exceptions\\WebhookException: timeout', $now);
|
|
}
|
|
|
|
// First run creates incident
|
|
$this->artisan('incidents:watch-failures')->assertSuccessful();
|
|
expect(DB::table('incidents_log')->count())->toBe(1);
|
|
|
|
// Second run (storm still ongoing) should NOT create another incident
|
|
$this->artisan('incidents:watch-failures')->assertSuccessful();
|
|
expect(DB::table('incidents_log')->count())->toBe(1);
|
|
});
|
|
|
|
test('creates separate incidents for different exception signatures', function () {
|
|
$now = Carbon::now();
|
|
// 201 failures with exception A
|
|
for ($i = 0; $i < 201; $i++) {
|
|
makeFailedWebhookJob('App\\Exceptions\\WebhookException: connection refused', $now);
|
|
}
|
|
// 201 failures with exception B
|
|
for ($i = 0; $i < 201; $i++) {
|
|
makeFailedWebhookJob('App\\Exceptions\\CurlException: SSL handshake failed', $now);
|
|
}
|
|
|
|
$this->artisan('incidents:watch-failures')->assertSuccessful();
|
|
|
|
expect(DB::table('incidents_log')->count())->toBe(2);
|
|
});
|