Files
ReadMeABook/tests/components/admin/indexers/IndexerConfigModal.test.tsx
T
kikootwo 411b5f88a4 Add per-indexer ratio-based seeding policy
Introduce a per-indexer ratioLimit alongside seedingTimeMinutes to control torrent cleanup. Updates include: documentation (scheduler and settings pages), types and API (saved indexer config now includes ratioLimit), setup and management UI (new TorrentSeedingFields component, modal wiring, validation and handlers), and processor logic (cleanup-seeded-torrents now requires AND-semantics between time and ratio; 0 disables a criterion, both 0 = never cleaned, undefined client ratio with ratioLimit>0 = not met). Tests were added/updated to cover ratio-only, time+ratio, missing-ratio, and UI interactions. Default behavior: ratioLimit defaults to 0 (no ratio requirement).
2026-05-18 15:07:50 -04:00

125 lines
3.6 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, ratioInput] = screen.getAllByRole('spinbutton');
fireEvent.change(priorityInput, { target: { value: '99' } });
expect(priorityInput).toHaveValue(25);
fireEvent.change(seedingInput, { target: { value: '-5' } });
expect(seedingInput).toHaveValue(0);
fireEvent.change(ratioInput, { target: { value: '-0.5' } });
expect(ratioInput).toHaveValue(0);
fireEvent.change(ratioInput, { target: { value: '1.5' } });
expect(ratioInput).toHaveValue(1.5);
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,
ratioLimit: 1.5,
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);
});
});