import { screen, waitFor } from '@testing-library/react'; import { fromPartial } from '@total-typescript/shoehorn'; import type { ServerData, ServersMap, ServerWithId } from '../../../src/servers/data'; import type { ImportServersBtnProps } from '../../../src/servers/helpers/ImportServersBtn'; import { ImportServersBtn } from '../../../src/servers/helpers/ImportServersBtn'; import type { ServersImporter } from '../../../src/servers/services/ServersImporter'; import { checkAccessibility } from '../../__helpers__/accessibility'; import { renderWithStore } from '../../__helpers__/setUpTest'; describe('', () => { const csvFile = new File([''], 'servers.csv', { type: 'text/csv' }); const onImportMock = vi.fn(); const importServersFromFile = vi.fn().mockResolvedValue([]); const serversImporterMock = fromPartial({ importServersFromFile }); const setUp = (props: Partial = {}, servers: ServersMap = {}) => renderWithStore( , { initialState: { servers }, }, ); it('passes a11y checks', () => checkAccessibility(setUp())); it('shows tooltip on button hover', async () => { const { user } = setUp(); expect(screen.queryByText(/^You can create servers by importing a CSV file/)).not.toBeInTheDocument(); await user.hover(screen.getByRole('button')); await waitFor( () => expect(screen.getByText(/^You can create servers by importing a CSV file/)).toBeInTheDocument(), ); }); it.each([ [undefined, ''], ['foo', 'foo'], ['bar', 'bar'], ])('allows a class name to be provided', (providedClassName, expectedClassName) => { setUp({ className: providedClassName }); expect(screen.getByRole('button')).toHaveAttribute('class', expect.stringContaining(expectedClassName)); }); it.each([ [undefined, 'Import from file'], ['foo', 'foo'], ['bar', 'bar'], ])('has expected text', (children, expectedText) => { setUp({ children }); expect(screen.getByRole('button')).toHaveTextContent(expectedText); }); it('imports servers when file input changes', async () => { const { user } = setUp(); await user.upload(screen.getByTestId('csv-file-input'), csvFile); expect(importServersFromFile).toHaveBeenCalledTimes(1); }); it.each([ { btnName: 'Save duplicate', savesDuplicatedServers: true }, { btnName: 'Discard', savesDuplicatedServers: false }, ])('creates duplicated servers depending on selected option in modal', async ({ btnName, savesDuplicatedServers }) => { const existingServerData: ServerData = { name: 'existingServer', url: 'http://s.test/existingUrl', apiKey: 'existingApiKey', }; const existingServer: ServerWithId = { ...existingServerData, id: 'existingserver-s.test', }; const newServer: ServerData = { name: 'newServer', url: 'http://s.test/newUrl', apiKey: 'newApiKey' }; const { user, store } = setUp({}, { [existingServer.id]: existingServer }); importServersFromFile.mockResolvedValue([existingServerData, newServer]); expect(screen.queryByRole('dialog')).not.toBeInTheDocument(); await user.upload(screen.getByTestId('csv-file-input'), csvFile); // Once the file is uploaded, non-duplicated servers are immediately created const { servers } = store.getState(); expect(Object.keys(servers)).toHaveLength(2); expect(screen.getByRole('dialog')).toBeInTheDocument(); await user.click(screen.getByRole('button', { name: btnName })); // If duplicated servers are saved, there's one extra server creation if (savesDuplicatedServers) { const { servers } = store.getState(); expect(Object.keys(servers)).toHaveLength(3); } // On import is called only once, no matter what expect(onImportMock).toHaveBeenCalledOnce(); }); });