| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728 |
- /**
- * Tests for the FileManagerPage component.
- */
- import { describe, it, expect, beforeEach } from 'vitest';
- import { screen, waitFor } from '@testing-library/react';
- import userEvent from '@testing-library/user-event';
- import { render } from '../utils';
- import { FileManagerPage } from '../../pages/FileManagerPage';
- import { http, HttpResponse } from 'msw';
- import { server } from '../mocks/server';
- // Mock data
- const mockFolders = [
- {
- id: 1,
- name: 'Functional Parts',
- parent_id: null,
- file_count: 5,
- project_id: null,
- archive_id: null,
- project_name: null,
- archive_name: null,
- children: [
- {
- id: 2,
- name: 'Brackets',
- parent_id: 1,
- file_count: 3,
- project_id: null,
- archive_id: null,
- project_name: null,
- archive_name: null,
- children: [],
- },
- ],
- },
- {
- id: 3,
- name: 'Art Projects',
- parent_id: null,
- file_count: 2,
- project_id: 1,
- archive_id: null,
- project_name: 'My Art Project',
- archive_name: null,
- children: [],
- },
- ];
- const mockFiles = [
- {
- id: 1,
- filename: 'benchy.gcode.3mf',
- file_path: '/library/benchy.gcode.3mf',
- file_size: 1048576,
- file_type: '3mf',
- folder_id: null,
- thumbnail_path: '/thumbnails/1.png',
- print_name: 'Benchy',
- print_time_seconds: 3600,
- print_count: 5,
- duplicate_count: 0,
- created_at: '2024-01-01T00:00:00Z',
- },
- {
- id: 2,
- filename: 'bracket.stl',
- file_path: '/library/bracket.stl',
- file_size: 524288,
- file_type: 'stl',
- folder_id: null,
- thumbnail_path: null,
- print_name: null,
- print_time_seconds: null,
- print_count: 0,
- duplicate_count: 2,
- created_at: '2024-01-02T00:00:00Z',
- },
- ];
- const mockStats = {
- total_files: 10,
- total_folders: 3,
- total_size_bytes: 104857600,
- disk_free_bytes: 10737418240,
- disk_total_bytes: 107374182400,
- };
- describe('FileManagerPage', () => {
- beforeEach(() => {
- // Clear localStorage to ensure consistent view mode
- localStorage.clear();
- server.use(
- http.get('/api/v1/library/folders', () => {
- return HttpResponse.json(mockFolders);
- }),
- http.get('/api/v1/library/files', () => {
- return HttpResponse.json(mockFiles);
- }),
- http.get('/api/v1/library/stats', () => {
- return HttpResponse.json(mockStats);
- }),
- http.get('/api/v1/settings/', () => {
- return HttpResponse.json({
- check_updates: false,
- check_printer_firmware: false,
- library_disk_warning_gb: 5,
- });
- }),
- http.post('/api/v1/library/folders', async ({ request }) => {
- const body = await request.json() as { name: string };
- return HttpResponse.json({ id: 4, name: body.name, parent_id: null, children: [] });
- }),
- http.delete('/api/v1/library/folders/:id', () => {
- return HttpResponse.json({ success: true });
- }),
- http.delete('/api/v1/library/files/:id', () => {
- return HttpResponse.json({ success: true });
- }),
- http.post('/api/v1/library/files/move', () => {
- return HttpResponse.json({ success: true });
- }),
- http.post('/api/v1/library/files/add-to-queue', () => {
- return HttpResponse.json({ added: [{ file_id: 1, queue_id: 1 }], errors: [] });
- }),
- http.get('/api/v1/projects/', () => {
- return HttpResponse.json([{ id: 1, name: 'Test Project', color: '#00ae42' }]);
- }),
- http.get('/api/v1/archives/', () => {
- return HttpResponse.json([{ id: 1, print_name: 'Test Archive', filename: 'test.3mf' }]);
- })
- );
- });
- describe('rendering', () => {
- it('renders the page title', async () => {
- render(<FileManagerPage />);
- await waitFor(() => {
- expect(screen.getByText('File Manager')).toBeInTheDocument();
- });
- });
- it('renders the page description', async () => {
- render(<FileManagerPage />);
- await waitFor(() => {
- expect(screen.getByText('Organize and manage your print files')).toBeInTheDocument();
- });
- });
- it('shows New Folder button', async () => {
- render(<FileManagerPage />);
- await waitFor(() => {
- expect(screen.getByText('New Folder')).toBeInTheDocument();
- });
- });
- it('shows Upload button', async () => {
- render(<FileManagerPage />);
- await waitFor(() => {
- expect(screen.getByText('Upload')).toBeInTheDocument();
- });
- });
- });
- describe('stats display', () => {
- it('shows file count', async () => {
- render(<FileManagerPage />);
- await waitFor(() => {
- expect(screen.getByText('Files:')).toBeInTheDocument();
- expect(screen.getByText('10')).toBeInTheDocument();
- });
- });
- it('shows folder count', async () => {
- render(<FileManagerPage />);
- await waitFor(() => {
- expect(screen.getByText('Folders:')).toBeInTheDocument();
- // Folder count appears multiple places, just verify the label is present
- const foldersLabel = screen.getByText('Folders:');
- expect(foldersLabel.nextElementSibling?.textContent).toBe('3');
- });
- });
- it('shows total size', async () => {
- render(<FileManagerPage />);
- await waitFor(() => {
- expect(screen.getByText('Size:')).toBeInTheDocument();
- expect(screen.getByText('100.0 MB')).toBeInTheDocument();
- });
- });
- it('shows free space', async () => {
- render(<FileManagerPage />);
- await waitFor(() => {
- expect(screen.getByText('Free:')).toBeInTheDocument();
- });
- });
- });
- describe('folder sidebar', () => {
- it('shows All Files option', async () => {
- render(<FileManagerPage />);
- await waitFor(() => {
- expect(screen.getByText('All Files')).toBeInTheDocument();
- });
- });
- it('shows folder tree', async () => {
- render(<FileManagerPage />);
- await waitFor(() => {
- expect(screen.getByText('Functional Parts')).toBeInTheDocument();
- expect(screen.getByText('Art Projects')).toBeInTheDocument();
- });
- });
- it('shows nested folders', async () => {
- render(<FileManagerPage />);
- await waitFor(() => {
- expect(screen.getByText('Brackets')).toBeInTheDocument();
- });
- });
- it('shows linked folder indicator', async () => {
- render(<FileManagerPage />);
- await waitFor(() => {
- // Art Projects has a project_id
- expect(screen.getByText('Art Projects')).toBeInTheDocument();
- });
- });
- });
- describe('file display', () => {
- it('shows files in grid', async () => {
- render(<FileManagerPage />);
- await waitFor(() => {
- expect(screen.getByText('Benchy')).toBeInTheDocument();
- });
- });
- it('shows file type badges', async () => {
- render(<FileManagerPage />);
- await waitFor(() => {
- // File type badges show uppercase type
- expect(screen.getAllByText('3MF').length).toBeGreaterThan(0);
- expect(screen.getAllByText('STL').length).toBeGreaterThan(0);
- });
- });
- it('shows print count', async () => {
- render(<FileManagerPage />);
- await waitFor(() => {
- expect(screen.getByText('Printed 5x')).toBeInTheDocument();
- });
- });
- it('shows duplicate badge', async () => {
- render(<FileManagerPage />);
- await waitFor(() => {
- // Duplicate badge shows count, there may be multiple "2"s on the page
- // so we check that at least one element with "2" exists
- const elements = screen.getAllByText('2');
- expect(elements.length).toBeGreaterThan(0);
- });
- });
- });
- describe('view modes', () => {
- it('has grid view button', async () => {
- render(<FileManagerPage />);
- await waitFor(() => {
- expect(screen.getByTitle('Grid view')).toBeInTheDocument();
- });
- });
- it('has list view button', async () => {
- render(<FileManagerPage />);
- await waitFor(() => {
- expect(screen.getByTitle('List view')).toBeInTheDocument();
- });
- });
- it('can switch to list view', async () => {
- const user = userEvent.setup();
- render(<FileManagerPage />);
- // Wait for files to load first
- await waitFor(() => {
- expect(screen.getByText('Benchy')).toBeInTheDocument();
- });
- // Both view mode buttons should be present and clickable
- const gridButton = screen.getByTitle('Grid view');
- const listButton = screen.getByTitle('List view');
- expect(gridButton).toBeInTheDocument();
- expect(listButton).toBeInTheDocument();
- // Click list view button - verify no errors occur
- await user.click(listButton);
- // Clicking grid button should also work
- await user.click(gridButton);
- // Verify files are still displayed after toggling
- expect(screen.getByText('Benchy')).toBeInTheDocument();
- });
- });
- describe('search and filter', () => {
- it('has search input', async () => {
- render(<FileManagerPage />);
- await waitFor(() => {
- expect(screen.getByPlaceholderText('Search files...')).toBeInTheDocument();
- });
- });
- it('has type filter', async () => {
- render(<FileManagerPage />);
- await waitFor(() => {
- expect(screen.getByText('All types')).toBeInTheDocument();
- });
- });
- it('has sort options', async () => {
- render(<FileManagerPage />);
- await waitFor(() => {
- // Sort dropdown should show Name as default option (persisted to localStorage)
- expect(screen.getByDisplayValue('Name')).toBeInTheDocument();
- });
- });
- });
- describe('selection', () => {
- it('shows select all button', async () => {
- render(<FileManagerPage />);
- await waitFor(() => {
- expect(screen.getByText('Select All')).toBeInTheDocument();
- });
- });
- it('can select files', async () => {
- const user = userEvent.setup();
- render(<FileManagerPage />);
- await waitFor(() => {
- expect(screen.getByText('Benchy')).toBeInTheDocument();
- });
- // Click on the file card to select it
- const fileCard = screen.getByText('Benchy').closest('div[class*="cursor-pointer"]');
- if (fileCard) {
- await user.click(fileCard);
- }
- await waitFor(() => {
- expect(screen.getByText('1 selected')).toBeInTheDocument();
- });
- });
- it('shows bulk actions when files selected', async () => {
- const user = userEvent.setup();
- render(<FileManagerPage />);
- await waitFor(() => {
- expect(screen.getByText('Select All')).toBeInTheDocument();
- });
- await user.click(screen.getByText('Select All'));
- await waitFor(() => {
- expect(screen.getByText('Move')).toBeInTheDocument();
- expect(screen.getByText('Delete')).toBeInTheDocument();
- });
- });
- });
- describe('new folder modal', () => {
- it('opens new folder modal', async () => {
- const user = userEvent.setup();
- render(<FileManagerPage />);
- await waitFor(() => {
- expect(screen.getByText('New Folder')).toBeInTheDocument();
- });
- await user.click(screen.getByText('New Folder'));
- await waitFor(() => {
- expect(screen.getByText('Folder Name')).toBeInTheDocument();
- expect(screen.getByPlaceholderText('e.g., Functional Parts')).toBeInTheDocument();
- });
- });
- it('can create a folder', async () => {
- const user = userEvent.setup();
- render(<FileManagerPage />);
- await waitFor(() => {
- expect(screen.getByText('New Folder')).toBeInTheDocument();
- });
- await user.click(screen.getByText('New Folder'));
- await waitFor(() => {
- expect(screen.getByPlaceholderText('e.g., Functional Parts')).toBeInTheDocument();
- });
- const input = screen.getByPlaceholderText('e.g., Functional Parts');
- await user.type(input, 'My New Folder');
- const createButton = screen.getByRole('button', { name: 'Create' });
- await user.click(createButton);
- // Modal should close after creation
- await waitFor(() => {
- expect(screen.queryByText('Folder Name')).not.toBeInTheDocument();
- });
- });
- });
- describe('empty state', () => {
- it('shows empty state when no files', async () => {
- server.use(
- http.get('/api/v1/library/files', () => {
- return HttpResponse.json([]);
- })
- );
- render(<FileManagerPage />);
- await waitFor(() => {
- expect(screen.getByText('No files yet')).toBeInTheDocument();
- expect(screen.getByText('Upload Files')).toBeInTheDocument();
- });
- });
- });
- describe('schedule print', () => {
- it('shows schedule print button for sliced files', async () => {
- const user = userEvent.setup();
- render(<FileManagerPage />);
- await waitFor(() => {
- expect(screen.getByText('Select All')).toBeInTheDocument();
- });
- // Select a sliced file (benchy.gcode.3mf)
- await user.click(screen.getByText('Select All'));
- await waitFor(() => {
- expect(screen.getByText(/Schedule/)).toBeInTheDocument();
- });
- });
- });
- describe('STL thumbnail generation', () => {
- it('shows Generate Thumbnails button', async () => {
- render(<FileManagerPage />);
- await waitFor(() => {
- expect(screen.getByText('Generate Thumbnails')).toBeInTheDocument();
- });
- });
- it('Generate Thumbnails button has correct title', async () => {
- render(<FileManagerPage />);
- await waitFor(() => {
- const button = screen.getByTitle('Generate thumbnails for STL files missing them');
- expect(button).toBeInTheDocument();
- });
- });
- it('can click Generate Thumbnails button', async () => {
- const user = userEvent.setup();
- server.use(
- http.post('/api/v1/library/generate-stl-thumbnails', () => {
- return HttpResponse.json({
- processed: 1,
- succeeded: 1,
- failed: 0,
- results: [{ file_id: 2, success: true }],
- });
- })
- );
- render(<FileManagerPage />);
- await waitFor(() => {
- expect(screen.getByText('Generate Thumbnails')).toBeInTheDocument();
- });
- const button = screen.getByText('Generate Thumbnails');
- await user.click(button);
- // Button should work without error
- await waitFor(() => {
- expect(screen.getByText('Generate Thumbnails')).toBeInTheDocument();
- });
- });
- it('shows STL file without thumbnail in file list', async () => {
- render(<FileManagerPage />);
- await waitFor(() => {
- // bracket.stl has no thumbnail_path
- expect(screen.getByText('bracket.stl')).toBeInTheDocument();
- expect(screen.getAllByText('STL').length).toBeGreaterThan(0);
- });
- });
- });
- describe('upload modal with advanced 3MF support', () => {
- it('opens upload modal', async () => {
- const user = userEvent.setup();
- render(<FileManagerPage />);
- await waitFor(() => {
- expect(screen.getByText('Upload')).toBeInTheDocument();
- });
- await user.click(screen.getByText('Upload'));
- await waitFor(() => {
- expect(screen.getByText('Upload Files')).toBeInTheDocument();
- expect(screen.getByText(/Drag & drop/)).toBeInTheDocument();
- });
- });
- it('shows 3MF extraction info when 3MF file is added', async () => {
- const user = userEvent.setup();
- render(<FileManagerPage />);
- await waitFor(() => {
- expect(screen.getByText('Upload')).toBeInTheDocument();
- });
- await user.click(screen.getByText('Upload'));
- await waitFor(() => {
- expect(screen.getByText('Upload Files')).toBeInTheDocument();
- });
- // Create a mock 3MF file
- const threemfFile = new File(['content'], 'model.gcode.3mf', { type: 'application/octet-stream' });
- // Get the hidden file input
- const fileInput = document.querySelector('input[type="file"]') as HTMLInputElement;
- expect(fileInput).toBeInTheDocument();
- // Simulate file selection
- await user.upload(fileInput, threemfFile);
- // 3MF extraction info should appear
- await waitFor(() => {
- expect(screen.getByText('3MF files detected')).toBeInTheDocument();
- expect(screen.getByText(/Printer model.*will be automatically extracted/i)).toBeInTheDocument();
- });
- });
- it('shows STL thumbnail option when STL file is added', async () => {
- const user = userEvent.setup();
- render(<FileManagerPage />);
- await waitFor(() => {
- expect(screen.getByText('Upload')).toBeInTheDocument();
- });
- await user.click(screen.getByText('Upload'));
- await waitFor(() => {
- expect(screen.getByText('Upload Files')).toBeInTheDocument();
- });
- // Create a mock STL file
- const stlFile = new File(['solid test'], 'model.stl', { type: 'application/sla' });
- // Get the hidden file input
- const fileInput = document.querySelector('input[type="file"]') as HTMLInputElement;
- expect(fileInput).toBeInTheDocument();
- // Simulate file selection
- await user.upload(fileInput, stlFile);
- // STL thumbnail option should appear
- await waitFor(() => {
- expect(screen.getByText('STL thumbnail generation')).toBeInTheDocument();
- expect(screen.getByText(/Thumbnails can be generated/i)).toBeInTheDocument();
- });
- });
- });
- describe('authentication-based UI changes', () => {
- it('hides "Uploaded By" column and user filter when auth is disabled', async () => {
- // Mock auth disabled (default)
- server.use(
- http.get('*/api/v1/auth/status', () => {
- return HttpResponse.json({
- auth_enabled: false,
- requires_setup: false,
- });
- }),
- http.get('/api/v1/library/files', () => {
- return HttpResponse.json([
- {
- id: 1,
- filename: 'test.3mf',
- file_path: '/library/test.3mf',
- file_size: 1048576,
- file_type: '3mf',
- folder_id: null,
- thumbnail_path: null,
- print_name: 'Test File',
- print_time_seconds: 3600,
- print_count: 0,
- duplicate_count: 0,
- created_at: '2024-01-01T00:00:00Z',
- created_by_username: 'testuser',
- },
- ]);
- })
- );
- render(<FileManagerPage />);
- // Switch to list view to see the column headers
- await waitFor(() => {
- expect(screen.getByText('Test File')).toBeInTheDocument();
- });
- const user = userEvent.setup();
- const listViewButton = screen.getByRole('button', { name: /list/i });
- await user.click(listViewButton);
- // "Uploaded By" column header should not be present
- await waitFor(() => {
- expect(screen.queryByText('Uploaded By')).not.toBeInTheDocument();
- });
- // User filter dropdown should not be present
- expect(screen.queryByPlaceholderText('Filter by user')).not.toBeInTheDocument();
- });
- it('shows "Uploaded By" column and user filter when auth is enabled', async () => {
- // Mock auth enabled
- server.use(
- http.get('*/api/v1/auth/status', () => {
- return HttpResponse.json({
- auth_enabled: true,
- requires_setup: false,
- });
- }),
- http.get('/api/v1/library/files', () => {
- return HttpResponse.json([
- {
- id: 1,
- filename: 'test.3mf',
- file_path: '/library/test.3mf',
- file_size: 1048576,
- file_type: '3mf',
- folder_id: null,
- thumbnail_path: null,
- print_name: 'Test File',
- print_time_seconds: 3600,
- print_count: 0,
- duplicate_count: 0,
- created_at: '2024-01-01T00:00:00Z',
- created_by_username: 'testuser',
- },
- ]);
- }),
- http.get('/api/v1/users/', () => {
- return HttpResponse.json([
- { id: 1, username: 'testuser' },
- { id: 2, username: 'admin' },
- ]);
- })
- );
- render(<FileManagerPage />);
- // Switch to list view to see the column headers
- await waitFor(() => {
- expect(screen.getByText('Test File')).toBeInTheDocument();
- });
- const user = userEvent.setup();
- const listViewButton = screen.getByRole('button', { name: /list/i });
- await user.click(listViewButton);
- // "Uploaded By" column header should be present
- await waitFor(() => {
- expect(screen.getByText('Uploaded By')).toBeInTheDocument();
- });
- // User filter dropdown should be present
- expect(screen.getByPlaceholderText('Filter by user')).toBeInTheDocument();
- // Username should be displayed in the column
- expect(screen.getByText('testuser')).toBeInTheDocument();
- });
- });
- });
|