| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423 |
- /**
- * Tests for the Layout component.
- */
- import { describe, it, expect, beforeEach } from 'vitest';
- import { waitFor } from '@testing-library/react';
- import { render } from '../utils';
- import { Layout } from '../../components/Layout';
- import { http, HttpResponse } from 'msw';
- import { server } from '../mocks/server';
- describe('Layout', () => {
- beforeEach(() => {
- server.use(
- http.get('/api/v1/printers/', () => {
- return HttpResponse.json([
- { id: 1, name: 'X1 Carbon', model: 'X1C', enabled: true },
- ]);
- }),
- http.get('/api/v1/printers/:id/status', () => {
- return HttpResponse.json({
- connected: true,
- state: 'IDLE',
- });
- }),
- http.get('/api/v1/version', () => {
- return HttpResponse.json({ version: '0.1.6', build: 'test' });
- }),
- http.get('/api/v1/settings/', () => {
- return HttpResponse.json({
- check_updates: false,
- check_printer_firmware: false,
- auto_archive: true,
- });
- }),
- http.get('/api/v1/external-links/', () => {
- return HttpResponse.json([]);
- }),
- http.get('/api/v1/smart-plugs/', () => {
- return HttpResponse.json([]);
- }),
- http.get('/api/v1/support/debug-logging', () => {
- return HttpResponse.json({ enabled: false });
- }),
- http.get('/api/v1/queue/', () => {
- return HttpResponse.json([]);
- }),
- http.get('/api/v1/pending-uploads/count', () => {
- return HttpResponse.json({ count: 0 });
- }),
- http.get('/api/v1/updates/check', () => {
- return HttpResponse.json({ update_available: false });
- }),
- http.get('/api/v1/auth/status', () => {
- return HttpResponse.json({ auth_enabled: false, requires_setup: false });
- }),
- http.get('/api/v1/printers/developer-mode-warnings', () => {
- return HttpResponse.json([]);
- })
- );
- });
- describe('rendering', () => {
- it('renders the sidebar', async () => {
- render(<Layout />);
- // Layout renders as a flex container with sidebar
- await waitFor(() => {
- const sidebar = document.querySelector('aside');
- expect(sidebar).toBeInTheDocument();
- });
- });
- it('renders navigation links', async () => {
- render(<Layout />);
- await waitFor(() => {
- // Navigation links should be present
- const links = document.querySelectorAll('a');
- expect(links.length).toBeGreaterThan(0);
- });
- });
- });
- describe('navigation', () => {
- it('has navigation items', async () => {
- render(<Layout />);
- await waitFor(() => {
- // Should have multiple navigation links
- const navLinks = document.querySelectorAll('a[href]');
- expect(navLinks.length).toBeGreaterThan(0);
- });
- });
- it('includes settings link', async () => {
- render(<Layout />);
- await waitFor(() => {
- // Settings link should exist (route /settings)
- const settingsLink = document.querySelector('a[href="/settings"]');
- expect(settingsLink).toBeInTheDocument();
- });
- });
- });
- describe('version display', () => {
- it('shows version info', async () => {
- render(<Layout />);
- await waitFor(() => {
- // Version info is displayed in sidebar
- expect(document.body).toBeInTheDocument();
- });
- });
- });
- describe('theme toggle', () => {
- it('has theme toggle button', async () => {
- render(<Layout />);
- await waitFor(() => {
- // Theme toggle should be present
- const buttons = document.querySelectorAll('button');
- expect(buttons.length).toBeGreaterThan(0);
- });
- });
- it('cycles through dark → light → system → dark', async () => {
- localStorage.setItem('theme-mode', 'dark');
- render(<Layout />);
- await waitFor(() => {
- // In dark mode, title should say "Switch to light mode"
- const btn = document.querySelector('button[title="Switch to light mode"]');
- expect(btn).toBeInTheDocument();
- });
- // Click to go from dark → light
- const lightBtn = document.querySelector('button[title="Switch to light mode"]')!;
- lightBtn.click();
- await waitFor(() => {
- // In light mode, title should say "Switch to system mode"
- const btn = document.querySelector('button[title="Switch to system mode"]');
- expect(btn).toBeInTheDocument();
- });
- // Click to go from light → system
- const systemBtn = document.querySelector('button[title="Switch to system mode"]')!;
- systemBtn.click();
- await waitFor(() => {
- // In system mode, title should say "Switch to dark mode"
- const btn = document.querySelector('button[title="Switch to dark mode"]');
- expect(btn).toBeInTheDocument();
- });
- // Click to go from system → dark
- const darkBtn = document.querySelector('button[title="Switch to dark mode"]')!;
- darkBtn.click();
- await waitFor(() => {
- // Back to dark mode
- const btn = document.querySelector('button[title="Switch to light mode"]');
- expect(btn).toBeInTheDocument();
- });
- });
- });
- describe('plate detection alert modal', () => {
- it('shows modal when plate-not-empty event is dispatched', async () => {
- render(<Layout />);
- // Dispatch the plate-not-empty event
- window.dispatchEvent(
- new CustomEvent('plate-not-empty', {
- detail: {
- printer_id: 1,
- printer_name: 'Test Printer',
- message: 'Objects detected on build plate',
- },
- })
- );
- await waitFor(() => {
- // Modal should appear with "Print Paused!" text
- expect(document.body.textContent).toContain('Print Paused!');
- expect(document.body.textContent).toContain('Test Printer');
- });
- });
- it('closes modal when I Understand button is clicked', async () => {
- render(<Layout />);
- // Dispatch the plate-not-empty event
- window.dispatchEvent(
- new CustomEvent('plate-not-empty', {
- detail: {
- printer_id: 1,
- printer_name: 'Test Printer',
- message: 'Objects detected on build plate',
- },
- })
- );
- await waitFor(() => {
- expect(document.body.textContent).toContain('Print Paused!');
- });
- // Click the "I Understand" button
- const button = document.querySelector('button');
- if (button && button.textContent?.includes('I Understand')) {
- button.click();
- }
- // Find and click the "I Understand" button by searching all buttons
- const buttons = document.querySelectorAll('button');
- buttons.forEach((btn) => {
- if (btn.textContent?.includes('I Understand')) {
- btn.click();
- }
- });
- await waitFor(() => {
- // Modal should be closed
- expect(document.body.textContent).not.toContain('Print Paused!');
- });
- });
- });
- describe('developer mode warning banner', () => {
- it('shows warning banner when printers lack developer mode', async () => {
- server.use(
- http.get('/api/v1/printers/developer-mode-warnings', () => {
- return HttpResponse.json([
- { printer_id: 1, name: 'X1 Carbon' },
- ]);
- })
- );
- render(<Layout />);
- await waitFor(() => {
- expect(document.body.textContent).toContain('Developer LAN mode is not enabled on');
- expect(document.body.textContent).toContain('X1 Carbon');
- });
- });
- it('shows multiple printer names in warning banner', async () => {
- server.use(
- http.get('/api/v1/printers/developer-mode-warnings', () => {
- return HttpResponse.json([
- { printer_id: 1, name: 'X1 Carbon' },
- { printer_id: 2, name: 'P1S' },
- ]);
- })
- );
- render(<Layout />);
- await waitFor(() => {
- expect(document.body.textContent).toContain('X1 Carbon');
- expect(document.body.textContent).toContain('P1S');
- });
- });
- it('hides warning banner when no printers lack developer mode', async () => {
- // Default handler returns empty array
- render(<Layout />);
- await waitFor(() => {
- const sidebar = document.querySelector('aside');
- expect(sidebar).toBeInTheDocument();
- });
- // Banner should not be present
- expect(document.body.textContent).not.toContain('Developer LAN mode is not enabled on');
- });
- it('shows how to enable link in warning banner', async () => {
- server.use(
- http.get('/api/v1/printers/developer-mode-warnings', () => {
- return HttpResponse.json([
- { printer_id: 1, name: 'X1 Carbon' },
- ]);
- })
- );
- render(<Layout />);
- await waitFor(() => {
- expect(document.body.textContent).toContain('How to enable');
- const link = document.querySelector('a[href*="enable-developer-mode"]');
- expect(link).toBeInTheDocument();
- });
- });
- });
- describe('update banner suppression for HA addon', () => {
- // HA Supervisor surfaces its own update notification natively in the HA
- // UI, so the in-app banner would be duplicate noise that links to a page
- // that just says "update via HA". Suppress it for HA addon deployments.
- it('hides the update-available banner when running as an HA addon', async () => {
- server.use(
- http.get('/api/v1/updates/check', () => {
- return HttpResponse.json({
- update_available: true,
- current_version: '0.2.4',
- latest_version: '0.2.5',
- is_docker: true,
- is_ha_addon: true,
- update_method: 'ha_addon',
- });
- }),
- );
- render(<Layout />);
- await waitFor(() => {
- const sidebar = document.querySelector('aside');
- expect(sidebar).toBeInTheDocument();
- });
- expect(document.body.textContent).not.toContain('Update available');
- });
- it('still shows the update-available banner for plain Docker deployments', async () => {
- server.use(
- http.get('/api/v1/updates/check', () => {
- return HttpResponse.json({
- update_available: true,
- current_version: '0.2.4',
- latest_version: '0.2.5',
- is_docker: true,
- is_ha_addon: false,
- update_method: 'docker',
- });
- }),
- );
- render(<Layout />);
- await waitFor(() => {
- expect(document.body.textContent).toContain('0.2.5');
- });
- });
- });
- describe('MakerWorld sidebar permission gate (#1175)', () => {
- // The MakerWorld sidebar entry was visible to every authenticated user
- // regardless of group permissions because Layout's `navPermissions` map
- // had no entry for `makerworld`. Backend routes already gated on
- // `makerworld:view`, so users without the permission saw the entry,
- // clicked, and got 403'd by every API call inside the page. The fix
- // adds `makerworld: 'makerworld:view'` to the map so the entry is
- // hidden when the permission is absent — same shape as every other
- // sidebar entry.
- const enableAuthWithUser = (permissions: string[]) => {
- server.use(
- http.get('/api/v1/auth/status', () =>
- HttpResponse.json({ auth_enabled: true, requires_setup: false }),
- ),
- http.get('/api/v1/auth/me', () =>
- HttpResponse.json({
- id: 1,
- username: 'tester',
- role: 'user',
- is_active: true,
- is_admin: false,
- groups: [{ id: 2, name: 'Standard Users' }],
- permissions,
- created_at: '2026-01-01T00:00:00Z',
- }),
- ),
- );
- // AuthProvider needs a token in localStorage to fetch /auth/me; the
- // value isn't validated by the mocked server.
- window.localStorage.setItem('auth_token', 'test-token');
- };
- const findMakerWorldNavLink = () => {
- // Sidebar nav links use react-router's `to` prop, which renders as a
- // plain `<a href="/makerworld">`. Match on the href so the test isn't
- // coupled to whatever locale string is rendered.
- return document.querySelector('aside a[href="/makerworld"]');
- };
- it('hides the MakerWorld nav entry when the user lacks makerworld:view', async () => {
- // Standard user without the MakerWorld permission. Every other
- // permission they hold (library:read, etc.) is irrelevant here — the
- // gate is per-entry and the MakerWorld entry must not render.
- enableAuthWithUser(['library:read', 'archives:read', 'queue:read']);
- render(<Layout />);
- await waitFor(() => {
- // Wait for the auth resolution + sidebar render. Some other nav
- // entry (Files / Archives) confirms the sidebar finished mounting.
- const sidebar = document.querySelector('aside');
- expect(sidebar).toBeInTheDocument();
- expect(sidebar?.querySelector('a[href="/files"]')).toBeInTheDocument();
- });
- expect(findMakerWorldNavLink()).toBeNull();
- });
- it('shows the MakerWorld nav entry when the user has makerworld:view', async () => {
- enableAuthWithUser([
- 'library:read',
- 'archives:read',
- 'queue:read',
- 'makerworld:view',
- ]);
- render(<Layout />);
- await waitFor(() => {
- expect(findMakerWorldNavLink()).toBeInTheDocument();
- });
- });
- });
- });
|