cdfae077a3
Клиент сам выставляет 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>
56 lines
2.2 KiB
TypeScript
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();
|
|
});
|
|
});
|