Files
portal/app/tests/Frontend/RecoveryCodesView.spec.ts
T
Дмитрий 19096552b4 phase2(auth): закрыты 4 оставшихся auth-экрана из v8_login.html
- RegisterView: email + password (strength-meter 0..4) + 2 click-wrap'а
  (оферта + ПДн). 3-й «маркетинг» из handoff НЕ реализован (расхождение
  #2 реестра v1.13 - handoff противоречит ТЗ §1.5/§4.1).
- TwoFactorView: 6 input-cell с auto-focus вперёд при вводе цифры,
  Backspace назад при empty, paste 6 цифр заполняет все.
- ForgotPasswordView: email + alert «5 попыток / 15 минут» по ТЗ §1.7.
- RecoveryCodesView: 8 кодов в 2-column grid + Скачать .txt (Blob/URL.createObjectURL)
  + Копировать (navigator.clipboard) + warning о невозможности повторного просмотра.

Router: 4 новых маршрута (/register, /2fa, /forgot, /recovery), все
meta.layout='auth', lazy-imports.

Vitest +14 тестов (всего 24/24 за 3.29s):
- RegisterView 4 (вкл. assertion на отсутствие маркетингового click-wrap)
- TwoFactorView 3, ForgotPasswordView 3, RecoveryCodesView 4

Stories +4 (Histoire 6/6 за 29.17s).

Регресс: lint+type-check+format OK; vitest 24/24; vite build 5 lazy-chunks
для views + Vuetify в отдельные chunks (app chunk 198KB→78KB); Pest 48/48 за 4.85s.

CLAUDE.md v1.19→v1.20, реестр Открытых_вопросов v1.28→v1.29.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-08 17:09:56 +03:00

37 lines
1.5 KiB
TypeScript

import { describe, it, expect } from 'vitest';
import { mount } from '@vue/test-utils';
import { createVuetify } from 'vuetify';
import RecoveryCodesView from '../../resources/js/views/auth/RecoveryCodesView.vue';
describe('RecoveryCodesView.vue', () => {
const factory = () =>
mount(RecoveryCodesView, {
global: { plugins: [createVuetify()] },
});
it('монтируется и содержит заголовок «Резервные коды»', () => {
const wrapper = factory();
expect(wrapper.text()).toContain('Резервные коды');
});
it('рендерит ровно 8 кодов в формате XXXX-XXXX', () => {
const wrapper = factory();
const codes = wrapper.findAll('.code-item');
expect(codes).toHaveLength(8);
codes.forEach((c) => {
expect(c.text()).toMatch(/^[A-Z0-9]{4}-[A-Z0-9]{4}$/);
});
});
it('содержит warning о невозможности повторного просмотра', () => {
const wrapper = factory();
expect(wrapper.text()).toContain('После закрытия страницы коды нельзя посмотреть снова');
});
it('содержит кнопки «Скачать .txt» и «Копировать»', () => {
const wrapper = factory();
expect(wrapper.text()).toContain('Скачать');
expect(wrapper.text()).toContain('Копировать');
});
});