f81308a0df
Move all test files out of source dirs into per-workspace tests/: - shared/tests/ (3 unit test files) - server/tests/ (1 integration test) - src/tests/ (8 characterization + scenario tests + testHelpers) Fix all relative import paths (App, storage, __mocks__, testHelpers). Fix jest.config testMatch globs in shared/ and server/ (rootDir + <rootDir>/tests pattern). Delete scripts/repro-pause-bug.js (debug scratch, superseded by turn.pause-add.test.js). Keep scripts/replay-combat.js + scripts/audit-rotation.js as manual demo/exploratory tools (NOT unit tests, not deterministic). No logic changes. All green: shared 49 + 1 validated RED, server 23, FE 62. Scenario test unchanged (240s timeout, pre-existing slow).
105 lines
4.7 KiB
JavaScript
105 lines
4.7 KiB
JavaScript
// test helpers: drive App UI to states. Used across characterization suites.
|
|
import React from 'react';
|
|
import { render, screen, fireEvent, waitFor, within } from '@testing-library/react';
|
|
import App from '../App';
|
|
import { MOCK_DB } from '../__mocks__/firebase/_mock-db';
|
|
|
|
// Scoped container: the "Add Participants" section (avoids label clashes with CharacterManager).
|
|
export function getParticipantForm() {
|
|
const heading = screen.getByText('Add Participants');
|
|
// closest section/div wrapping the form
|
|
let node = heading;
|
|
for (let i = 0; i < 6; i++) {
|
|
node = node.parentElement;
|
|
if (!node) break;
|
|
if (node.querySelector('form')) return node;
|
|
}
|
|
return heading.parentElement;
|
|
}
|
|
|
|
// Render app, wait for auth + campaign list.
|
|
export async function renderApp() {
|
|
window.history.replaceState({}, '', '/');
|
|
global.alert = jest.fn();
|
|
window.open = jest.fn();
|
|
const utils = render(<App />);
|
|
await waitFor(() => screen.getByRole('button', { name: /Create Campaign/i }));
|
|
return utils;
|
|
}
|
|
|
|
// Open create-campaign modal, fill name, submit. Returns campaign id from recorded call.
|
|
export async function createCampaignViaUI(name = 'Test Campaign') {
|
|
fireEvent.click(screen.getByRole('button', { name: /Create Campaign/i }));
|
|
await waitFor(() => screen.getByLabelText(/Campaign Name/i));
|
|
fireEvent.change(screen.getByLabelText(/Campaign Name/i), { target: { value: name } });
|
|
fireEvent.click(screen.getByRole('button', { name: /^Create$/i }));
|
|
// wait for setDoc recorded
|
|
const { getCalls } = require('../__mocks__/firebase/_mock-db');
|
|
await waitFor(() => getCalls().find(c => c.fn === 'setDoc' && c.path.includes('/campaigns/')));
|
|
const call = getCalls().find(c => c.fn === 'setDoc' && c.path.includes('/campaigns/'));
|
|
return call.path.split('/').pop(); // campaign id
|
|
}
|
|
|
|
// Click campaign card by name to select it. Returns selected campaign id.
|
|
export async function selectCampaignByName(name) {
|
|
const card = await waitFor(() => screen.getByText(name));
|
|
fireEvent.click(card);
|
|
await waitFor(() => screen.getByText(/Managing:/i));
|
|
}
|
|
|
|
// Open create-encounter modal, fill name, submit. Assumes campaign selected.
|
|
export async function createEncounterViaUI(name = 'Test Encounter') {
|
|
fireEvent.click(screen.getByRole('button', { name: /Create Encounter/i }));
|
|
await waitFor(() => screen.getByLabelText(/Encounter Name/i));
|
|
fireEvent.change(screen.getByLabelText(/Encounter Name/i), { target: { value: name } });
|
|
fireEvent.click(screen.getByRole('button', { name: /^Create$/i }));
|
|
const { getCalls } = require('../__mocks__/firebase/_mock-db');
|
|
await waitFor(() => getCalls().find(c => c.fn === 'setDoc' && c.path.includes('/encounters/')));
|
|
const call = getCalls().find(c => c.fn === 'setDoc' && c.path.includes('/encounters/'));
|
|
return call.path.split('/').pop();
|
|
}
|
|
|
|
// Click encounter card by name. Assumes campaign selected.
|
|
export async function selectEncounterByName(name) {
|
|
const card = await waitFor(() => screen.getByText(name));
|
|
fireEvent.click(card);
|
|
await waitFor(() => screen.getByText(/Managing Encounter:/i));
|
|
}
|
|
|
|
// Add a monster participant via the ParticipantManager form. Assumes encounter selected.
|
|
export async function addMonsterViaUI(name = 'Goblin', maxHp = 7, initMod = 2) {
|
|
const form = within(getParticipantForm());
|
|
fireEvent.change(form.getByPlaceholderText('e.g., Dire Wolf'), { target: { value: name } });
|
|
fireEvent.change(form.getByLabelText(/Init Mod/i), { target: { value: String(initMod) } });
|
|
fireEvent.change(form.getByLabelText(/Max HP/i), { target: { value: String(maxHp) } });
|
|
fireEvent.click(form.getByRole('button', { name: /Add to Encounter/i }));
|
|
const { getCalls } = require('../__mocks__/firebase/_mock-db');
|
|
await waitFor(() => {
|
|
const calls = getCalls().filter(c => c.fn === 'updateDoc' && c.path.includes('/encounters/'));
|
|
const last = calls[calls.length - 1];
|
|
return last && last.data.participants && last.data.participants.some(p => p.name === name);
|
|
});
|
|
}
|
|
|
|
// Full setup: app -> campaign -> encounter selected.
|
|
export async function setupReady(campName = 'Camp', encName = 'Enc') {
|
|
await renderApp();
|
|
await createCampaignViaUI(campName);
|
|
await selectCampaignByName(campName);
|
|
await createEncounterViaUI(encName);
|
|
await selectEncounterByName(encName);
|
|
}
|
|
|
|
// Start combat. Assumes encounter selected with active participants.
|
|
export async function startCombatViaUI() {
|
|
fireEvent.click(screen.getByRole('button', { name: /Start Combat/i }));
|
|
const { getCalls } = require('../__mocks__/firebase/_mock-db');
|
|
await waitFor(() => {
|
|
const calls = getCalls().filter(c => c.fn === 'updateDoc' && c.path.includes('/encounters/'));
|
|
const last = calls[calls.length - 1];
|
|
return last && last.data.isStarted === true;
|
|
});
|
|
}
|
|
|
|
export { MOCK_DB };
|