Files
portal/app/tests/Frontend/TopupDialogInvoice.spec.ts
T
Дмитрий cdfae077a3
Accessibility (Pa11y live) / a11y (push) Has been cancelled
SAST — Semgrep / Semgrep SAST scan (push) Has been cancelled
feat(биллинг): оплата по счёту (Этап 1) — счёт, акт, отметка оплаты
Клиент сам выставляет PDF-счёт (TopupDialog вкладка «По счёту»), счета и
акты — в отдельной вкладке «Счета». Админ (/admin/invoices) отмечает оплату
одной кнопкой → атомарно зачисляет баланс (BillingTopupService), формирует
Акт (без НДС, saas_upd_documents ДОП) и шлёт клиенту письмо «Счёт оплачен»
с вложением PDF-акта. PDF открываются inline в браузере (ASCII-имя).

- Сервисы InvoiceNumberGenerator/InvoiceService/ActService/InvoicePaymentService/PdfRenderer
- Контроллеры InvoiceController (клиент) + AdminInvoiceController (список+mark-paid)
- Модели SaasInvoice/SaasInvoiceItem/SaasUpdDocument; шаблоны pdf/invoice|act
- Нумерация СЧ-ГГГГ-NNNNN (advisory-lock); просрочка invoices:expire (cron)
- Наименование услуги: «Оплата генерации рекламных лидов»
- Зависимость barryvdh/laravel-dompdf (default_font dejavu sans); схема БД не менялась
- Этап 2 (автомат через ВТБ API) — отдельно, спека/план в docs/superpowers

Тесты: счета 13, Billing 138, фронт зелёные; larastan baseline +6 (Pest false-pos).

Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
2026-06-29 11:32:21 +03:00

56 lines
2.2 KiB
TypeScript

import { describe, it, expect, vi, beforeEach } from 'vitest';
import { mount, flushPromises } from '@vue/test-utils';
import { createVuetify } from 'vuetify';
import TopupDialog from '../../resources/js/components/billing/TopupDialog.vue';
import * as billingApi from '../../resources/js/api/billing';
describe('TopupDialog — оплата по счёту', () => {
beforeEach(() => {
vi.clearAllMocks();
vi.stubGlobal('open', vi.fn());
});
it('при способе «По счёту» и сумме вызывает createInvoice и открывает PDF', async () => {
const spy = vi.spyOn(billingApi, 'createInvoice').mockResolvedValue({
id: 1,
invoice_number: 'СЧ-2026-00001',
amount_total: '1500.00',
pdf_url: '/api/billing/invoices/1/pdf',
});
const wrapper = mount(TopupDialog, {
props: { modelValue: true },
global: { plugins: [createVuetify()] },
});
const vm = wrapper.vm as unknown as { method: string; amount: number | null; submit: () => Promise<void> };
vm.method = 'invoice';
vm.amount = 1500;
await vm.submit();
await flushPromises();
expect(spy).toHaveBeenCalledWith(1500);
expect(window.open).toHaveBeenCalledWith('/api/billing/invoices/1/pdf', '_blank');
expect(wrapper.emitted('invoiced')?.[0]).toEqual(['СЧ-2026-00001']);
});
it('способ «Картой» вызывает topup, не createInvoice', async () => {
const topupSpy = vi.spyOn(billingApi, 'topup').mockResolvedValue({ balance_rub: '2000.00' });
const invoiceSpy = vi.spyOn(billingApi, 'createInvoice');
const wrapper = mount(TopupDialog, {
props: { modelValue: true },
global: { plugins: [createVuetify()] },
});
const vm = wrapper.vm as unknown as { method: string; amount: number | null; submit: () => Promise<void> };
vm.method = 'card';
vm.amount = 2000;
await vm.submit();
await flushPromises();
expect(topupSpy).toHaveBeenCalledWith(2000);
expect(invoiceSpy).not.toHaveBeenCalled();
});
});