import { describe, it, expect, vi } from 'vitest'; import { HttpException } from '@nestjs/common'; import { PackingController } from '../../../src/nest/packing/packing.controller'; import type { PackingService } from '../../../src/nest/packing/packing.service'; import type { User } from '../../../src/types'; const user = { id: 1, role: 'user', email: 'u@example.test' } as User; const admin = { id: 1, role: 'admin', email: 'a@example.test' } as User; const trip = { id: 5, user_id: 1 }; /** Service mock with trip access granted + edit allowed by default. */ function makeService(overrides: Partial = {}): PackingService { return { verifyTripAccess: vi.fn().mockReturnValue(trip), canEdit: vi.fn().mockReturnValue(true), broadcast: vi.fn(), notifyTagged: vi.fn(), ...overrides, } as unknown as PackingService; } function thrown(fn: () => unknown): { status: number; body: unknown } { try { fn(); } catch (err) { expect(err).toBeInstanceOf(HttpException); const e = err as HttpException; return { status: e.getStatus(), body: e.getResponse() }; } throw new Error('expected the handler to throw'); } describe('PackingController (parity with the legacy /api/trips/:tripId/packing route)', () => { it('404 when the trip is not accessible', () => { const svc = makeService({ verifyTripAccess: vi.fn().mockReturnValue(undefined) }); expect(thrown(() => new PackingController(svc).list(user, '5'))).toEqual({ status: 404, body: { error: 'Trip not found' }, }); }); it('GET / returns items for an accessible trip', () => { const svc = makeService({ listItems: vi.fn().mockReturnValue([{ id: 1 }]) } as Partial); expect(new PackingController(svc).list(user, '5')).toEqual({ items: [{ id: 1 }] }); }); describe('POST / (create)', () => { it('403 without packing_edit permission', () => { const svc = makeService({ canEdit: vi.fn().mockReturnValue(false) }); expect(thrown(() => new PackingController(svc).create(user, '5', { name: 'Socks' }))).toEqual({ status: 403, body: { error: 'No permission' }, }); }); it('400 when name missing', () => { const svc = makeService(); expect(thrown(() => new PackingController(svc).create(user, '5', {}))).toEqual({ status: 400, body: { error: 'Item name is required' }, }); }); it('creates an item and broadcasts', () => { const createItem = vi.fn().mockReturnValue({ id: 9, name: 'Socks' }); const broadcast = vi.fn(); const svc = makeService({ createItem, broadcast } as Partial); expect(new PackingController(svc).create(user, '5', { name: 'Socks' }, 'sock')).toEqual({ item: { id: 9, name: 'Socks' } }); expect(broadcast).toHaveBeenCalledWith('5', 'packing:created', { item: { id: 9, name: 'Socks' } }, 'sock'); }); }); it('GET / lists items for the trip (success path)', () => { const listItems = vi.fn().mockReturnValue([{ id: 1 }, { id: 2 }]); const svc = makeService({ listItems } as Partial); expect(new PackingController(svc).list(user, '5')).toEqual({ items: [{ id: 1 }, { id: 2 }] }); expect(listItems).toHaveBeenCalledWith('5'); }); describe('POST /import', () => { it('400 when items is not a non-empty array (empty array)', () => { const svc = makeService(); expect(thrown(() => new PackingController(svc).importItems(user, '5', []))).toEqual({ status: 400, body: { error: 'items must be a non-empty array' }, }); }); it('400 when items is not an array at all (non-array branch)', () => { const svc = makeService(); expect(thrown(() => new PackingController(svc).importItems(user, '5', 'nope'))).toEqual({ status: 400, body: { error: 'items must be a non-empty array' }, }); }); it('403 without packing_edit permission', () => { const svc = makeService({ canEdit: vi.fn().mockReturnValue(false) }); expect(thrown(() => new PackingController(svc).importItems(user, '5', [{ name: 'a' }]))).toEqual({ status: 403, body: { error: 'No permission' }, }); }); it('imports and broadcasts per item', () => { const bulkImport = vi.fn().mockReturnValue([{ id: 1 }, { id: 2 }]); const broadcast = vi.fn(); const svc = makeService({ bulkImport, broadcast } as Partial); const res = new PackingController(svc).importItems(user, '5', [{ name: 'a' }, { name: 'b' }], 'sock'); expect(res).toEqual({ items: [{ id: 1 }, { id: 2 }], count: 2 }); expect(broadcast).toHaveBeenCalledTimes(2); }); }); describe('PUT /:id (update)', () => { it('404 when the item is missing', () => { const svc = makeService({ updateItem: vi.fn().mockReturnValue(null) } as Partial); expect(thrown(() => new PackingController(svc).update(user, '5', '9', { name: 'X' }))).toEqual({ status: 404, body: { error: 'Item not found' }, }); }); it('updates, forwards changed keys, and broadcasts', () => { const updateItem = vi.fn().mockReturnValue({ id: 9, name: 'X' }); const broadcast = vi.fn(); const svc = makeService({ updateItem, broadcast } as Partial); new PackingController(svc).update(user, '5', '9', { name: 'X', checked: true }, 'sock'); expect(updateItem).toHaveBeenCalledWith('5', '9', expect.objectContaining({ name: 'X', checked: true }), ['name', 'checked']); expect(broadcast).toHaveBeenCalledWith('5', 'packing:updated', { item: { id: 9, name: 'X' } }, 'sock'); }); }); describe('PUT /reorder', () => { it('reorders the items and reports success', () => { const reorderItems = vi.fn(); const svc = makeService({ reorderItems } as Partial); expect(new PackingController(svc).reorder(user, '5', [3, 1, 2])).toEqual({ success: true }); expect(reorderItems).toHaveBeenCalledWith('5', [3, 1, 2]); }); it('403 without packing_edit permission', () => { const svc = makeService({ canEdit: vi.fn().mockReturnValue(false) }); expect(thrown(() => new PackingController(svc).reorder(user, '5', [1]))).toEqual({ status: 403, body: { error: 'No permission' }, }); }); }); describe('DELETE /:id (remove)', () => { it('404 when the item is missing', () => { const svc = makeService({ deleteItem: vi.fn().mockReturnValue(false) } as Partial); expect(thrown(() => new PackingController(svc).remove(user, '5', '9'))).toEqual({ status: 404, body: { error: 'Item not found' }, }); }); it('deletes the item and broadcasts', () => { const deleteItem = vi.fn().mockReturnValue(true); const broadcast = vi.fn(); const svc = makeService({ deleteItem, broadcast } as Partial); expect(new PackingController(svc).remove(user, '5', '9', 'sock')).toEqual({ success: true }); expect(broadcast).toHaveBeenCalledWith('5', 'packing:deleted', { itemId: 9 }, 'sock'); }); }); describe('bags', () => { it('GET /bags lists bags for the trip', () => { const listBags = vi.fn().mockReturnValue([{ id: 3, name: 'Carry-on' }]); const svc = makeService({ listBags } as Partial); expect(new PackingController(svc).listBags(user, '5')).toEqual({ bags: [{ id: 3, name: 'Carry-on' }] }); }); it('400 on bag create with blank name', () => { const svc = makeService(); expect(thrown(() => new PackingController(svc).createBag(user, '5', { name: ' ' }))).toEqual({ status: 400, body: { error: 'Name is required' }, }); }); it('400 on bag create with no name at all (optional-chain short-circuit)', () => { const svc = makeService(); expect(thrown(() => new PackingController(svc).createBag(user, '5', {}))).toEqual({ status: 400, body: { error: 'Name is required' }, }); }); it('creates a bag and broadcasts', () => { const createBag = vi.fn().mockReturnValue({ id: 3, name: 'Carry-on' }); const broadcast = vi.fn(); const svc = makeService({ createBag, broadcast } as Partial); expect(new PackingController(svc).createBag(user, '5', { name: 'Carry-on', color: '#fff' }, 'sock')).toEqual({ bag: { id: 3, name: 'Carry-on' }, }); expect(broadcast).toHaveBeenCalledWith('5', 'packing:bag-created', { bag: { id: 3, name: 'Carry-on' } }, 'sock'); }); it('404 on bag update when missing', () => { const svc = makeService({ updateBag: vi.fn().mockReturnValue(null) } as Partial); expect(thrown(() => new PackingController(svc).updateBag(user, '5', '3', { name: 'X' }))).toEqual({ status: 404, body: { error: 'Bag not found' }, }); }); it('updates a bag, forwards changed keys and broadcasts', () => { const updateBag = vi.fn().mockReturnValue({ id: 3, name: 'X' }); const broadcast = vi.fn(); const svc = makeService({ updateBag, broadcast } as Partial); new PackingController(svc).updateBag(user, '5', '3', { name: 'X', color: '#000' }, 'sock'); expect(updateBag).toHaveBeenCalledWith('5', '3', expect.objectContaining({ name: 'X', color: '#000' }), ['name', 'color']); expect(broadcast).toHaveBeenCalledWith('5', 'packing:bag-updated', { bag: { id: 3, name: 'X' } }, 'sock'); }); it('404 on bag delete when missing', () => { const svc = makeService({ deleteBag: vi.fn().mockReturnValue(false) } as Partial); expect(thrown(() => new PackingController(svc).deleteBag(user, '5', '3'))).toEqual({ status: 404, body: { error: 'Bag not found' }, }); }); it('deletes a bag and broadcasts', () => { const deleteBag = vi.fn().mockReturnValue(true); const broadcast = vi.fn(); const svc = makeService({ deleteBag, broadcast } as Partial); expect(new PackingController(svc).deleteBag(user, '5', '3', 'sock')).toEqual({ success: true }); expect(broadcast).toHaveBeenCalledWith('5', 'packing:bag-deleted', { bagId: 3 }, 'sock'); }); it('404 on set-members when the bag is missing', () => { const svc = makeService({ setBagMembers: vi.fn().mockReturnValue(null) } as Partial); expect(thrown(() => new PackingController(svc).setBagMembers(user, '5', '3', [1, 2]))).toEqual({ status: 404, body: { error: 'Bag not found' }, }); }); it('sets bag members and broadcasts (array branch)', () => { const setBagMembers = vi.fn().mockReturnValue([{ user_id: 1 }, { user_id: 2 }]); const broadcast = vi.fn(); const svc = makeService({ setBagMembers, broadcast } as Partial); const res = new PackingController(svc).setBagMembers(user, '5', '3', [1, 2], 'sock'); expect(res).toEqual({ members: [{ user_id: 1 }, { user_id: 2 }] }); expect(setBagMembers).toHaveBeenCalledWith('5', '3', [1, 2]); expect(broadcast).toHaveBeenCalledWith('5', 'packing:bag-members-updated', { bagId: 3, members: [{ user_id: 1 }, { user_id: 2 }] }, 'sock'); }); it('coerces non-array members to an empty list (ternary else branch)', () => { const setBagMembers = vi.fn().mockReturnValue([]); const svc = makeService({ setBagMembers } as Partial); new PackingController(svc).setBagMembers(user, '5', '3', 'not-an-array'); expect(setBagMembers).toHaveBeenCalledWith('5', '3', []); }); }); describe('templates', () => { it('GET /templates returns the template list for an accessible trip', () => { const listTemplates = vi.fn().mockReturnValue([{ id: 1, name: 'Beach', item_count: 4 }]); const svc = makeService({ listTemplates } as Partial); expect(new PackingController(svc).listTemplates(user, '5')).toEqual({ templates: [{ id: 1, name: 'Beach', item_count: 4 }], }); }); it('404 when applying a missing/empty template (POST stays 200 otherwise)', () => { const svc = makeService({ applyTemplate: vi.fn().mockReturnValue(null) } as Partial); expect(thrown(() => new PackingController(svc).applyTemplate(user, '5', 't1'))).toEqual({ status: 404, body: { error: 'Template not found or empty' }, }); }); it('applies a template, broadcasts the added items and reports the count', () => { const applyTemplate = vi.fn().mockReturnValue([{ id: 1 }, { id: 2 }, { id: 3 }]); const broadcast = vi.fn(); const svc = makeService({ applyTemplate, broadcast } as Partial); const res = new PackingController(svc).applyTemplate(user, '5', 't1', 'sock'); expect(res).toEqual({ items: [{ id: 1 }, { id: 2 }, { id: 3 }], count: 3 }); expect(broadcast).toHaveBeenCalledWith('5', 'packing:template-applied', { items: [{ id: 1 }, { id: 2 }, { id: 3 }] }, 'sock'); }); it('400 when an admin saves a template with no name (whitespace)', () => { const saveAsTemplate = vi.fn(); const svc = makeService({ saveAsTemplate } as Partial); expect(thrown(() => new PackingController(svc).saveAsTemplate(admin, '5', ' '))).toEqual({ status: 400, body: { error: 'Template name is required' }, }); expect(saveAsTemplate).not.toHaveBeenCalled(); }); it('400 when an admin saves a template with no name at all (optional-chain)', () => { const saveAsTemplate = vi.fn(); const svc = makeService({ saveAsTemplate } as Partial); expect(thrown(() => new PackingController(svc).saveAsTemplate(admin, '5'))).toEqual({ status: 400, body: { error: 'Template name is required' }, }); expect(saveAsTemplate).not.toHaveBeenCalled(); }); it('403 when a non-admin tries to save a template', () => { const saveAsTemplate = vi.fn(); const svc = makeService({ saveAsTemplate } as Partial); expect(thrown(() => new PackingController(svc).saveAsTemplate(user, '5', 'My template'))).toEqual({ status: 403, body: { error: 'Admin access required' }, }); expect(saveAsTemplate).not.toHaveBeenCalled(); }); it('400 when an admin saves a template with no items', () => { const svc = makeService({ saveAsTemplate: vi.fn().mockReturnValue(null) } as Partial); expect(thrown(() => new PackingController(svc).saveAsTemplate(admin, '5', 'My template'))).toEqual({ status: 400, body: { error: 'No items to save' }, }); }); it('saves a template for an admin', () => { const saveAsTemplate = vi.fn().mockReturnValue({ id: 7, name: 'My template' }); const svc = makeService({ saveAsTemplate } as Partial); expect(new PackingController(svc).saveAsTemplate(admin, '5', 'My template')).toEqual({ template: { id: 7, name: 'My template' }, }); expect(saveAsTemplate).toHaveBeenCalledWith('5', admin.id, 'My template'); }); }); describe('category assignees', () => { it('GET /category-assignees returns the assignee list for an accessible trip', () => { const getCategoryAssignees = vi.fn().mockReturnValue([{ category: 'Clothes', user_id: 2 }]); const svc = makeService({ getCategoryAssignees } as Partial); expect(new PackingController(svc).categoryAssignees(user, '5')).toEqual({ assignees: [{ category: 'Clothes', user_id: 2 }], }); expect(getCategoryAssignees).toHaveBeenCalledWith('5'); }); it('decodes the URI-encoded category name before forwarding', () => { const updateCategoryAssignees = vi.fn().mockReturnValue([]); const broadcast = vi.fn(); const notifyTagged = vi.fn(); const svc = makeService({ updateCategoryAssignees, broadcast, notifyTagged } as Partial); new PackingController(svc).updateCategoryAssignees(user, '5', 'Toys%20%26%20Games', [2]); expect(updateCategoryAssignees).toHaveBeenCalledWith('5', 'Toys & Games', [2]); }); it('updates assignees, broadcasts and fires the tag notification', () => { const updateCategoryAssignees = vi.fn().mockReturnValue([{ user_id: 2 }]); const broadcast = vi.fn(); const notifyTagged = vi.fn(); const svc = makeService({ updateCategoryAssignees, broadcast, notifyTagged } as Partial); const res = new PackingController(svc).updateCategoryAssignees(user, '5', 'Clothes', [2], 'sock'); expect(res).toEqual({ assignees: [{ user_id: 2 }] }); expect(broadcast).toHaveBeenCalledWith('5', 'packing:assignees', { category: 'Clothes', assignees: [{ user_id: 2 }] }, 'sock'); expect(notifyTagged).toHaveBeenCalledWith('5', user, 'Clothes', [2]); }); }); });