2026-05-24 12:00:56 +03:00
|
|
|
|
<?php
|
|
|
|
|
|
|
|
|
|
|
|
declare(strict_types=1);
|
|
|
|
|
|
|
|
|
|
|
|
use App\Jobs\Billing\BalancePreflightSweepJob;
|
|
|
|
|
|
use App\Mail\BalanceFrozenMail;
|
|
|
|
|
|
use App\Models\PricingTier;
|
|
|
|
|
|
use App\Models\Project;
|
|
|
|
|
|
use App\Models\Tenant;
|
|
|
|
|
|
use Illuminate\Foundation\Testing\DatabaseTransactions;
|
|
|
|
|
|
use Illuminate\Support\Facades\DB;
|
|
|
|
|
|
use Illuminate\Support\Facades\Mail;
|
|
|
|
|
|
use Tests\Concerns\SharesSupplierPdo;
|
|
|
|
|
|
|
|
|
|
|
|
// Изоляция: liderra_testing persistent (RefreshDatabase off). DatabaseTransactions
|
|
|
|
|
|
// откатывает default-pgsql после каждого теста; SharesSupplierPdo делает pgsql_supplier
|
|
|
|
|
|
// общим PDO с pgsql — иначе job-запись balance_freeze_log (pgsql_supplier) не видит
|
|
|
|
|
|
// незакоммиченного tenant и падает на FK (паттерн Спека B / AutoPauseFlowTest).
|
|
|
|
|
|
uses(DatabaseTransactions::class);
|
|
|
|
|
|
uses(SharesSupplierPdo::class);
|
|
|
|
|
|
|
|
|
|
|
|
beforeEach(function () {
|
|
|
|
|
|
// RLS-контекст (системный tenant 0) — паттерн supplier-тестов.
|
|
|
|
|
|
DB::statement("SELECT set_config('app.current_tenant_id', '0', true)");
|
|
|
|
|
|
PricingTier::query()->create(['tier_no' => 1, 'leads_in_tier' => null, 'price_per_lead_kopecks' => 5000, 'is_active' => true, 'effective_from' => now()]);
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
it('freezes tenant whose balance no longer covers daily limit', function () {
|
|
|
|
|
|
Mail::fake();
|
|
|
|
|
|
// 500₽ / 50₽ = 10 лидов; проекты хотят 25 → заморозка.
|
|
|
|
|
|
$tenant = Tenant::factory()->create(['balance_rub' => '500.00', 'frozen_by_balance_at' => null]);
|
|
|
|
|
|
Project::factory()->for($tenant)->create(['is_active' => true, 'daily_limit_target' => 25]);
|
|
|
|
|
|
|
|
|
|
|
|
(new BalancePreflightSweepJob)->handle();
|
|
|
|
|
|
|
|
|
|
|
|
expect($tenant->fresh()->frozen_by_balance_at)->not->toBeNull();
|
|
|
|
|
|
Mail::assertQueued(BalanceFrozenMail::class);
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
it('unfreezes tenant whose balance now covers daily limit', function () {
|
|
|
|
|
|
Mail::fake();
|
|
|
|
|
|
// 2000₽ / 50₽ = 40 лидов; хотят 25 → разморозка.
|
|
|
|
|
|
$tenant = Tenant::factory()->create(['balance_rub' => '2000.00', 'frozen_by_balance_at' => now()->subDay()]);
|
|
|
|
|
|
Project::factory()->for($tenant)->create(['is_active' => true, 'daily_limit_target' => 25]);
|
|
|
|
|
|
|
|
|
|
|
|
(new BalancePreflightSweepJob)->handle();
|
|
|
|
|
|
|
|
|
|
|
|
expect($tenant->fresh()->frozen_by_balance_at)->toBeNull();
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
it('is idempotent — does not re-freeze already frozen tenant', function () {
|
|
|
|
|
|
Mail::fake();
|
|
|
|
|
|
$frozenAt = now()->subDay();
|
|
|
|
|
|
$tenant = Tenant::factory()->create(['balance_rub' => '0.00', 'frozen_by_balance_at' => $frozenAt]);
|
|
|
|
|
|
Project::factory()->for($tenant)->create(['is_active' => true, 'daily_limit_target' => 25]);
|
|
|
|
|
|
|
|
|
|
|
|
(new BalancePreflightSweepJob)->handle();
|
|
|
|
|
|
|
2026-05-25 03:28:00 +03:00
|
|
|
|
// Дата заморозки не перезаписана; для ЭТОГО tenant повторного письма нет.
|
|
|
|
|
|
// NB: per-tenant фильтр, т.к. liderra_testing persistent (DemoSeeder тенанты
|
|
|
|
|
|
// могут попасть в sweep и тоже получить BalanceFrozenMail — не наш ответ).
|
2026-05-24 12:00:56 +03:00
|
|
|
|
expect($tenant->fresh()->frozen_by_balance_at->timestamp)->toBe($frozenAt->timestamp);
|
2026-05-25 03:28:00 +03:00
|
|
|
|
Mail::assertNotQueued(BalanceFrozenMail::class, fn ($mail) => $mail->tenant->id === $tenant->id);
|
2026-05-24 12:00:56 +03:00
|
|
|
|
});
|