Files
TREK/client/tests/unit/slices/dayNotesSlice.test.ts
T
2026-04-07 12:31:09 +02:00

177 lines
6.3 KiB
TypeScript

import { describe, it, expect, beforeEach, vi } from 'vitest';
import { http, HttpResponse } from 'msw';
import { useTripStore } from '../../../src/store/tripStore';
import { resetAllStores, seedStore } from '../../helpers/store';
import { buildDay, buildDayNote } from '../../helpers/factories';
import { server } from '../../helpers/msw/server';
vi.mock('../../../src/api/websocket', () => ({
connect: vi.fn(),
disconnect: vi.fn(),
getSocketId: vi.fn(() => null),
joinTrip: vi.fn(),
leaveTrip: vi.fn(),
addListener: vi.fn(),
removeListener: vi.fn(),
setRefetchCallback: vi.fn(),
}));
beforeEach(() => {
resetAllStores();
});
describe('dayNotesSlice', () => {
describe('addDayNote', () => {
it('FE-DAYNOTES-001: addDayNote inserts temp note immediately, replaces on success', async () => {
seedStore(useTripStore, { dayNotes: { '1': [] } });
let tempAdded = false;
const realNote = buildDayNote({ id: 500, day_id: 1, text: 'New note' });
server.use(
http.post('/api/trips/1/days/1/notes', async () => {
const state = useTripStore.getState();
const notes = state.dayNotes['1'];
if (notes.some(n => n.id < 0)) {
tempAdded = true;
}
return HttpResponse.json({ note: realNote });
}),
);
const result = await useTripStore.getState().addDayNote(1, 1, { text: 'New note', sort_order: 0 });
expect(tempAdded).toBe(true);
expect(result.id).toBe(500);
const notes = useTripStore.getState().dayNotes['1'];
expect(notes).toHaveLength(1);
expect(notes[0].id).toBe(500);
});
it('FE-DAYNOTES-002: addDayNote on failure rolls back — temp note removed', async () => {
seedStore(useTripStore, { dayNotes: { '1': [] } });
server.use(
http.post('/api/trips/1/days/1/notes', () =>
HttpResponse.json({ message: 'Error' }, { status: 500 })
),
);
await expect(
useTripStore.getState().addDayNote(1, 1, { text: 'Fail note', sort_order: 0 })
).rejects.toThrow();
expect(useTripStore.getState().dayNotes['1']).toHaveLength(0);
});
});
describe('updateDayNote', () => {
it('FE-DAYNOTES-003: updateDayNote replaces note in map by id', async () => {
const note = buildDayNote({ id: 10, day_id: 1, text: 'Old text' });
seedStore(useTripStore, { dayNotes: { '1': [note] } });
const updated = { ...note, text: 'Updated text' };
server.use(
http.put('/api/trips/1/days/1/notes/10', () =>
HttpResponse.json({ note: updated })
),
);
const result = await useTripStore.getState().updateDayNote(1, 1, 10, { text: 'Updated text' });
expect(result.text).toBe('Updated text');
expect(useTripStore.getState().dayNotes['1'][0].text).toBe('Updated text');
});
});
describe('deleteDayNote', () => {
it('FE-DAYNOTES-004: deleteDayNote optimistically removes note, restores on failure', async () => {
const note = buildDayNote({ id: 10, day_id: 1 });
seedStore(useTripStore, { dayNotes: { '1': [note] } });
server.use(
http.delete('/api/trips/1/days/1/notes/10', () =>
HttpResponse.json({ message: 'Error' }, { status: 500 })
),
);
await expect(useTripStore.getState().deleteDayNote(1, 1, 10)).rejects.toThrow();
// Rolled back
expect(useTripStore.getState().dayNotes['1']).toHaveLength(1);
expect(useTripStore.getState().dayNotes['1'][0].id).toBe(10);
});
it('FE-DAYNOTES-004b: deleteDayNote success removes note from correct day', async () => {
const note1 = buildDayNote({ id: 10, day_id: 1 });
const note2 = buildDayNote({ id: 20, day_id: 1 });
seedStore(useTripStore, { dayNotes: { '1': [note1, note2] } });
await useTripStore.getState().deleteDayNote(1, 1, 10);
const notes = useTripStore.getState().dayNotes['1'];
expect(notes).toHaveLength(1);
expect(notes[0].id).toBe(20);
});
});
describe('moveDayNote', () => {
it('FE-DAYNOTES-005: moveDayNote removes from source, adds to target (delete+create)', async () => {
const note = buildDayNote({ id: 10, day_id: 1, text: 'Move me' });
const newNote = buildDayNote({ id: 99, day_id: 2, text: 'Move me' });
seedStore(useTripStore, { dayNotes: { '1': [note], '2': [] } });
server.use(
http.delete('/api/trips/1/days/1/notes/10', () => HttpResponse.json({ success: true })),
http.post('/api/trips/1/days/2/notes', () => HttpResponse.json({ note: newNote })),
);
await useTripStore.getState().moveDayNote(1, 1, 2, 10);
expect(useTripStore.getState().dayNotes['1']).toHaveLength(0);
expect(useTripStore.getState().dayNotes['2']).toHaveLength(1);
expect(useTripStore.getState().dayNotes['2'][0].id).toBe(99);
});
it('FE-DAYNOTES-006: moveDayNote rolls back to source day on failure', async () => {
const note = buildDayNote({ id: 10, day_id: 1, text: 'Move me' });
seedStore(useTripStore, { dayNotes: { '1': [note], '2': [] } });
server.use(
http.delete('/api/trips/1/days/1/notes/10', () =>
HttpResponse.json({ message: 'Error' }, { status: 500 })
),
);
await expect(useTripStore.getState().moveDayNote(1, 1, 2, 10)).rejects.toThrow();
expect(useTripStore.getState().dayNotes['1']).toHaveLength(1);
expect(useTripStore.getState().dayNotes['1'][0].id).toBe(10);
});
});
describe('updateDayNotes', () => {
it('FE-DAYNOTES-007: updateDayNotes persists notes text and updates days array', async () => {
const day = buildDay({ id: 1, trip_id: 1, notes: null });
seedStore(useTripStore, { days: [day] });
await useTripStore.getState().updateDayNotes(1, 1, 'My travel notes');
const updatedDay = useTripStore.getState().days.find(d => d.id === 1);
expect(updatedDay?.notes).toBe('My travel notes');
});
});
describe('updateDayTitle', () => {
it('FE-DAYNOTES-008: updateDayTitle persists title and updates days array', async () => {
const day = buildDay({ id: 1, trip_id: 1, title: null });
seedStore(useTripStore, { days: [day] });
await useTripStore.getState().updateDayTitle(1, 1, 'Day at the Beach');
const updatedDay = useTripStore.getState().days.find(d => d.id === 1);
expect(updatedDay?.title).toBe('Day at the Beach');
});
});
});