StreamOverlayPage.test.tsx 9.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368
  1. /**
  2. * Tests for the StreamOverlayPage component.
  3. */
  4. import { describe, it, expect, beforeEach, afterEach, vi } from 'vitest';
  5. import { screen, waitFor, render as rtlRender } from '@testing-library/react';
  6. import { StreamOverlayPage } from '../../pages/StreamOverlayPage';
  7. import { http, HttpResponse } from 'msw';
  8. import { server } from '../mocks/server';
  9. import { MemoryRouter, Route, Routes } from 'react-router-dom';
  10. import { QueryClient, QueryClientProvider } from '@tanstack/react-query';
  11. import { ThemeProvider } from '../../contexts/ThemeContext';
  12. import { ToastProvider } from '../../contexts/ToastContext';
  13. const mockPrinter = {
  14. id: 1,
  15. name: 'X1 Carbon',
  16. ip_address: '192.168.1.100',
  17. serial_number: '00M09A350100001',
  18. access_code: '12345678',
  19. model: 'X1C',
  20. enabled: true,
  21. };
  22. const mockStatusIdle = {
  23. id: 1,
  24. name: 'X1 Carbon',
  25. connected: true,
  26. state: 'IDLE',
  27. progress: 0,
  28. current_print: null,
  29. remaining_time: null,
  30. layer_num: null,
  31. total_layers: null,
  32. stg_cur_name: null,
  33. };
  34. const mockStatusPrinting = {
  35. id: 1,
  36. name: 'X1 Carbon',
  37. connected: true,
  38. state: 'RUNNING',
  39. progress: 45,
  40. current_print: 'Benchy.gcode.3mf',
  41. remaining_time: 82,
  42. layer_num: 150,
  43. total_layers: 300,
  44. stg_cur_name: null,
  45. };
  46. // Custom render for StreamOverlayPage
  47. function renderOverlayPage(printerId: number, queryParams = '') {
  48. const queryClient = new QueryClient({
  49. defaultOptions: {
  50. queries: { retry: false, gcTime: 0 },
  51. mutations: { retry: false },
  52. },
  53. });
  54. return rtlRender(
  55. <QueryClientProvider client={queryClient}>
  56. <MemoryRouter initialEntries={[`/overlay/${printerId}${queryParams}`]}>
  57. <ThemeProvider>
  58. <ToastProvider>
  59. <Routes>
  60. <Route path="/overlay/:printerId" element={<StreamOverlayPage />} />
  61. </Routes>
  62. </ToastProvider>
  63. </ThemeProvider>
  64. </MemoryRouter>
  65. </QueryClientProvider>
  66. );
  67. }
  68. describe('StreamOverlayPage', () => {
  69. const originalTitle = document.title;
  70. beforeEach(() => {
  71. // Mock WebSocket. vitest 4 dropped support for arrow-function constructor
  72. // mocks (`new (() => ...)` throws "is not a constructor"); use a plain
  73. // function so `new WebSocket(...)` resolves correctly.
  74. vi.stubGlobal(
  75. 'WebSocket',
  76. vi.fn().mockImplementation(function (this: { close: () => void; onmessage: null; onerror: null }) {
  77. this.close = vi.fn();
  78. this.onmessage = null;
  79. this.onerror = null;
  80. }),
  81. );
  82. server.use(
  83. http.get('/api/v1/printers/:id', () => {
  84. return HttpResponse.json(mockPrinter);
  85. }),
  86. http.get('/api/v1/printers/:id/status', () => {
  87. return HttpResponse.json(mockStatusIdle);
  88. })
  89. );
  90. });
  91. afterEach(() => {
  92. document.title = originalTitle;
  93. vi.unstubAllGlobals();
  94. });
  95. describe('rendering', () => {
  96. it('renders overlay page for printer', async () => {
  97. renderOverlayPage(1);
  98. await waitFor(() => {
  99. expect(screen.getByText('Printer is idle')).toBeInTheDocument();
  100. });
  101. });
  102. it('shows Bambuddy logo', async () => {
  103. renderOverlayPage(1);
  104. await waitFor(() => {
  105. expect(screen.getByAltText('Bambuddy')).toBeInTheDocument();
  106. });
  107. });
  108. it('logo links to GitHub', async () => {
  109. renderOverlayPage(1);
  110. await waitFor(() => {
  111. const logo = screen.getByAltText('Bambuddy');
  112. const link = logo.closest('a');
  113. expect(link).toHaveAttribute('href', 'https://github.com/maziggy/bambuddy');
  114. });
  115. });
  116. });
  117. describe('printing state', () => {
  118. beforeEach(() => {
  119. server.use(
  120. http.get('/api/v1/printers/:id/status', () => {
  121. return HttpResponse.json(mockStatusPrinting);
  122. })
  123. );
  124. });
  125. it('shows filename when printing', async () => {
  126. renderOverlayPage(1);
  127. await waitFor(() => {
  128. expect(screen.getByText('Benchy')).toBeInTheDocument();
  129. });
  130. });
  131. it('shows progress percentage', async () => {
  132. renderOverlayPage(1);
  133. await waitFor(() => {
  134. expect(screen.getByText('45%')).toBeInTheDocument();
  135. });
  136. });
  137. it('shows layer count', async () => {
  138. renderOverlayPage(1);
  139. await waitFor(() => {
  140. expect(screen.getByText('150')).toBeInTheDocument();
  141. expect(screen.getByText('300')).toBeInTheDocument();
  142. });
  143. });
  144. it('shows status text', async () => {
  145. renderOverlayPage(1);
  146. await waitFor(() => {
  147. expect(screen.getByText('Printing')).toBeInTheDocument();
  148. });
  149. });
  150. });
  151. describe('invalid printer', () => {
  152. it('shows invalid printer message for ID 0', async () => {
  153. renderOverlayPage(0);
  154. await waitFor(() => {
  155. expect(screen.getByText('Invalid printer ID')).toBeInTheDocument();
  156. });
  157. });
  158. });
  159. describe('query parameters', () => {
  160. it('respects size parameter', async () => {
  161. renderOverlayPage(1, '?size=large');
  162. await waitFor(() => {
  163. // Just verify it renders without error
  164. expect(screen.getByAltText('Bambuddy')).toBeInTheDocument();
  165. });
  166. });
  167. it('respects show parameter to hide elements', async () => {
  168. server.use(
  169. http.get('/api/v1/printers/:id/status', () => {
  170. return HttpResponse.json(mockStatusPrinting);
  171. })
  172. );
  173. renderOverlayPage(1, '?show=progress');
  174. await waitFor(() => {
  175. // Progress should be visible
  176. expect(screen.getByText('45%')).toBeInTheDocument();
  177. // Status text should be hidden when not in show list
  178. expect(screen.queryByText('Printing')).not.toBeInTheDocument();
  179. });
  180. });
  181. });
  182. describe('FPS configuration', () => {
  183. it('uses default FPS of 15 when not specified', async () => {
  184. renderOverlayPage(1);
  185. await waitFor(() => {
  186. const img = screen.getByAltText('Camera stream') as HTMLImageElement;
  187. expect(img.src).toContain('fps=15');
  188. });
  189. });
  190. it('uses custom FPS when specified in query params', async () => {
  191. renderOverlayPage(1, '?fps=30');
  192. await waitFor(() => {
  193. const img = screen.getByAltText('Camera stream') as HTMLImageElement;
  194. expect(img.src).toContain('fps=30');
  195. });
  196. });
  197. it('clamps FPS to maximum of 30', async () => {
  198. renderOverlayPage(1, '?fps=60');
  199. await waitFor(() => {
  200. const img = screen.getByAltText('Camera stream') as HTMLImageElement;
  201. expect(img.src).toContain('fps=30');
  202. });
  203. });
  204. it('clamps FPS to minimum of 1', async () => {
  205. renderOverlayPage(1, '?fps=0');
  206. await waitFor(() => {
  207. const img = screen.getByAltText('Camera stream') as HTMLImageElement;
  208. expect(img.src).toContain('fps=1');
  209. });
  210. });
  211. it('handles invalid FPS value gracefully', async () => {
  212. renderOverlayPage(1, '?fps=invalid');
  213. await waitFor(() => {
  214. const img = screen.getByAltText('Camera stream') as HTMLImageElement;
  215. // Should fall back to default of 15
  216. expect(img.src).toContain('fps=15');
  217. });
  218. });
  219. });
  220. describe('camera toggle (status-only mode)', () => {
  221. it('shows camera by default', async () => {
  222. renderOverlayPage(1);
  223. await waitFor(() => {
  224. expect(screen.getByAltText('Camera stream')).toBeInTheDocument();
  225. });
  226. });
  227. it('hides camera when camera=false', async () => {
  228. renderOverlayPage(1, '?camera=false');
  229. await waitFor(() => {
  230. // Status should still be visible
  231. expect(screen.getByText('Printer is idle')).toBeInTheDocument();
  232. });
  233. // Camera should not be rendered
  234. expect(screen.queryByAltText('Camera stream')).not.toBeInTheDocument();
  235. });
  236. it('hides camera when camera=0', async () => {
  237. renderOverlayPage(1, '?camera=0');
  238. await waitFor(() => {
  239. expect(screen.getByText('Printer is idle')).toBeInTheDocument();
  240. });
  241. expect(screen.queryByAltText('Camera stream')).not.toBeInTheDocument();
  242. });
  243. it('shows camera when camera=true', async () => {
  244. renderOverlayPage(1, '?camera=true');
  245. await waitFor(() => {
  246. expect(screen.getByAltText('Camera stream')).toBeInTheDocument();
  247. });
  248. });
  249. it('shows camera when camera=1', async () => {
  250. renderOverlayPage(1, '?camera=1');
  251. await waitFor(() => {
  252. expect(screen.getByAltText('Camera stream')).toBeInTheDocument();
  253. });
  254. });
  255. });
  256. describe('combined parameters', () => {
  257. it('supports fps and camera together', async () => {
  258. renderOverlayPage(1, '?fps=25&camera=true');
  259. await waitFor(() => {
  260. const img = screen.getByAltText('Camera stream') as HTMLImageElement;
  261. expect(img.src).toContain('fps=25');
  262. });
  263. });
  264. it('supports status-only with custom size', async () => {
  265. renderOverlayPage(1, '?camera=false&size=large');
  266. await waitFor(() => {
  267. expect(screen.getByText('Printer is idle')).toBeInTheDocument();
  268. });
  269. expect(screen.queryByAltText('Camera stream')).not.toBeInTheDocument();
  270. });
  271. it('supports show parameter with fps', async () => {
  272. server.use(
  273. http.get('/api/v1/printers/:id/status', () => {
  274. return HttpResponse.json(mockStatusPrinting);
  275. })
  276. );
  277. renderOverlayPage(1, '?fps=20&show=progress');
  278. await waitFor(() => {
  279. const img = screen.getByAltText('Camera stream') as HTMLImageElement;
  280. expect(img.src).toContain('fps=20');
  281. expect(screen.getByText('45%')).toBeInTheDocument();
  282. });
  283. });
  284. });
  285. describe('offline state', () => {
  286. beforeEach(() => {
  287. server.use(
  288. http.get('/api/v1/printers/:id/status', () => {
  289. return HttpResponse.json({
  290. ...mockStatusIdle,
  291. connected: false,
  292. });
  293. })
  294. );
  295. });
  296. it('shows offline message when printer disconnected', async () => {
  297. renderOverlayPage(1);
  298. await waitFor(() => {
  299. expect(screen.getByText('Printer offline')).toBeInTheDocument();
  300. });
  301. });
  302. });
  303. });