PrintersPage.test.tsx 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311
  1. /**
  2. * Tests for the PrintersPage component.
  3. */
  4. import { describe, it, expect, beforeEach } from 'vitest';
  5. import { screen, waitFor } from '@testing-library/react';
  6. import { render } from '../utils';
  7. import { PrintersPage } from '../../pages/PrintersPage';
  8. import { http, HttpResponse } from 'msw';
  9. import { server } from '../mocks/server';
  10. const mockPrinters = [
  11. {
  12. id: 1,
  13. name: 'X1 Carbon',
  14. ip_address: '192.168.1.100',
  15. serial_number: '00M09A350100001',
  16. access_code: '12345678',
  17. model: 'X1C',
  18. enabled: true,
  19. nozzle_diameter: 0.4,
  20. nozzle_type: 'hardened_steel',
  21. location: 'Workshop',
  22. auto_archive: true,
  23. created_at: '2024-01-01T00:00:00Z',
  24. updated_at: '2024-01-01T00:00:00Z',
  25. },
  26. {
  27. id: 2,
  28. name: 'P1S Backup',
  29. ip_address: '192.168.1.101',
  30. serial_number: '00W00A123456789',
  31. access_code: '87654321',
  32. model: 'P1S',
  33. enabled: false,
  34. nozzle_diameter: 0.4,
  35. nozzle_type: 'stainless_steel',
  36. location: null,
  37. auto_archive: true,
  38. created_at: '2024-01-02T00:00:00Z',
  39. updated_at: '2024-01-02T00:00:00Z',
  40. },
  41. ];
  42. const mockPrinterStatus = {
  43. connected: true,
  44. state: 'IDLE',
  45. progress: 0,
  46. layer_num: 0,
  47. total_layers: 0,
  48. temperatures: {
  49. nozzle: 25,
  50. bed: 25,
  51. chamber: 25,
  52. },
  53. remaining_time: 0,
  54. filename: null,
  55. wifi_signal: -50,
  56. };
  57. describe('PrintersPage', () => {
  58. beforeEach(() => {
  59. server.use(
  60. http.get('/api/v1/printers/', () => {
  61. return HttpResponse.json(mockPrinters);
  62. }),
  63. http.get('/api/v1/printers/:id/status', () => {
  64. return HttpResponse.json(mockPrinterStatus);
  65. }),
  66. http.get('/api/v1/queue/', () => {
  67. return HttpResponse.json([]);
  68. })
  69. );
  70. });
  71. describe('rendering', () => {
  72. it('renders the page title', async () => {
  73. render(<PrintersPage />);
  74. await waitFor(() => {
  75. expect(screen.getByText('Printers')).toBeInTheDocument();
  76. });
  77. });
  78. it('shows printer cards', async () => {
  79. render(<PrintersPage />);
  80. await waitFor(() => {
  81. expect(screen.getByText('X1 Carbon')).toBeInTheDocument();
  82. expect(screen.getByText('P1S Backup')).toBeInTheDocument();
  83. });
  84. });
  85. it('shows printer models', async () => {
  86. render(<PrintersPage />);
  87. await waitFor(() => {
  88. expect(screen.getByText('X1C')).toBeInTheDocument();
  89. expect(screen.getByText('P1S')).toBeInTheDocument();
  90. });
  91. });
  92. it('shows printer status', async () => {
  93. render(<PrintersPage />);
  94. await waitFor(() => {
  95. // Status should be shown - may vary based on state
  96. expect(screen.getByText('X1 Carbon')).toBeInTheDocument();
  97. });
  98. });
  99. });
  100. describe('printer info', () => {
  101. it('shows IP address', async () => {
  102. render(<PrintersPage />);
  103. await waitFor(() => {
  104. expect(screen.getByText('192.168.1.100')).toBeInTheDocument();
  105. });
  106. });
  107. it('shows location when set', async () => {
  108. render(<PrintersPage />);
  109. await waitFor(() => {
  110. // Printers should render - location display may vary
  111. expect(screen.getByText('X1 Carbon')).toBeInTheDocument();
  112. });
  113. });
  114. });
  115. describe('temperature display', () => {
  116. it('shows nozzle temperature', async () => {
  117. render(<PrintersPage />);
  118. await waitFor(() => {
  119. // Temperatures are shown in the UI
  120. expect(screen.getAllByText(/25/)).toBeTruthy();
  121. });
  122. });
  123. });
  124. describe('empty state', () => {
  125. it('shows empty state when no printers', async () => {
  126. server.use(
  127. http.get('/api/v1/printers/', () => {
  128. return HttpResponse.json([]);
  129. })
  130. );
  131. render(<PrintersPage />);
  132. await waitFor(() => {
  133. expect(screen.getByText(/no printers/i)).toBeInTheDocument();
  134. });
  135. });
  136. });
  137. describe('printer actions', () => {
  138. it('has action buttons', async () => {
  139. render(<PrintersPage />);
  140. await waitFor(() => {
  141. expect(screen.getByText('X1 Carbon')).toBeInTheDocument();
  142. });
  143. // There should be some interactive elements for printer actions
  144. const buttons = screen.getAllByRole('button');
  145. expect(buttons.length).toBeGreaterThan(0);
  146. });
  147. });
  148. describe('disabled printer', () => {
  149. it('shows disabled state for disabled printers', async () => {
  150. render(<PrintersPage />);
  151. await waitFor(() => {
  152. expect(screen.getByText('P1S Backup')).toBeInTheDocument();
  153. });
  154. // Disabled printers have visual indication
  155. const disabledPrinter = screen.getByText('P1S Backup').closest('div');
  156. expect(disabledPrinter).toBeInTheDocument();
  157. });
  158. });
  159. describe('firmware version badge', () => {
  160. const firmwareUpToDate = {
  161. printer_id: 1,
  162. current_version: '01.09.00.00',
  163. latest_version: '01.09.00.00',
  164. update_available: false,
  165. download_url: null,
  166. release_notes: 'Bug fixes and improvements.',
  167. };
  168. const firmwareUpdateAvailable = {
  169. printer_id: 1,
  170. current_version: '01.08.00.00',
  171. latest_version: '01.09.00.00',
  172. update_available: true,
  173. download_url: 'https://example.com/firmware.bin',
  174. release_notes: 'New features added.',
  175. };
  176. it('shows green badge when firmware is up to date', async () => {
  177. server.use(
  178. http.get('/api/v1/firmware/updates/:id', () => {
  179. return HttpResponse.json(firmwareUpToDate);
  180. }),
  181. http.get('/api/v1/settings/', () => {
  182. return HttpResponse.json({
  183. check_printer_firmware: true,
  184. auto_archive: true,
  185. save_thumbnails: true,
  186. });
  187. })
  188. );
  189. render(<PrintersPage />);
  190. await waitFor(() => {
  191. expect(screen.getAllByText('01.09.00.00').length).toBeGreaterThan(0);
  192. });
  193. const badge = screen.getAllByText('01.09.00.00')[0].closest('button');
  194. expect(badge).toBeInTheDocument();
  195. expect(badge?.className).toContain('text-status-ok');
  196. });
  197. it('shows orange badge when firmware update is available', async () => {
  198. server.use(
  199. http.get('/api/v1/firmware/updates/:id', () => {
  200. return HttpResponse.json(firmwareUpdateAvailable);
  201. }),
  202. http.get('/api/v1/settings/', () => {
  203. return HttpResponse.json({
  204. check_printer_firmware: true,
  205. auto_archive: true,
  206. save_thumbnails: true,
  207. });
  208. })
  209. );
  210. render(<PrintersPage />);
  211. await waitFor(() => {
  212. expect(screen.getAllByText('01.08.00.00').length).toBeGreaterThan(0);
  213. });
  214. const badge = screen.getAllByText('01.08.00.00')[0].closest('button');
  215. expect(badge).toBeInTheDocument();
  216. expect(badge?.className).toContain('text-orange-400');
  217. });
  218. it('hides badge when firmware check is disabled', async () => {
  219. server.use(
  220. http.get('/api/v1/settings/', () => {
  221. return HttpResponse.json({
  222. check_printer_firmware: false,
  223. auto_archive: true,
  224. save_thumbnails: true,
  225. });
  226. })
  227. );
  228. render(<PrintersPage />);
  229. await waitFor(() => {
  230. expect(screen.getByText('X1 Carbon')).toBeInTheDocument();
  231. });
  232. // Version should not appear when firmware check is disabled
  233. expect(screen.queryByText('01.09.00.00')).not.toBeInTheDocument();
  234. expect(screen.queryByText('01.08.00.00')).not.toBeInTheDocument();
  235. });
  236. it('hides badge when API has no firmware data for the model', async () => {
  237. const firmwareNoData = {
  238. printer_id: 1,
  239. current_version: '01.01.03.00',
  240. latest_version: null,
  241. update_available: false,
  242. download_url: null,
  243. release_notes: null,
  244. };
  245. server.use(
  246. http.get('/api/v1/firmware/updates/:id', () => {
  247. return HttpResponse.json(firmwareNoData);
  248. }),
  249. http.get('/api/v1/settings/', () => {
  250. return HttpResponse.json({
  251. check_printer_firmware: true,
  252. auto_archive: true,
  253. save_thumbnails: true,
  254. });
  255. })
  256. );
  257. render(<PrintersPage />);
  258. await waitFor(() => {
  259. expect(screen.getByText('X1 Carbon')).toBeInTheDocument();
  260. });
  261. // Badge should not appear when API returns no latest_version
  262. expect(screen.queryByText('01.01.03.00')).not.toBeInTheDocument();
  263. });
  264. });
  265. });