feat(балансы): backend плитки балансов внешних сервисов

Ежедневный контроль баланса DaData/Поставщик/Yandex Cloud плиткой дашборда.

- Таблица external_service_balances (pgsql_supplier, BYPASSRLS, last-value upsert)
- BalanceHealth: чистая логика светофора (red <floor или <3д; amber <floor или <7д)
- BalanceProvider+DTO; провайдеры DaData(API)/YC(OAuth→IAM→billing)/Supplier(Playwright)
- RefreshExternalBalancesJob: изоляция провайдеров (try/catch), расписание 06:30 МСК
- AdminDashboardController::balances() + плитка в summary + topup_url (кнопка «Пополнить»)
- Тесты: BalanceHealth, 3 провайдера, джоба, endpoint (102 теста зелёные)

Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
This commit is contained in:
Дмитрий
2026-06-28 07:11:02 +03:00
parent 95ea4b764e
commit 88e816c576
21 changed files with 966 additions and 12 deletions
+34
View File
@@ -0,0 +1,34 @@
<?php
declare(strict_types=1);
use App\Services\External\DadataBalanceProvider;
use Illuminate\Support\Facades\Http;
use Tests\TestCase;
uses(TestCase::class); // нужен booted-app: config()/app()/Http::fake()
it('читает баланс DaData по API', function () {
config()->set('services.dadata.api_key', 'test-token');
Http::fake(['dadata.ru/*' => Http::response(['balance' => 4500.0], 200)]);
$r = app(DadataBalanceProvider::class)->fetch();
expect($r->ok)->toBeTrue();
expect($r->balance)->toBe(4500.0);
expect($r->serviceKey)->toBe('dadata');
});
it('ошибка API → fail, не бросает', function () {
config()->set('services.dadata.api_key', 'test-token');
Http::fake(['dadata.ru/*' => Http::response('forbidden', 403)]);
$r = app(DadataBalanceProvider::class)->fetch();
expect($r->ok)->toBeFalse();
expect($r->error)->not->toBeNull();
});
it('нет ключа → fail без сетевого вызова', function () {
config()->set('services.dadata.api_key', '');
Http::fake();
$r = app(DadataBalanceProvider::class)->fetch();
expect($r->ok)->toBeFalse();
Http::assertNothingSent();
});