Files
portal/app/tests/Frontend/AutopodborCreateScreen.spec.ts
T
2026-06-28 15:47:56 +03:00

69 lines
3.6 KiB
TypeScript

import { describe, it, expect, beforeEach, vi } from 'vitest';
import { mount } from '@vue/test-utils';
import { createPinia, setActivePinia } from 'pinia';
import { createVuetify } from 'vuetify';
import { reactive, ref } from 'vue';
vi.mock('../../resources/js/api/autopodbor');
import CreateScreen from '../../resources/js/views/autopodbor/screens/CreateScreen.vue';
import { useAutopodborStore } from '../../resources/js/stores/autopodborStore';
const vuetify = createVuetify();
function makeNav() {
return { go: vi.fn(), ctx: reactive({ runId: 5, competitorId: 3, selectedSourceIds: [11, 12], loadMsg: '', loadSub: '', editProjectId: null, createdCount: 0, launched: false }), screen: ref('create') };
}
function seed(store: any) {
store.competitor = { id: 3, name: 'Окна Комфорт', is_federal: false, relevance_pct: 100, origin: 'auto', site_url: 'okna.ru', directory_urls: [], description: 'd', studied_at: '2026-06-28', study_run_id: 9, search_run_id: 5 };
store.sources = [
{ id: 11, competitor_id: 3, signal_type: 'site', identifier: 'okna-komfort-kzn.ru', phone_kind: null, provenance_url: null, provenance_label: '2ГИС', created_project_id: null, existing_project_id: null },
{ id: 12, competitor_id: 3, signal_type: 'call', identifier: '78432001122', phone_kind: 'real', provenance_url: null, provenance_label: '2ГИС', created_project_id: null, existing_project_id: null },
];
}
function mountCreate(nav: any) {
return mount(CreateScreen, { global: { plugins: [vuetify], provide: { autopodborNav: nav } } });
}
describe('CreateScreen', () => {
beforeEach(() => { setActivePinia(createPinia()); vi.clearAllMocks(); });
it('показывает выбранные источники и число проектов', async () => {
const store = useAutopodborStore(); seed(store);
const w = mountCreate(makeNav());
await new Promise(r => setTimeout(r, 0));
expect(w.text()).toContain('okna-komfort-kzn.ru');
expect(w.text()).toContain('Окна Комфорт'); // производное имя
});
it('«Создать (без запуска)» зовёт makeProjects(launch=false) и идёт на done', async () => {
const store = useAutopodborStore(); seed(store);
vi.spyOn(store, 'makeProjects').mockResolvedValue([{ id: 1, name: 'Окна Комфорт' }, { id: 2, name: 'Окна Комфорт ✓' }] as any);
const nav = makeNav();
const w = mountCreate(nav);
(w.vm as any).regionCode = 16;
await new Promise(r => setTimeout(r, 0));
const btn = w.findAll('button').find(b => b.text().includes('без запуска'));
await btn!.trigger('click');
await new Promise(r => setTimeout(r, 0));
expect(store.makeProjects).toHaveBeenCalled();
const arg = (store.makeProjects as any).mock.calls[0][0];
expect(arg.launch).toBe(false);
expect(arg.source_ids).toEqual([11, 12]);
expect(nav.ctx.createdCount).toBe(2);
expect(nav.go).toHaveBeenCalledWith('done');
});
it('409 нехватки баланса показывает сообщение и возвращает на create', async () => {
const store = useAutopodborStore(); seed(store);
vi.spyOn(store, 'makeProjects').mockRejectedValue({ response: { data: { error: 'balance_insufficient' } } });
const nav = makeNav();
const w = mountCreate(nav);
(w.vm as any).regionCode = 16;
await new Promise(r => setTimeout(r, 0));
const btn = w.findAll('button').find(b => b.text().includes('запустить'));
await btn!.trigger('click');
await new Promise(r => setTimeout(r, 0));
expect(store.makeProjects).toHaveBeenCalled();
expect(nav.go).toHaveBeenCalledWith('create');
});
});