FileManagerModal.test.tsx 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417
  1. /**
  2. * Tests for the FileManagerModal component.
  3. * Tests file browsing, selection, navigation, and file operations.
  4. */
  5. import { describe, it, expect, vi, beforeEach } from 'vitest';
  6. import { screen, fireEvent, waitFor } from '@testing-library/react';
  7. import { render } from '../utils';
  8. import { FileManagerModal } from '../../components/FileManagerModal';
  9. import { http, HttpResponse } from 'msw';
  10. import { server } from '../mocks/server';
  11. const mockFiles = [
  12. {
  13. name: 'cache',
  14. path: '/cache',
  15. size: 0,
  16. is_directory: true,
  17. mtime: '2024-01-15T10:00:00Z',
  18. },
  19. {
  20. name: 'model',
  21. path: '/model',
  22. size: 0,
  23. is_directory: true,
  24. mtime: '2024-01-15T10:00:00Z',
  25. },
  26. {
  27. name: 'benchy.3mf',
  28. path: '/benchy.3mf',
  29. size: 1048575,
  30. is_directory: false,
  31. mtime: '2024-01-15T10:00:00Z',
  32. },
  33. {
  34. name: 'print_job.gcode',
  35. path: '/print_job.gcode',
  36. size: 2048000,
  37. is_directory: false,
  38. mtime: '2024-01-14T10:00:00Z',
  39. },
  40. ];
  41. const mockStorage = {
  42. used_bytes: 1073741824, // 1 GB
  43. free_bytes: 3221225472, // 3 GB
  44. };
  45. describe('FileManagerModal', () => {
  46. const mockOnClose = vi.fn();
  47. beforeEach(() => {
  48. vi.clearAllMocks();
  49. server.use(
  50. http.get('/api/v1/printers/:id/files', () => {
  51. return HttpResponse.json({ files: mockFiles });
  52. }),
  53. http.get('/api/v1/printers/:id/storage', () => {
  54. return HttpResponse.json(mockStorage);
  55. }),
  56. http.delete('/api/v1/printers/:id/files', () => {
  57. return HttpResponse.json({ success: true });
  58. })
  59. );
  60. });
  61. describe('rendering', () => {
  62. it('renders the modal with header', async () => {
  63. render(
  64. <FileManagerModal
  65. printerId={1}
  66. printerName="X1 Carbon"
  67. onClose={mockOnClose}
  68. />
  69. );
  70. expect(screen.getByText('File Manager')).toBeInTheDocument();
  71. expect(screen.getByText('X1 Carbon')).toBeInTheDocument();
  72. });
  73. it('renders storage info', async () => {
  74. render(
  75. <FileManagerModal
  76. printerId={1}
  77. printerName="X1 Carbon"
  78. onClose={mockOnClose}
  79. />
  80. );
  81. await waitFor(() => {
  82. expect(screen.getByText(/Used:/)).toBeInTheDocument();
  83. expect(screen.getByText(/Free:/)).toBeInTheDocument();
  84. });
  85. });
  86. it('renders quick navigation buttons', () => {
  87. render(
  88. <FileManagerModal
  89. printerId={1}
  90. printerName="X1 Carbon"
  91. onClose={mockOnClose}
  92. />
  93. );
  94. expect(screen.getByText('Root')).toBeInTheDocument();
  95. expect(screen.getByText('Cache')).toBeInTheDocument();
  96. expect(screen.getByText('Models')).toBeInTheDocument();
  97. expect(screen.getByText('Timelapse')).toBeInTheDocument();
  98. });
  99. it('renders file list', async () => {
  100. render(
  101. <FileManagerModal
  102. printerId={1}
  103. printerName="X1 Carbon"
  104. onClose={mockOnClose}
  105. />
  106. );
  107. await waitFor(() => {
  108. expect(screen.getByText('cache')).toBeInTheDocument();
  109. expect(screen.getByText('model')).toBeInTheDocument();
  110. expect(screen.getByText('benchy.3mf')).toBeInTheDocument();
  111. expect(screen.getByText('print_job.gcode')).toBeInTheDocument();
  112. });
  113. });
  114. it('shows file sizes for files', async () => {
  115. render(
  116. <FileManagerModal
  117. printerId={1}
  118. printerName="X1 Carbon"
  119. onClose={mockOnClose}
  120. />
  121. );
  122. await waitFor(() => {
  123. // 1024000 bytes = 1024.0 KB
  124. expect(screen.getByText('1024.0 KB')).toBeInTheDocument();
  125. });
  126. });
  127. });
  128. describe('navigation', () => {
  129. it('navigates into a folder when clicked', async () => {
  130. server.use(
  131. http.get('/api/v1/printers/:id/files', ({ request }) => {
  132. const url = new URL(request.url);
  133. const path = url.searchParams.get('path');
  134. if (path === '/cache') {
  135. return HttpResponse.json({
  136. files: [
  137. { name: 'temp.dat', path: '/cache/temp.dat', size: 512, is_directory: false },
  138. ],
  139. });
  140. }
  141. return HttpResponse.json({ files: mockFiles });
  142. })
  143. );
  144. render(
  145. <FileManagerModal
  146. printerId={1}
  147. printerName="X1 Carbon"
  148. onClose={mockOnClose}
  149. />
  150. );
  151. await waitFor(() => {
  152. expect(screen.getByText('cache')).toBeInTheDocument();
  153. });
  154. // Click on cache folder
  155. fireEvent.click(screen.getByText('cache'));
  156. await waitFor(() => {
  157. expect(screen.getByText('temp.dat')).toBeInTheDocument();
  158. });
  159. });
  160. it('shows current path', async () => {
  161. render(
  162. <FileManagerModal
  163. printerId={1}
  164. printerName="X1 Carbon"
  165. onClose={mockOnClose}
  166. />
  167. );
  168. expect(screen.getByText('/')).toBeInTheDocument();
  169. });
  170. });
  171. describe('file selection', () => {
  172. it('selects a file when checkbox is clicked', async () => {
  173. render(
  174. <FileManagerModal
  175. printerId={1}
  176. printerName="X1 Carbon"
  177. onClose={mockOnClose}
  178. />
  179. );
  180. await waitFor(() => {
  181. expect(screen.getByText('benchy.3mf')).toBeInTheDocument();
  182. });
  183. // Find and click a checkbox (files have checkboxes, directories don't)
  184. const checkboxes = screen.getAllByRole('button').filter(btn =>
  185. btn.querySelector('svg')?.classList.contains('lucide-square')
  186. );
  187. if (checkboxes.length > 0) {
  188. fireEvent.click(checkboxes[0]);
  189. await waitFor(() => {
  190. expect(screen.getByText('1 selected')).toBeInTheDocument();
  191. });
  192. }
  193. });
  194. it('enables download button when files are selected', async () => {
  195. render(
  196. <FileManagerModal
  197. printerId={1}
  198. printerName="X1 Carbon"
  199. onClose={mockOnClose}
  200. />
  201. );
  202. await waitFor(() => {
  203. expect(screen.getByText('benchy.3mf')).toBeInTheDocument();
  204. });
  205. // Download button should be disabled initially
  206. const downloadButton = screen.getByRole('button', { name: /Download/i });
  207. expect(downloadButton).toBeDisabled();
  208. });
  209. it('shows Select All button when files exist', async () => {
  210. render(
  211. <FileManagerModal
  212. printerId={1}
  213. printerName="X1 Carbon"
  214. onClose={mockOnClose}
  215. />
  216. );
  217. await waitFor(() => {
  218. expect(screen.getByText('Select All')).toBeInTheDocument();
  219. });
  220. });
  221. });
  222. describe('search and filter', () => {
  223. it('renders search input', () => {
  224. render(
  225. <FileManagerModal
  226. printerId={1}
  227. printerName="X1 Carbon"
  228. onClose={mockOnClose}
  229. />
  230. );
  231. expect(screen.getByPlaceholderText('Filter files...')).toBeInTheDocument();
  232. });
  233. it('filters files based on search query', async () => {
  234. render(
  235. <FileManagerModal
  236. printerId={1}
  237. printerName="X1 Carbon"
  238. onClose={mockOnClose}
  239. />
  240. );
  241. await waitFor(() => {
  242. expect(screen.getByText('benchy.3mf')).toBeInTheDocument();
  243. });
  244. const searchInput = screen.getByPlaceholderText('Filter files...');
  245. fireEvent.change(searchInput, { target: { value: 'benchy' } });
  246. await waitFor(() => {
  247. expect(screen.getByText('benchy.3mf')).toBeInTheDocument();
  248. expect(screen.queryByText('print_job.gcode')).not.toBeInTheDocument();
  249. });
  250. });
  251. });
  252. describe('sorting', () => {
  253. it('renders sort dropdown', () => {
  254. render(
  255. <FileManagerModal
  256. printerId={1}
  257. printerName="X1 Carbon"
  258. onClose={mockOnClose}
  259. />
  260. );
  261. expect(screen.getByRole('combobox')).toBeInTheDocument();
  262. });
  263. it('has sort options available', () => {
  264. render(
  265. <FileManagerModal
  266. printerId={1}
  267. printerName="X1 Carbon"
  268. onClose={mockOnClose}
  269. />
  270. );
  271. const sortSelect = screen.getByRole('combobox');
  272. expect(sortSelect).toBeInTheDocument();
  273. // Check that options exist
  274. expect(screen.getByText('Name (A-Z)')).toBeInTheDocument();
  275. });
  276. });
  277. describe('close behavior', () => {
  278. it('calls onClose when X button is clicked', async () => {
  279. render(
  280. <FileManagerModal
  281. printerId={1}
  282. printerName="X1 Carbon"
  283. onClose={mockOnClose}
  284. />
  285. );
  286. const closeButton = screen.getAllByRole('button').find(btn =>
  287. btn.querySelector('.lucide-x')
  288. );
  289. if (closeButton) {
  290. fireEvent.click(closeButton);
  291. expect(mockOnClose).toHaveBeenCalled();
  292. }
  293. });
  294. it('calls onClose when clicking outside the modal', () => {
  295. render(
  296. <FileManagerModal
  297. printerId={1}
  298. printerName="X1 Carbon"
  299. onClose={mockOnClose}
  300. />
  301. );
  302. // Click on the backdrop
  303. const backdrop = document.querySelector('.fixed.inset-0');
  304. if (backdrop) {
  305. fireEvent.click(backdrop);
  306. expect(mockOnClose).toHaveBeenCalled();
  307. }
  308. });
  309. it('calls onClose when Escape key is pressed', () => {
  310. render(
  311. <FileManagerModal
  312. printerId={1}
  313. printerName="X1 Carbon"
  314. onClose={mockOnClose}
  315. />
  316. );
  317. fireEvent.keyDown(window, { key: 'Escape' });
  318. expect(mockOnClose).toHaveBeenCalled();
  319. });
  320. });
  321. describe('empty state', () => {
  322. it('shows empty message when directory has no files', async () => {
  323. server.use(
  324. http.get('/api/v1/printers/:id/files', () => {
  325. return HttpResponse.json({ files: [] });
  326. })
  327. );
  328. render(
  329. <FileManagerModal
  330. printerId={1}
  331. printerName="X1 Carbon"
  332. onClose={mockOnClose}
  333. />
  334. );
  335. await waitFor(() => {
  336. expect(screen.getByText('No files in this directory')).toBeInTheDocument();
  337. });
  338. });
  339. });
  340. describe('loading state', () => {
  341. it('shows loading spinner while fetching files', () => {
  342. // Delay the response to see loading state
  343. server.use(
  344. http.get('/api/v1/printers/:id/files', async () => {
  345. await new Promise((r) => setTimeout(r, 100));
  346. return HttpResponse.json({ files: mockFiles });
  347. })
  348. );
  349. render(
  350. <FileManagerModal
  351. printerId={1}
  352. printerName="X1 Carbon"
  353. onClose={mockOnClose}
  354. />
  355. );
  356. // The loader should be present initially
  357. const loader = document.querySelector('.animate-spin');
  358. expect(loader).toBeInTheDocument();
  359. });
  360. });
  361. });