mirror of
https://github.com/decolua/9router.git
synced 2026-05-08 12:01:28 +00:00
* Improve dashboard responsive layouts * Improve proxy pools mobile layout * Improve CLI tool card input responsiveness --------- Co-authored-by: Delynn Assistant <zhen@dkzhen.org>
288 lines
13 KiB
JavaScript
288 lines
13 KiB
JavaScript
"use client";
|
|
|
|
import { useState, useEffect } from "react";
|
|
import { Card, Button, Badge, Input } from "@/shared/components";
|
|
|
|
const DEFAULT_MITM_ROUTER_BASE = "http://localhost:20128";
|
|
|
|
/**
|
|
* Shared MITM infrastructure card — manages SSL cert + server start/stop.
|
|
* DNS per-tool is handled separately in MitmToolCard.
|
|
*/
|
|
export default function MitmServerCard({ apiKeys, cloudEnabled, onStatusChange }) {
|
|
const [status, setStatus] = useState(null);
|
|
const [loading, setLoading] = useState(false);
|
|
const [showPasswordModal, setShowPasswordModal] = useState(false);
|
|
const [sudoPassword, setSudoPassword] = useState("");
|
|
const [selectedApiKey, setSelectedApiKey] = useState("");
|
|
const [pendingAction, setPendingAction] = useState(null);
|
|
const [modalError, setModalError] = useState(null);
|
|
const [actionError, setActionError] = useState(null);
|
|
const [mitmRouterBaseUrl, setMitmRouterBaseUrl] = useState(DEFAULT_MITM_ROUTER_BASE);
|
|
|
|
const isWindows = typeof navigator !== "undefined" && navigator.userAgent?.includes("Windows");
|
|
const isAdmin = status?.isAdmin !== false;
|
|
|
|
useEffect(() => {
|
|
if (apiKeys?.length > 0 && !selectedApiKey) {
|
|
setSelectedApiKey(apiKeys[0].key);
|
|
}
|
|
}, [apiKeys, selectedApiKey]);
|
|
|
|
useEffect(() => {
|
|
fetchStatus();
|
|
}, []);
|
|
|
|
const fetchStatus = async () => {
|
|
try {
|
|
const res = await fetch("/api/cli-tools/antigravity-mitm");
|
|
if (res.ok) {
|
|
const data = await res.json();
|
|
setStatus(data);
|
|
if (data.mitmRouterBaseUrl) {
|
|
setMitmRouterBaseUrl(data.mitmRouterBaseUrl);
|
|
}
|
|
onStatusChange?.(data);
|
|
}
|
|
} catch {
|
|
setStatus({ running: false, certExists: false, dnsStatus: {} });
|
|
}
|
|
};
|
|
|
|
const handleAction = (action) => {
|
|
setActionError(null);
|
|
if (isWindows || status?.hasCachedPassword) {
|
|
doAction(action, "");
|
|
} else {
|
|
setPendingAction(action);
|
|
setShowPasswordModal(true);
|
|
setModalError(null);
|
|
}
|
|
};
|
|
|
|
const doAction = async (action, password) => {
|
|
setLoading(true);
|
|
setActionError(null);
|
|
try {
|
|
let res;
|
|
if (action === "trust-cert") {
|
|
res = await fetch("/api/cli-tools/antigravity-mitm", {
|
|
method: "PATCH",
|
|
headers: { "Content-Type": "application/json" },
|
|
body: JSON.stringify({ action: "trust-cert", sudoPassword: password }),
|
|
});
|
|
} else if (action === "start") {
|
|
const keyToUse = selectedApiKey?.trim()
|
|
|| (apiKeys?.length > 0 ? apiKeys[0].key : null)
|
|
|| (!cloudEnabled ? "sk_9router" : null);
|
|
res = await fetch("/api/cli-tools/antigravity-mitm", {
|
|
method: "POST",
|
|
headers: { "Content-Type": "application/json" },
|
|
body: JSON.stringify({
|
|
apiKey: keyToUse,
|
|
sudoPassword: password,
|
|
mitmRouterBaseUrl: mitmRouterBaseUrl.trim() || DEFAULT_MITM_ROUTER_BASE,
|
|
}),
|
|
});
|
|
} else {
|
|
res = await fetch("/api/cli-tools/antigravity-mitm", {
|
|
method: "DELETE",
|
|
headers: { "Content-Type": "application/json" },
|
|
body: JSON.stringify({ sudoPassword: password }),
|
|
});
|
|
}
|
|
if (!res.ok) {
|
|
const data = await res.json().catch(() => ({}));
|
|
setActionError(data.error || `Failed to ${action} MITM server`);
|
|
return;
|
|
}
|
|
setShowPasswordModal(false);
|
|
setSudoPassword("");
|
|
await fetchStatus();
|
|
} catch (e) {
|
|
setActionError(e.message || "Network error");
|
|
} finally {
|
|
setLoading(false);
|
|
setPendingAction(null);
|
|
}
|
|
};
|
|
|
|
const handleConfirmPassword = () => {
|
|
if (!sudoPassword.trim()) {
|
|
setModalError("Sudo password is required");
|
|
return;
|
|
}
|
|
doAction(pendingAction, sudoPassword);
|
|
};
|
|
|
|
const isRunning = status?.running;
|
|
|
|
return (
|
|
<>
|
|
<Card padding="sm" className="border-primary/20 bg-primary/5">
|
|
<div className="flex flex-col gap-3">
|
|
{/* Header */}
|
|
<div className="flex flex-col gap-2 sm:flex-row sm:items-center sm:justify-between">
|
|
<div className="flex min-w-0 flex-wrap items-center gap-2">
|
|
<span className="material-symbols-outlined text-primary text-[20px]">security</span>
|
|
<span className="font-semibold text-sm text-text-main">MITM Server</span>
|
|
{isRunning ? (
|
|
<Badge variant="success" size="sm">Running</Badge>
|
|
) : (
|
|
<Badge variant="default" size="sm">Stopped</Badge>
|
|
)}
|
|
</div>
|
|
<div className="flex flex-wrap items-center gap-1 text-xs text-text-muted" data-i18n-skip="true">
|
|
{[
|
|
{ label: "Cert", ok: status?.certExists },
|
|
{ label: "Trusted", ok: status?.certTrusted },
|
|
{ label: "Server", ok: isRunning },
|
|
].map(({ label, ok }) => (
|
|
<span key={label} className={`flex items-center gap-0.5 px-1.5 py-0.5 rounded ${ok ? "text-green-600" : "text-text-muted"}`}>
|
|
<span className="material-symbols-outlined text-[12px]">
|
|
{ok ? "check_circle" : "cancel"}
|
|
</span>
|
|
{label}
|
|
</span>
|
|
))}
|
|
</div>
|
|
</div>
|
|
|
|
{/* Purpose & How it works */}
|
|
<div className="px-2 py-2 rounded-lg bg-surface/50 border border-border/50 flex flex-col gap-2">
|
|
<p className="text-[11px] text-text-muted leading-relaxed">
|
|
<span className="font-medium text-text-main">Purpose:</span> Use Antigravity IDE & GitHub Copilot → with ANY provider/model from 9Router
|
|
</p>
|
|
<p className="text-[11px] text-text-muted leading-relaxed">
|
|
<span className="font-medium text-text-main">How it works:</span> Antigravity/Copilot IDE request → DNS redirect to localhost:443 → MITM proxy intercepts → 9Router → response to Antigravity/Copilot
|
|
</p>
|
|
</div>
|
|
|
|
{/* Base URL + API Key — same row pattern as Claude Code / cli-tools */}
|
|
<div className="flex flex-col gap-2">
|
|
<div className="grid gap-1 sm:grid-cols-[8rem_auto_1fr] sm:items-center sm:gap-2">
|
|
<span className="text-xs font-semibold text-text-main sm:text-right sm:text-sm">9Router Base URL</span>
|
|
<span className="material-symbols-outlined hidden text-text-muted text-[14px] sm:inline">arrow_forward</span>
|
|
<input
|
|
type="text"
|
|
value={mitmRouterBaseUrl}
|
|
onChange={(e) => setMitmRouterBaseUrl(e.target.value)}
|
|
placeholder={DEFAULT_MITM_ROUTER_BASE}
|
|
disabled={isRunning}
|
|
className="flex-1 min-w-0 px-2 py-1.5 bg-surface rounded border border-border text-xs text-text-main focus:outline-none focus:ring-1 focus:ring-primary/50 disabled:opacity-50"
|
|
/>
|
|
</div>
|
|
{!isRunning && (
|
|
<div className="grid gap-1 sm:grid-cols-[8rem_auto_1fr] sm:items-center sm:gap-2">
|
|
<span className="text-xs font-semibold text-text-main sm:text-right sm:text-sm">API Key</span>
|
|
<span className="material-symbols-outlined hidden text-text-muted text-[14px] sm:inline">arrow_forward</span>
|
|
<input
|
|
type="text"
|
|
list="mitm-api-keys"
|
|
value={selectedApiKey}
|
|
onChange={(e) => setSelectedApiKey(e.target.value)}
|
|
placeholder={cloudEnabled ? "Enter or pick API key" : "sk_9router (default)"}
|
|
className="flex-1 min-w-0 px-2 py-1.5 bg-surface rounded border border-border text-xs text-text-main focus:outline-none focus:ring-1 focus:ring-primary/50"
|
|
/>
|
|
{apiKeys?.length > 0 && (
|
|
<datalist id="mitm-api-keys">
|
|
{apiKeys.map((key) => (
|
|
<option key={key.id} value={key.key}>{key.name || key.key}</option>
|
|
))}
|
|
</datalist>
|
|
)}
|
|
</div>
|
|
)}
|
|
</div>
|
|
|
|
{/* Action buttons */}
|
|
<div className="flex flex-col gap-2 sm:flex-row sm:flex-wrap sm:items-center" data-i18n-skip="true">
|
|
{status?.certExists && !status?.certTrusted && (
|
|
<button
|
|
onClick={() => handleAction("trust-cert")}
|
|
disabled={loading}
|
|
className="flex w-full items-center justify-center gap-1.5 rounded-lg border border-yellow-500/30 bg-yellow-500/10 px-4 py-2 text-xs font-medium text-yellow-600 transition-colors hover:bg-yellow-500/20 disabled:opacity-50 sm:w-auto sm:py-1.5"
|
|
>
|
|
<span className="material-symbols-outlined text-[16px]">verified_user</span>
|
|
Trust Cert
|
|
</button>
|
|
)}
|
|
{isRunning ? (
|
|
<button
|
|
onClick={() => handleAction("stop")}
|
|
disabled={loading}
|
|
className="flex w-full items-center justify-center gap-1.5 rounded-lg border border-red-500/30 bg-red-500/10 px-4 py-2 text-xs font-medium text-red-500 transition-colors hover:bg-red-500/20 disabled:opacity-50 sm:w-auto sm:py-1.5"
|
|
>
|
|
<span className="material-symbols-outlined text-[16px]">stop_circle</span>
|
|
Stop Server
|
|
</button>
|
|
) : (
|
|
<button
|
|
onClick={() => handleAction("start")}
|
|
disabled={loading || (isWindows && !isAdmin)}
|
|
className="flex w-full items-center justify-center gap-1.5 rounded-lg border border-primary/30 bg-primary/10 px-4 py-2 text-xs font-medium text-primary transition-colors hover:bg-primary/20 disabled:opacity-50 sm:w-auto sm:py-1.5"
|
|
>
|
|
<span className="material-symbols-outlined text-[16px]">play_circle</span>
|
|
Start Server
|
|
</button>
|
|
)}
|
|
{isRunning && (
|
|
<p className="text-xs text-text-muted">Enable DNS per tool below to activate interception</p>
|
|
)}
|
|
</div>
|
|
|
|
{/* Action error */}
|
|
{actionError && (
|
|
<div className="flex items-start gap-2 px-2 py-1.5 rounded text-xs bg-red-500/10 text-red-600 dark:text-red-400 border border-red-500/20">
|
|
<span className="material-symbols-outlined text-[14px] mt-0.5 shrink-0">error</span>
|
|
<span>{actionError}</span>
|
|
</div>
|
|
)}
|
|
|
|
{/* Windows admin warning */}
|
|
{isWindows && !isAdmin && (
|
|
<div className="flex items-center gap-2 px-2 py-1.5 rounded text-xs bg-red-500/10 text-red-600 border border-red-500/20">
|
|
<span className="material-symbols-outlined text-[14px]">shield_lock</span>
|
|
<span>Administrator required — restart 9Router as Administrator to use MITM</span>
|
|
</div>
|
|
)}
|
|
</div>
|
|
</Card>
|
|
|
|
{/* Password Modal */}
|
|
{showPasswordModal && (
|
|
<div className="fixed inset-0 z-50 flex items-center justify-center bg-black/50 backdrop-blur-sm">
|
|
<div className="mx-4 flex w-full max-w-sm flex-col gap-4 rounded-xl border border-border bg-surface p-5 shadow-xl sm:p-6">
|
|
<h3 className="font-semibold text-text-main">Sudo Password Required</h3>
|
|
<div className="flex items-start gap-3 p-3 bg-yellow-500/10 border border-yellow-500/30 rounded-lg">
|
|
<span className="material-symbols-outlined text-yellow-500 text-[20px]">warning</span>
|
|
<p className="text-xs text-text-muted">Required for SSL certificate and server startup</p>
|
|
</div>
|
|
<Input
|
|
type="password"
|
|
placeholder="Enter sudo password"
|
|
value={sudoPassword}
|
|
onChange={(e) => setSudoPassword(e.target.value)}
|
|
onKeyDown={(e) => { if (e.key === "Enter" && !loading) handleConfirmPassword(); }}
|
|
/>
|
|
{modalError && (
|
|
<div className="flex items-center gap-2 px-2 py-1.5 rounded text-xs bg-red-500/10 text-red-600">
|
|
<span className="material-symbols-outlined text-[14px]">error</span>
|
|
<span>{modalError}</span>
|
|
</div>
|
|
)}
|
|
<div className="flex items-center justify-end gap-2">
|
|
<Button variant="ghost" size="sm" onClick={() => { setShowPasswordModal(false); setSudoPassword(""); setModalError(null); }} disabled={loading}>
|
|
Cancel
|
|
</Button>
|
|
<Button variant="primary" size="sm" onClick={handleConfirmPassword} loading={loading}>
|
|
Confirm
|
|
</Button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
)}
|
|
</>
|
|
);
|
|
}
|