Files
portal/app/tests/Feature/Notifications/NewLeadsDigestJobTest.php
T
Дмитрий 89ed1714e8
Accessibility (Pa11y live) / a11y (push) Has been cancelled
style(test): импорт Collection в digest-тесте (lint-фиксап, функц. идентично)
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
2026-06-21 18:45:21 +03:00

132 lines
5.5 KiB
PHP

<?php
declare(strict_types=1);
use App\Jobs\SendNewLeadsDigestJob;
use App\Mail\NewLeadNotification;
use App\Mail\NewLeadsDigestMail;
use App\Models\Deal;
use App\Models\Tenant;
use App\Models\User;
use App\Services\NotificationService;
use Illuminate\Foundation\Testing\DatabaseTransactions;
use Illuminate\Support\Collection;
use Illuminate\Support\Facades\Cache;
use Illuminate\Support\Facades\DB;
use Illuminate\Support\Facades\Mail;
uses(DatabaseTransactions::class);
beforeEach(function () {
Mail::fake();
});
function digestUser(Tenant $tenant, string $email, bool $emailOn): User
{
return User::factory()->create([
'tenant_id' => $tenant->id,
'email' => $email,
'notification_preferences' => [
'new_lead' => ['email' => $emailOn, 'inapp' => true],
],
]);
}
it('шлёт одно письмо-сводку с N сделками подписанному пользователю', function () {
$tenant = Tenant::factory()->create();
digestUser($tenant, 'digest-on@example.test', true);
DB::statement('SET app.current_tenant_id = '.$tenant->id);
Deal::factory()->for($tenant)->create(['received_at' => now()->subMinutes(5), 'is_test' => false]);
Deal::factory()->for($tenant)->create(['received_at' => now()->subMinutes(10), 'is_test' => false]);
Deal::factory()->for($tenant)->create(['received_at' => now()->subMinutes(15), 'is_test' => false]);
(new SendNewLeadsDigestJob)->handle(app(NotificationService::class));
Mail::assertSent(
NewLeadsDigestMail::class,
fn (NewLeadsDigestMail $m) => $m->hasTo('digest-on@example.test') && $m->deals->count() === 3,
);
});
it('не шлёт сводку пользователю с выключенным email', function () {
$tenant = Tenant::factory()->create();
digestUser($tenant, 'digest-off@example.test', false);
DB::statement('SET app.current_tenant_id = '.$tenant->id);
Deal::factory()->for($tenant)->create(['received_at' => now()->subMinutes(5), 'is_test' => false]);
(new SendNewLeadsDigestJob)->handle(app(NotificationService::class));
Mail::assertNotSent(
NewLeadsDigestMail::class,
fn (NewLeadsDigestMail $m) => $m->hasTo('digest-off@example.test'),
);
});
it('не шлёт сводку, если за окно нет новых сделок', function () {
$tenant = Tenant::factory()->create();
digestUser($tenant, 'digest-old@example.test', true);
DB::statement('SET app.current_tenant_id = '.$tenant->id);
Deal::factory()->for($tenant)->create(['received_at' => now()->subMinutes(45), 'is_test' => false]);
(new SendNewLeadsDigestJob)->handle(app(NotificationService::class));
Mail::assertNotSent(
NewLeadsDigestMail::class,
fn (NewLeadsDigestMail $m) => $m->hasTo('digest-old@example.test'),
);
});
it('повторный прогон в том же окне НЕ дублирует дайджест (N-4)', function () {
$tenant = Tenant::factory()->create();
digestUser($tenant, 'digest-dup@example.test', true);
DB::statement('SET app.current_tenant_id = '.$tenant->id);
Deal::factory()->for($tenant)->create(['received_at' => now()->subMinutes(5), 'is_test' => false]);
Deal::factory()->for($tenant)->create(['received_at' => now()->subMinutes(10), 'is_test' => false]);
// ранбук R3b сам велит дёргать джоб вручную → два прогона в одном окне
(new SendNewLeadsDigestJob)->handle(app(NotificationService::class));
(new SendNewLeadsDigestJob)->handle(app(NotificationService::class));
// письмо-сводка ушло РОВНО один раз — уже отправленные сделки не дублируются
Mail::assertSent(NewLeadsDigestMail::class, 1);
});
it('при падении отправки НЕ помечает сделки — следующий прогон повторит (N-4)', function () {
$tenant = Tenant::factory()->create();
digestUser($tenant, 'digest-fail@example.test', true);
DB::statement('SET app.current_tenant_id = '.$tenant->id);
$deal = Deal::factory()->for($tenant)->create(['received_at' => now()->subMinutes(5), 'is_test' => false]);
// джоб прерван ДО отправки: notify бросает (анон-subclass — тип-чистый для larastan)
$throwing = new class extends NotificationService
{
public function notifyNewLeadsDigest(Tenant $tenant, Collection $deals): void
{
throw new RuntimeException('mail down');
}
};
try {
(new SendNewLeadsDigestJob)->handle($throwing);
} catch (Throwable) {
// ожидаемо — джоб упал, очередь повторит
}
expect(Cache::has('digest_sent:'.$deal->id))->toBeFalse();
// следующий прогон с рабочим сервисом — дайджест НЕ потерян, уходит
(new SendNewLeadsDigestJob)->handle(app(NotificationService::class));
Mail::assertSent(NewLeadsDigestMail::class, 1);
});
it('notifyNewLead больше не шлёт пер-лид письмо', function () {
$tenant = Tenant::factory()->create();
digestUser($tenant, 'perlead@example.test', true);
DB::statement('SET app.current_tenant_id = '.$tenant->id);
$deal = Deal::factory()->for($tenant)->create(['received_at' => now(), 'is_test' => false]);
app(NotificationService::class)->notifyNewLead($tenant, $deal);
Mail::assertNotSent(NewLeadNotification::class);
});