Files
ReadMeABook/tests/components/admin/indexers/IndexerConfigModal.test.tsx
kikootwo 4b90b35748 Add Transmission/NZBGet and per-client paths and much more
Extend multi-download-client support to include Transmission and NZBGet and introduce per-client custom download paths. Adds protocol mapping and new client types, Transmission/NZBGet integration services, API CRUD and validation changes, UI components/modal updates and live path previews, and manager routing by protocol. Includes DB migrations (download_path on download_history, interactive_search_access on users), schema updates, and related processor/service fixes and tests to ensure backward compatibility and proper path resolution.
2026-02-09 19:45:43 -05:00

118 lines
3.3 KiB
TypeScript

/**
* Component: Indexer Config Modal Tests
* Documentation: documentation/frontend/components.md
*/
// @vitest-environment jsdom
import { describe, expect, it, vi } from 'vitest';
import { fireEvent, render, screen, within } from '@testing-library/react';
import { IndexerConfigModal } from '@/components/admin/indexers/IndexerConfigModal';
describe('IndexerConfigModal', () => {
it('clamps numeric inputs and saves configuration', () => {
const onSave = vi.fn();
const onClose = vi.fn();
render(
<IndexerConfigModal
isOpen
onClose={onClose}
mode="add"
indexer={{ id: 1, name: 'Prowlarr', protocol: 'torrent', supportsRss: true }}
onSave={onSave}
/>
);
const [priorityInput, seedingInput] = screen.getAllByRole('spinbutton');
fireEvent.change(priorityInput, { target: { value: '99' } });
expect(priorityInput).toHaveValue(25);
fireEvent.change(seedingInput, { target: { value: '-5' } });
expect(seedingInput).toHaveValue(0);
const rssToggle = screen.getByRole('checkbox');
fireEvent.click(rssToggle);
fireEvent.click(screen.getByRole('button', { name: 'Add Indexer' }));
expect(onSave).toHaveBeenCalledWith(
expect.objectContaining({
id: 1,
name: 'Prowlarr',
priority: 25,
seedingTimeMinutes: 0,
rssEnabled: false,
audiobookCategories: expect.arrayContaining([3030]),
ebookCategories: expect.arrayContaining([7020]),
})
);
expect(onClose).toHaveBeenCalledTimes(1);
});
it('shows warning when all audiobook categories are deselected but still allows save', () => {
const onSave = vi.fn();
render(
<IndexerConfigModal
isOpen
onClose={vi.fn()}
mode="add"
indexer={{ id: 2, name: 'NoCats', protocol: 'torrent', supportsRss: true }}
onSave={onSave}
/>
);
// Find the Audiobook toggle in the category tree and click it to deselect
const audiobookLabel = screen.getByText('Audiobook');
const audiobookRow = audiobookLabel.closest('div')?.parentElement;
if (!audiobookRow) {
throw new Error('Audiobook row not found');
}
fireEvent.click(within(audiobookRow).getByRole('switch'));
// Warning should be shown instead of blocking save
expect(screen.getByText(/will not be searched for audiobooks/i)).toBeInTheDocument();
fireEvent.click(screen.getByRole('button', { name: 'Add Indexer' }));
// Save should still be called with empty audiobook categories
expect(onSave).toHaveBeenCalledWith(
expect.objectContaining({
audiobookCategories: [],
})
);
});
it('forces RSS to false when the indexer does not support RSS', () => {
const onSave = vi.fn();
const onClose = vi.fn();
render(
<IndexerConfigModal
isOpen
onClose={onClose}
mode="add"
indexer={{ id: 3, name: 'NoRSS', protocol: 'torrent', supportsRss: false }}
onSave={onSave}
/>
);
const rssToggle = screen.getByRole('checkbox');
expect(rssToggle).toBeDisabled();
fireEvent.click(screen.getByRole('button', { name: 'Add Indexer' }));
expect(onSave).toHaveBeenCalledWith(
expect.objectContaining({
id: 3,
name: 'NoRSS',
rssEnabled: false,
})
);
expect(onClose).toHaveBeenCalledTimes(1);
});
});