Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
22 changes: 16 additions & 6 deletions frontend/src/api/client.ts
Original file line number Diff line number Diff line change
Expand Up @@ -107,7 +107,9 @@ export class APIClient {
// empty or non-JSON error body — fall through to status-based message
}
const errorMessage =
(typeof errorData.error === "object" ? (errorData.error as any)?.message : errorData.error) ||
(typeof errorData.error === "object"
? (errorData.error as any)?.message
: errorData.error) ||
errorData.message ||
`HTTP ${response.status}: ${response.statusText}`;
const errorDetails =
Expand All @@ -125,7 +127,8 @@ export class APIClient {
const errorMessage =
(typeof data.error === "object" ? data.error?.message : data.error) ||
"API request failed";
const errorDetails = (typeof data.error === "object" ? (data.error as any)?.details : "") || "";
const errorDetails =
(typeof data.error === "object" ? (data.error as any)?.details : "") || "";
throw new APIError(response.status, errorMessage, errorDetails);
}

Expand Down Expand Up @@ -174,7 +177,9 @@ export class APIClient {
try {
const errorData = await response.json();
const errorMessage =
(typeof errorData.error === "object" ? (errorData.error as any)?.message : errorData.error) ||
(typeof errorData.error === "object"
? (errorData.error as any)?.message
: errorData.error) ||
errorData.message ||
`HTTP ${response.status}: ${response.statusText}`;
const errorDetails =
Expand Down Expand Up @@ -203,7 +208,8 @@ export class APIClient {
const errorMessage =
(typeof data.error === "object" ? data.error?.message : data.error) ||
"API request failed";
const errorDetails = (typeof data.error === "object" ? (data.error as any)?.details : "") || "";
const errorDetails =
(typeof data.error === "object" ? (data.error as any)?.details : "") || "";
throw new APIError(response.status, errorMessage, errorDetails);
}

Expand Down Expand Up @@ -526,11 +532,15 @@ export class APIClient {
}

async getProviderHistoricalStats(days = 30, interval = "daily") {
return this.request<ProviderHistoricalStatsResponse>(`/system/provider-stats?days=${days}&interval=${interval}`);
return this.request<ProviderHistoricalStatsResponse>(
`/system/provider-stats?days=${days}&interval=${interval}`,
);
}

async getProviderSpeedHistory(days = 30) {
return this.request<ProviderSpeedTestHistoryResponse>(`/system/provider-speed-history?days=${days}`);
return this.request<ProviderSpeedTestHistoryResponse>(
`/system/provider-speed-history?days=${days}`,
);
}

async directHealthCheck(id: number) {
Expand Down
111 changes: 111 additions & 0 deletions frontend/src/components/config/NetworkConfigSection.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,111 @@
import { Globe } from "lucide-react";
import { useEffect, useState } from "react";
import type { ConfigResponse, NetworkConfig } from "../../types/config";

interface NetworkConfigSectionProps {
config: ConfigResponse;
onUpdate?: (section: string, data: NetworkConfig) => Promise<void>;
isReadOnly?: boolean;
isUpdating?: boolean;
}

const emptyNetwork: NetworkConfig = {
http_proxy: "",
https_proxy: "",
no_proxy: "",
};

export function NetworkConfigSection({
config,
onUpdate,
isReadOnly,
isUpdating,
}: NetworkConfigSectionProps) {
const baseline = config.network ?? emptyNetwork;
const [data, setData] = useState<NetworkConfig>(baseline);
const [hasChanges, setHasChanges] = useState(false);

useEffect(() => {
setData(config.network ?? emptyNetwork);
setHasChanges(false);
}, [config.network]);

const handleChange = (field: keyof NetworkConfig, value: string) => {
const next: NetworkConfig = { ...data, [field]: value };
setData(next);
setHasChanges(JSON.stringify(next) !== JSON.stringify(baseline));
};

const handleSave = async () => {
if (!onUpdate || !hasChanges) return;
await onUpdate("network", data);
setHasChanges(false);
};

return (
<div className="space-y-6">
<div className="flex items-center gap-2">
<Globe className="h-5 w-5" aria-hidden="true" />
<h2 className="font-semibold text-xl">Network &amp; Proxy</h2>
</div>

<div className="alert alert-info">
<div className="text-sm">
Applied to every outbound HTTP request used for indexer search, NZB grabbing, Arrs
(Radarr/Sonarr/Lidarr/Readarr/Whisparr), SABnzbd fallback, and the NZBLNK resolver.
Internal endpoints (RC server, self-loopback) are not affected. Leave fields blank to
connect directly. Changes take effect on the next external request — restart AltMount if
you want long-lived clients to pick up the new proxy immediately.
</div>
</div>

<fieldset className="fieldset">
<legend className="fieldset-legend">HTTP Proxy</legend>
<input
type="text"
className="input"
placeholder="http://user:pass@host:3128"
value={data.http_proxy}
disabled={isReadOnly}
onChange={(e) => handleChange("http_proxy", e.target.value)}
/>
<p className="label">Used for plain HTTP outbound requests.</p>
</fieldset>

<fieldset className="fieldset">
<legend className="fieldset-legend">HTTPS Proxy</legend>
<input
type="text"
className="input"
placeholder="http://user:pass@host:3128"
value={data.https_proxy}
disabled={isReadOnly}
onChange={(e) => handleChange("https_proxy", e.target.value)}
/>
<p className="label">Used for HTTPS outbound requests. May be the same as HTTP Proxy.</p>
</fieldset>

<fieldset className="fieldset">
<legend className="fieldset-legend">No Proxy</legend>
<input
type="text"
className="input"
placeholder="localhost,127.0.0.1,10.0.0.0/8,*.internal"
value={data.no_proxy}
disabled={isReadOnly}
onChange={(e) => handleChange("no_proxy", e.target.value)}
/>
<p className="label">Comma-separated hosts, IPs, or CIDRs that bypass the proxy.</p>
</fieldset>

<button
type="button"
className="btn btn-primary"
onClick={handleSave}
disabled={!hasChanges || isUpdating || isReadOnly}
>
{isUpdating ? "Saving..." : "Save Changes"}
</button>
</div>
);
}
Loading
Loading