/** * BlocklistsPage — external IP blocklist source management. * * Provides three sections: * 1. **Blocklist Sources** — table of configured URLs with enable/disable * toggle, edit, delete, and preview actions. * 2. **Import Schedule** — frequency preset (hourly/daily/weekly) + time * picker + "Run Now" button showing last/next run times. * 3. **Import Log** — paginated table of completed import runs. */ import { useCallback, useState } from "react"; import { Badge, Button, Dialog, DialogActions, DialogBody, DialogContent, DialogSurface, DialogTitle, Field, Input, MessageBar, MessageBarBody, Select, Spinner, Switch, Table, TableBody, TableCell, TableCellLayout, TableHeader, TableHeaderCell, TableRow, Text, makeStyles, tokens, } from "@fluentui/react-components"; import { useCommonSectionStyles } from "../theme/commonStyles"; import { AddRegular, ArrowClockwiseRegular, DeleteRegular, EditRegular, EyeRegular, PlayRegular, } from "@fluentui/react-icons"; import { useBlocklists, useImportLog, useRunImport, useSchedule, } from "../hooks/useBlocklist"; import type { BlocklistSource, ImportRunResult, PreviewResponse, ScheduleConfig, ScheduleFrequency, } from "../types/blocklist"; // --------------------------------------------------------------------------- // Styles // --------------------------------------------------------------------------- const useStyles = makeStyles({ root: { display: "flex", flexDirection: "column", gap: tokens.spacingVerticalXL, }, tableWrapper: { overflowX: "auto" }, actionsCell: { display: "flex", gap: tokens.spacingHorizontalS, flexWrap: "wrap" }, mono: { fontFamily: "Consolas, 'Courier New', monospace", fontSize: "12px" }, centred: { display: "flex", justifyContent: "center", padding: tokens.spacingVerticalL, }, scheduleForm: { display: "flex", flexWrap: "wrap", gap: tokens.spacingHorizontalM, alignItems: "flex-end", }, scheduleField: { minWidth: "140px" }, metaRow: { display: "flex", gap: tokens.spacingHorizontalL, flexWrap: "wrap", paddingTop: tokens.spacingVerticalS, }, metaItem: { display: "flex", flexDirection: "column", gap: "2px" }, runResult: { display: "flex", flexDirection: "column", gap: tokens.spacingVerticalXS, maxHeight: "320px", overflowY: "auto", }, pagination: { display: "flex", justifyContent: "flex-end", gap: tokens.spacingHorizontalS, alignItems: "center", paddingTop: tokens.spacingVerticalS, }, dialogForm: { display: "flex", flexDirection: "column", gap: tokens.spacingVerticalM, minWidth: "380px", }, previewList: { fontFamily: "Consolas, 'Courier New', monospace", fontSize: "12px", maxHeight: "280px", overflowY: "auto", backgroundColor: tokens.colorNeutralBackground3, padding: tokens.spacingVerticalS, borderRadius: tokens.borderRadiusMedium, }, errorRow: { backgroundColor: tokens.colorStatusDangerBackground1 }, }); // --------------------------------------------------------------------------- // Source form dialog // --------------------------------------------------------------------------- interface SourceFormValues { name: string; url: string; enabled: boolean; } interface SourceFormDialogProps { open: boolean; mode: "add" | "edit"; initial: SourceFormValues; saving: boolean; error: string | null; onClose: () => void; onSubmit: (values: SourceFormValues) => void; } function SourceFormDialog({ open, mode, initial, saving, error, onClose, onSubmit, }: SourceFormDialogProps): React.JSX.Element { const styles = useStyles(); const [values, setValues] = useState(initial); // Sync when dialog re-opens with new initial data. const handleOpen = useCallback((): void => { setValues(initial); }, [initial]); return ( { if (!data.open) onClose(); }} > {mode === "add" ? "Add Blocklist Source" : "Edit Blocklist Source"}
{error && ( {error} )} { setValues((p) => ({ ...p, name: d.value })); }} placeholder="e.g. Blocklist.de — All" /> { setValues((p) => ({ ...p, url: d.value })); }} placeholder="https://lists.blocklist.de/lists/all.txt" /> { setValues((p) => ({ ...p, enabled: d.checked })); }} />
); } // --------------------------------------------------------------------------- // Preview dialog // --------------------------------------------------------------------------- interface PreviewDialogProps { open: boolean; source: BlocklistSource | null; onClose: () => void; fetchPreview: (id: number) => Promise; } function PreviewDialog({ open, source, onClose, fetchPreview }: PreviewDialogProps): React.JSX.Element { const styles = useStyles(); const [data, setData] = useState(null); const [loading, setLoading] = useState(false); const [error, setError] = useState(null); // Load preview when dialog opens. const handleOpen = useCallback((): void => { if (!source) return; setData(null); setError(null); setLoading(true); fetchPreview(source.id) .then((result) => { setData(result); setLoading(false); }) .catch((err: unknown) => { setError(err instanceof Error ? err.message : "Failed to fetch preview"); setLoading(false); }); }, [source, fetchPreview]); return ( { if (!d.open) onClose(); }} > Preview — {source?.name ?? ""} {loading && (
)} {error && ( {error} )} {data && (
{data.valid_count} valid IPs / {data.skipped_count} skipped of{" "} {data.total_lines} total lines. Showing first {data.entries.length}:
{data.entries.map((entry) => (
{entry}
))}
)}
); } // --------------------------------------------------------------------------- // Import result dialog // --------------------------------------------------------------------------- interface ImportResultDialogProps { open: boolean; result: ImportRunResult | null; onClose: () => void; } function ImportResultDialog({ open, result, onClose }: ImportResultDialogProps): React.JSX.Element { const styles = useStyles(); if (!result) return <>; return ( { if (!d.open) onClose(); }} > Import Complete
Total imported: {result.total_imported}  |  Skipped: {result.total_skipped}  |  Sources with errors: {result.errors_count} {result.results.map((r, i) => (
{r.source_url}
Imported: {r.ips_imported} | Skipped: {r.ips_skipped} {r.error ? ` | Error: ${r.error}` : ""}
))}
); } // --------------------------------------------------------------------------- // Sources section // --------------------------------------------------------------------------- const EMPTY_SOURCE: SourceFormValues = { name: "", url: "", enabled: true }; interface SourcesSectionProps { onRunImport: () => void; runImportRunning: boolean; } function SourcesSection({ onRunImport, runImportRunning }: SourcesSectionProps): React.JSX.Element { const styles = useStyles(); const sectionStyles = useCommonSectionStyles(); const { sources, loading, error, refresh, createSource, updateSource, removeSource, previewSource } = useBlocklists(); const [dialogOpen, setDialogOpen] = useState(false); const [dialogMode, setDialogMode] = useState<"add" | "edit">("add"); const [dialogInitial, setDialogInitial] = useState(EMPTY_SOURCE); const [editingId, setEditingId] = useState(null); const [saving, setSaving] = useState(false); const [saveError, setSaveError] = useState(null); const [previewOpen, setPreviewOpen] = useState(false); const [previewSourceItem, setPreviewSourceItem] = useState(null); const openAdd = useCallback((): void => { setDialogMode("add"); setDialogInitial(EMPTY_SOURCE); setEditingId(null); setSaveError(null); setDialogOpen(true); }, []); const openEdit = useCallback((source: BlocklistSource): void => { setDialogMode("edit"); setDialogInitial({ name: source.name, url: source.url, enabled: source.enabled }); setEditingId(source.id); setSaveError(null); setDialogOpen(true); }, []); const handleSubmit = useCallback( (values: SourceFormValues): void => { setSaving(true); setSaveError(null); const op = dialogMode === "add" ? createSource({ name: values.name, url: values.url, enabled: values.enabled }) : updateSource(editingId ?? -1, { name: values.name, url: values.url, enabled: values.enabled, }); op.then(() => { setSaving(false); setDialogOpen(false); }).catch((err: unknown) => { setSaving(false); setSaveError(err instanceof Error ? err.message : "Failed to save source"); }); }, [dialogMode, editingId, createSource, updateSource], ); const handleToggleEnabled = useCallback( (source: BlocklistSource): void => { void updateSource(source.id, { enabled: !source.enabled }); }, [updateSource], ); const handleDelete = useCallback( (source: BlocklistSource): void => { void removeSource(source.id); }, [removeSource], ); const handlePreview = useCallback((source: BlocklistSource): void => { setPreviewSourceItem(source); setPreviewOpen(true); }, []); return (
Blocklist Sources
{error && ( {error} )} {loading ? (
) : sources.length === 0 ? (
No blocklist sources configured. Click "Add Source" to get started.
) : (
Name URL Enabled Actions {sources.map((source) => ( {source.name} {source.url} { handleToggleEnabled(source); }} label={source.enabled ? "On" : "Off"} />
))}
)} { setDialogOpen(false); }} onSubmit={handleSubmit} /> { setPreviewOpen(false); }} fetchPreview={previewSource} />
); } // --------------------------------------------------------------------------- // Schedule section // --------------------------------------------------------------------------- const FREQUENCY_LABELS: Record = { hourly: "Every N hours", daily: "Daily", weekly: "Weekly", }; const DAYS = [ "Monday", "Tuesday", "Wednesday", "Thursday", "Friday", "Saturday", "Sunday", ]; interface ScheduleSectionProps { onRunImport: () => void; runImportRunning: boolean; } function ScheduleSection({ onRunImport, runImportRunning }: ScheduleSectionProps): React.JSX.Element { const styles = useStyles(); const sectionStyles = useCommonSectionStyles(); const { info, loading, error, saveSchedule } = useSchedule(); const [saving, setSaving] = useState(false); const [saveMsg, setSaveMsg] = useState(null); const config = info?.config ?? { frequency: "daily" as ScheduleFrequency, interval_hours: 24, hour: 3, minute: 0, day_of_week: 0, }; const [draft, setDraft] = useState(config); // Sync draft when data loads. const handleSave = useCallback((): void => { setSaving(true); saveSchedule(draft) .then(() => { setSaveMsg("Schedule saved."); setSaving(false); setTimeout(() => { setSaveMsg(null); }, 3000); }) .catch((err: unknown) => { setSaveMsg(err instanceof Error ? err.message : "Failed to save schedule"); setSaving(false); }); }, [draft, saveSchedule]); return (
Import Schedule
{error && ( {error} )} {saveMsg && ( {saveMsg} )} {loading ? (
) : ( <>
{draft.frequency === "hourly" && ( { setDraft((p) => ({ ...p, interval_hours: Math.max(1, parseInt(d.value, 10) || 1) })); }} min={1} max={168} /> )} {draft.frequency !== "hourly" && ( <> {draft.frequency === "weekly" && ( )} )}
Last run {info?.last_run_at ?? "Never"}
Next run {info?.next_run_at ?? "Not scheduled"}
)}
); } // --------------------------------------------------------------------------- // Import log section // --------------------------------------------------------------------------- function ImportLogSection(): React.JSX.Element { const styles = useStyles(); const sectionStyles = useCommonSectionStyles(); const { data, loading, error, page, setPage, refresh } = useImportLog(undefined, 20); return (
Import Log
{error && ( {error} )} {loading ? (
) : !data || data.items.length === 0 ? (
No import runs recorded yet.
) : ( <>
Timestamp Source URL Imported Skipped Status {data.items.map((entry) => ( {entry.timestamp} {entry.source_url} {entry.ips_imported} {entry.ips_skipped} {entry.errors ? ( Error ) : ( OK )} ))}
{data.total_pages > 1 && (
Page {page} of {data.total_pages}
)} )}
); } // --------------------------------------------------------------------------- // Main page // --------------------------------------------------------------------------- export function BlocklistsPage(): React.JSX.Element { const styles = useStyles(); const { running, lastResult, error: importError, runNow } = useRunImport(); const [importResultOpen, setImportResultOpen] = useState(false); const handleRunImport = useCallback((): void => { void runNow().then(() => { setImportResultOpen(true); }); }, [runNow]); return (
Blocklists {importError && ( Import error: {importError} )} { setImportResultOpen(false); }} />
); }