Add OIDC dashboard auth (#1020)
This commit is contained in:
parent
a48fa4eb21
commit
c3d91b019b
14 changed files with 1016 additions and 67 deletions
|
|
@ -15,6 +15,19 @@ export default function ProfilePage() {
|
|||
const [passLoading, setPassLoading] = useState(false);
|
||||
const [dbLoading, setDbLoading] = useState(false);
|
||||
const [dbStatus, setDbStatus] = useState({ type: "", message: "" });
|
||||
const [oidcForm, setOidcForm] = useState({
|
||||
authMode: "password",
|
||||
oidcIssuerUrl: "",
|
||||
oidcClientId: "",
|
||||
oidcScopes: "openid profile email",
|
||||
oidcLoginLabel: "Sign in with OIDC",
|
||||
});
|
||||
const [oidcClientSecret, setOidcClientSecret] = useState("");
|
||||
const [oidcStatus, setOidcStatus] = useState({ type: "", message: "" });
|
||||
const [oidcLoading, setOidcLoading] = useState(false);
|
||||
const [oidcTestLoading, setOidcTestLoading] = useState(false);
|
||||
const [oidcTestStatus, setOidcTestStatus] = useState({ type: "", message: "" });
|
||||
const [oidcRedirectUri, setOidcRedirectUri] = useState("/api/auth/oidc/callback");
|
||||
const importFileRef = useRef(null);
|
||||
const [proxyForm, setProxyForm] = useState({
|
||||
outboundProxyEnabled: false,
|
||||
|
|
@ -30,6 +43,14 @@ export default function ProfilePage() {
|
|||
.then((res) => res.json())
|
||||
.then((data) => {
|
||||
setSettings(data);
|
||||
setOidcForm({
|
||||
authMode: data?.authMode || "password",
|
||||
oidcIssuerUrl: data?.oidcIssuerUrl || "",
|
||||
oidcClientId: data?.oidcClientId || "",
|
||||
oidcScopes: data?.oidcScopes || "openid profile email",
|
||||
oidcLoginLabel: data?.oidcLoginLabel || "Sign in with OIDC",
|
||||
});
|
||||
setOidcClientSecret("");
|
||||
setProxyForm({
|
||||
outboundProxyEnabled: data?.outboundProxyEnabled === true,
|
||||
outboundProxyUrl: data?.outboundProxyUrl || "",
|
||||
|
|
@ -43,6 +64,12 @@ export default function ProfilePage() {
|
|||
});
|
||||
}, []);
|
||||
|
||||
useEffect(() => {
|
||||
if (typeof window !== "undefined") {
|
||||
setOidcRedirectUri(`${window.location.origin}/api/auth/oidc/callback`);
|
||||
}
|
||||
}, []);
|
||||
|
||||
const updateOutboundProxy = async (e) => {
|
||||
e.preventDefault();
|
||||
if (settings.outboundProxyEnabled !== true) return;
|
||||
|
|
@ -256,6 +283,143 @@ export default function ProfilePage() {
|
|||
}
|
||||
};
|
||||
|
||||
const updateOidcForm = (field, value) => {
|
||||
setOidcForm((prev) => ({ ...prev, [field]: value }));
|
||||
};
|
||||
|
||||
const saveOidcSettings = async (authMode = oidcForm.authMode || "password") => {
|
||||
const issuerUrl = oidcForm.oidcIssuerUrl.trim();
|
||||
const clientId = oidcForm.oidcClientId.trim();
|
||||
const scopes = oidcForm.oidcScopes.trim();
|
||||
const loginLabel = oidcForm.oidcLoginLabel.trim();
|
||||
const secret = oidcClientSecret.trim();
|
||||
|
||||
if (authMode !== "password" && (!issuerUrl || !clientId || !secret) && !settings.oidcConfigured) {
|
||||
setOidcStatus({ type: "error", message: "Issuer URL, client ID, and client secret are required to enable OIDC." });
|
||||
return;
|
||||
}
|
||||
|
||||
setOidcLoading(true);
|
||||
setOidcStatus({ type: "", message: "" });
|
||||
setOidcTestStatus({ type: "", message: "" });
|
||||
|
||||
try {
|
||||
const payload = {
|
||||
authMode,
|
||||
oidcIssuerUrl: issuerUrl,
|
||||
oidcClientId: clientId,
|
||||
oidcScopes: scopes || "openid profile email",
|
||||
oidcLoginLabel: loginLabel || "Sign in with OIDC",
|
||||
};
|
||||
if (secret) {
|
||||
payload.oidcClientSecret = secret;
|
||||
}
|
||||
|
||||
const res = await fetch("/api/settings", {
|
||||
method: "PATCH",
|
||||
headers: { "Content-Type": "application/json" },
|
||||
body: JSON.stringify(payload),
|
||||
});
|
||||
|
||||
const data = await res.json();
|
||||
if (res.ok) {
|
||||
setSettings((prev) => ({ ...prev, ...data }));
|
||||
setOidcForm({
|
||||
authMode: data?.authMode || authMode,
|
||||
oidcIssuerUrl: data?.oidcIssuerUrl || issuerUrl,
|
||||
oidcClientId: data?.oidcClientId || clientId,
|
||||
oidcScopes: data?.oidcScopes || scopes || "openid profile email",
|
||||
oidcLoginLabel: data?.oidcLoginLabel || loginLabel || "Sign in with OIDC",
|
||||
});
|
||||
setOidcClientSecret("");
|
||||
setOidcStatus({
|
||||
type: "success",
|
||||
message:
|
||||
authMode === "oidc"
|
||||
? "OIDC login enabled"
|
||||
: authMode === "both"
|
||||
? "Password and OIDC login enabled"
|
||||
: "OIDC settings saved",
|
||||
});
|
||||
} else {
|
||||
setOidcStatus({ type: "error", message: data.error || "Failed to save OIDC settings" });
|
||||
}
|
||||
} catch (err) {
|
||||
setOidcStatus({ type: "error", message: "An error occurred" });
|
||||
} finally {
|
||||
setOidcLoading(false);
|
||||
}
|
||||
};
|
||||
|
||||
const testOidcConnection = async () => {
|
||||
const issuerUrl = oidcForm.oidcIssuerUrl.trim();
|
||||
const clientId = oidcForm.oidcClientId.trim();
|
||||
const scopes = oidcForm.oidcScopes.trim();
|
||||
const secret = oidcClientSecret.trim();
|
||||
|
||||
if (!issuerUrl || !clientId) {
|
||||
setOidcTestStatus({ type: "error", message: "Issuer URL and client ID are required to test the connection." });
|
||||
return;
|
||||
}
|
||||
|
||||
setOidcTestLoading(true);
|
||||
setOidcStatus({ type: "", message: "" });
|
||||
setOidcTestStatus({ type: "", message: "" });
|
||||
|
||||
try {
|
||||
const saveRes = await fetch("/api/settings", {
|
||||
method: "PATCH",
|
||||
headers: { "Content-Type": "application/json" },
|
||||
body: JSON.stringify({
|
||||
authMode: oidcForm.authMode || settings.authMode || "password",
|
||||
oidcIssuerUrl: issuerUrl,
|
||||
oidcClientId: clientId,
|
||||
oidcScopes: scopes || "openid profile email",
|
||||
oidcLoginLabel: oidcForm.oidcLoginLabel.trim() || "Sign in with OIDC",
|
||||
...(secret ? { oidcClientSecret: secret } : {}),
|
||||
}),
|
||||
});
|
||||
|
||||
const saved = await saveRes.json().catch(() => ({}));
|
||||
if (!saveRes.ok) {
|
||||
setOidcTestStatus({
|
||||
type: "error",
|
||||
message: saved.error || "Failed to save OIDC settings before testing",
|
||||
});
|
||||
return;
|
||||
}
|
||||
|
||||
const res = await fetch("/api/auth/oidc/test", {
|
||||
method: "POST",
|
||||
headers: { "Content-Type": "application/json" },
|
||||
body: JSON.stringify({
|
||||
issuerUrl: saved.oidcIssuerUrl || issuerUrl,
|
||||
clientId: saved.oidcClientId || clientId,
|
||||
scopes: saved.oidcScopes || scopes || "openid profile email",
|
||||
}),
|
||||
});
|
||||
|
||||
const data = await res.json().catch(() => ({}));
|
||||
if (res.ok && data?.ok) {
|
||||
const statusMessage = data.clientSecretTested
|
||||
? data.clientSecretValid === true
|
||||
? `Connection OK. Discovery loaded from ${data.issuerUrl}. Client secret validated too.`
|
||||
: `Connection OK. Discovery loaded from ${data.issuerUrl}. Client secret was not checked.`
|
||||
: `Connection OK. Discovery loaded from ${data.issuerUrl}.`;
|
||||
setOidcTestStatus({
|
||||
type: "success",
|
||||
message: statusMessage,
|
||||
});
|
||||
} else {
|
||||
setOidcTestStatus({ type: "error", message: data.error || "OIDC connection test failed" });
|
||||
}
|
||||
} catch (err) {
|
||||
setOidcTestStatus({ type: "error", message: "An error occurred" });
|
||||
} finally {
|
||||
setOidcTestLoading(false);
|
||||
}
|
||||
};
|
||||
|
||||
const updateObservabilityEnabled = async (enabled) => {
|
||||
try {
|
||||
const res = await fetch("/api/settings", {
|
||||
|
|
@ -509,6 +673,155 @@ export default function ProfilePage() {
|
|||
</div>
|
||||
</Card>
|
||||
|
||||
{/* OIDC */}
|
||||
<Card>
|
||||
<div className="flex items-center gap-3 mb-4">
|
||||
<div className="p-2 rounded-lg bg-indigo-500/10 text-indigo-500 shrink-0">
|
||||
<span className="material-symbols-outlined text-[20px]">lock_open</span>
|
||||
</div>
|
||||
<h3 className="text-base sm:text-lg font-semibold">OIDC Dashboard Login</h3>
|
||||
</div>
|
||||
<div className="flex flex-col gap-4">
|
||||
<p className="text-xs sm:text-sm text-text-muted">
|
||||
Use Authentik or any OIDC provider to sign in to the dashboard. You can enable password-only, OIDC-only, or both for the dashboard; model API access still uses API keys.
|
||||
</p>
|
||||
|
||||
<div className="flex flex-col gap-2">
|
||||
<label className="font-medium text-sm sm:text-base">Auth Mode</label>
|
||||
<div className="grid grid-cols-1 sm:grid-cols-3 gap-2">
|
||||
{[
|
||||
{
|
||||
value: "password",
|
||||
title: "Password only",
|
||||
desc: "Keep the legacy password login.",
|
||||
},
|
||||
{
|
||||
value: "oidc",
|
||||
title: "OIDC only",
|
||||
desc: "Require OIDC for dashboard access.",
|
||||
},
|
||||
{
|
||||
value: "both",
|
||||
title: "Both",
|
||||
desc: "Allow either password or OIDC.",
|
||||
},
|
||||
].map((option) => {
|
||||
const active = oidcForm.authMode === option.value;
|
||||
return (
|
||||
<button
|
||||
key={option.value}
|
||||
type="button"
|
||||
onClick={() => updateOidcForm("authMode", option.value)}
|
||||
className={cn(
|
||||
"text-left rounded-lg border p-3 transition-colors",
|
||||
active
|
||||
? "border-primary bg-primary/5"
|
||||
: "border-border bg-bg hover:bg-black/5 dark:hover:bg-white/5"
|
||||
)}
|
||||
disabled={loading || oidcLoading}
|
||||
>
|
||||
<p className="font-medium text-sm sm:text-base">{option.title}</p>
|
||||
<p className="text-xs sm:text-sm text-text-muted mt-1">{option.desc}</p>
|
||||
</button>
|
||||
);
|
||||
})}
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div className="grid grid-cols-1 gap-4">
|
||||
<div className="flex flex-col gap-2">
|
||||
<label className="font-medium text-sm sm:text-base">Issuer URL</label>
|
||||
<Input
|
||||
placeholder="https://auth.example.com/application/o/9router/"
|
||||
value={oidcForm.oidcIssuerUrl}
|
||||
onChange={(e) => updateOidcForm("oidcIssuerUrl", e.target.value)}
|
||||
disabled={loading || oidcLoading}
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div className="flex flex-col gap-2">
|
||||
<label className="font-medium text-sm sm:text-base">Client ID</label>
|
||||
<Input
|
||||
placeholder="9router-dashboard"
|
||||
value={oidcForm.oidcClientId}
|
||||
onChange={(e) => updateOidcForm("oidcClientId", e.target.value)}
|
||||
disabled={loading || oidcLoading}
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div className="flex flex-col gap-2">
|
||||
<label className="font-medium text-sm sm:text-base">Client Secret</label>
|
||||
<Input
|
||||
type="password"
|
||||
placeholder="Leave blank to keep existing secret"
|
||||
value={oidcClientSecret}
|
||||
onChange={(e) => setOidcClientSecret(e.target.value)}
|
||||
disabled={loading || oidcLoading}
|
||||
/>
|
||||
<p className="text-xs sm:text-sm text-text-muted">This value is write-only after saving.</p>
|
||||
</div>
|
||||
|
||||
<div className="flex flex-col gap-2">
|
||||
<label className="font-medium text-sm sm:text-base">Scopes</label>
|
||||
<Input
|
||||
placeholder="openid profile email"
|
||||
value={oidcForm.oidcScopes}
|
||||
onChange={(e) => updateOidcForm("oidcScopes", e.target.value)}
|
||||
disabled={loading || oidcLoading}
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div className="flex flex-col gap-2">
|
||||
<label className="font-medium text-sm sm:text-base">Login Button Label</label>
|
||||
<Input
|
||||
placeholder="Sign in with OIDC"
|
||||
value={oidcForm.oidcLoginLabel}
|
||||
onChange={(e) => updateOidcForm("oidcLoginLabel", e.target.value)}
|
||||
disabled={loading || oidcLoading}
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div className="rounded-lg border border-border bg-bg p-3 text-xs sm:text-sm text-text-muted">
|
||||
<p className="font-medium text-text-main mb-1">Redirect URI</p>
|
||||
<code className="block break-all font-mono">{oidcRedirectUri}</code>
|
||||
</div>
|
||||
|
||||
<div className="flex flex-col sm:flex-row gap-2 pt-2 border-t border-border/50">
|
||||
<Button type="button" variant="primary" loading={oidcLoading} onClick={() => saveOidcSettings()} className="w-full sm:w-auto">
|
||||
Save auth mode
|
||||
</Button>
|
||||
<Button type="button" variant="outline" loading={oidcTestLoading} onClick={testOidcConnection} className="w-full sm:w-auto">
|
||||
Test connection
|
||||
</Button>
|
||||
</div>
|
||||
|
||||
{oidcTestStatus.message && (
|
||||
<p className={`text-xs sm:text-sm ${oidcTestStatus.type === "error" ? "text-red-500" : "text-green-500"}`}>
|
||||
{oidcTestStatus.message}
|
||||
</p>
|
||||
)}
|
||||
|
||||
{oidcStatus.message && (
|
||||
<p className={`text-xs sm:text-sm ${oidcStatus.type === "error" ? "text-red-500" : "text-green-500"}`}>
|
||||
{oidcStatus.message}
|
||||
</p>
|
||||
)}
|
||||
|
||||
{settings.authMode === "oidc" && (
|
||||
<p className="text-xs sm:text-sm text-amber-600 dark:text-amber-400">
|
||||
OIDC login is currently active. Password login is disabled until you switch back.
|
||||
</p>
|
||||
)}
|
||||
|
||||
{settings.authMode === "both" && (
|
||||
<p className="text-xs sm:text-sm text-amber-600 dark:text-amber-400">
|
||||
Password and OIDC login are both active.
|
||||
</p>
|
||||
)}
|
||||
</div>
|
||||
</Card>
|
||||
|
||||
{/* Routing Preferences */}
|
||||
<Card>
|
||||
<div className="flex items-center gap-3 mb-4">
|
||||
|
|
|
|||
|
|
@ -1,12 +1,9 @@
|
|||
import { NextResponse } from "next/server";
|
||||
import { getSettings } from "@/lib/localDb";
|
||||
import bcrypt from "bcryptjs";
|
||||
import { SignJWT } from "jose";
|
||||
import { cookies } from "next/headers";
|
||||
|
||||
const SECRET = new TextEncoder().encode(
|
||||
process.env.JWT_SECRET || "9router-default-secret-change-me"
|
||||
);
|
||||
import { setDashboardAuthCookie } from "@/lib/auth/dashboardSession";
|
||||
import { isOidcConfigured } from "@/lib/auth/oidc";
|
||||
|
||||
function isTunnelRequest(request, settings) {
|
||||
const host = (request.headers.get("host") || "").split(":")[0].toLowerCase();
|
||||
|
|
@ -28,6 +25,10 @@ export async function POST(request) {
|
|||
// Default password is '123456' if not set
|
||||
const storedHash = settings.password;
|
||||
|
||||
if (settings.authMode === "oidc" && isOidcConfigured(settings)) {
|
||||
return NextResponse.json({ error: "Password login is disabled. Use OIDC sign in." }, { status: 403 });
|
||||
}
|
||||
|
||||
let isValid = false;
|
||||
if (storedHash) {
|
||||
isValid = await bcrypt.compare(password, storedHash);
|
||||
|
|
@ -38,23 +39,8 @@ export async function POST(request) {
|
|||
}
|
||||
|
||||
if (isValid) {
|
||||
const forceSecureCookie = process.env.AUTH_COOKIE_SECURE === "true";
|
||||
const forwardedProto = request.headers.get("x-forwarded-proto");
|
||||
const isHttpsRequest = forwardedProto === "https";
|
||||
const useSecureCookie = forceSecureCookie || isHttpsRequest;
|
||||
|
||||
const token = await new SignJWT({ authenticated: true })
|
||||
.setProtectedHeader({ alg: "HS256" })
|
||||
.setExpirationTime("24h")
|
||||
.sign(SECRET);
|
||||
|
||||
const cookieStore = await cookies();
|
||||
cookieStore.set("auth_token", token, {
|
||||
httpOnly: true,
|
||||
secure: useSecureCookie,
|
||||
sameSite: "lax",
|
||||
path: "/",
|
||||
});
|
||||
await setDashboardAuthCookie(cookieStore, request);
|
||||
|
||||
return NextResponse.json({ success: true });
|
||||
}
|
||||
|
|
|
|||
|
|
@ -1,8 +1,12 @@
|
|||
import { NextResponse } from "next/server";
|
||||
import { cookies } from "next/headers";
|
||||
import { clearDashboardAuthCookie } from "@/lib/auth/dashboardSession";
|
||||
|
||||
export async function POST() {
|
||||
const cookieStore = await cookies();
|
||||
cookieStore.delete("auth_token");
|
||||
clearDashboardAuthCookie(cookieStore);
|
||||
cookieStore.delete("oidc_state");
|
||||
cookieStore.delete("oidc_nonce");
|
||||
cookieStore.delete("oidc_code_verifier");
|
||||
return NextResponse.json({ success: true });
|
||||
}
|
||||
|
|
|
|||
87
src/app/api/auth/oidc/callback/route.js
Normal file
87
src/app/api/auth/oidc/callback/route.js
Normal file
|
|
@ -0,0 +1,87 @@
|
|||
import { NextResponse } from "next/server";
|
||||
import { cookies } from "next/headers";
|
||||
import {
|
||||
exchangeOidcCode,
|
||||
fetchOidcDiscovery,
|
||||
getOidcRuntimeConfig,
|
||||
getPublicOrigin,
|
||||
pickOidcDisplayName,
|
||||
pickOidcEmail,
|
||||
verifyOidcIdToken,
|
||||
} from "@/lib/auth/oidc";
|
||||
import { setDashboardAuthCookie } from "@/lib/auth/dashboardSession";
|
||||
|
||||
function clearOidcCookies(cookieStore) {
|
||||
cookieStore.delete("oidc_state");
|
||||
cookieStore.delete("oidc_nonce");
|
||||
cookieStore.delete("oidc_code_verifier");
|
||||
}
|
||||
|
||||
export async function GET(request) {
|
||||
const url = new URL(request.url);
|
||||
const error = url.searchParams.get("error");
|
||||
if (error) {
|
||||
return NextResponse.redirect(new URL(`/login?error=${encodeURIComponent(error)}`, getPublicOrigin(request)));
|
||||
}
|
||||
|
||||
const code = url.searchParams.get("code");
|
||||
const state = url.searchParams.get("state");
|
||||
if (!code || !state) {
|
||||
return NextResponse.redirect(new URL("/login?error=oidc_missing_code", getPublicOrigin(request)));
|
||||
}
|
||||
|
||||
const cookieStore = await cookies();
|
||||
const storedState = cookieStore.get("oidc_state")?.value;
|
||||
const storedNonce = cookieStore.get("oidc_nonce")?.value;
|
||||
const codeVerifier = cookieStore.get("oidc_code_verifier")?.value;
|
||||
|
||||
if (!storedState || !storedNonce || !codeVerifier || storedState !== state) {
|
||||
clearOidcCookies(cookieStore);
|
||||
return NextResponse.redirect(new URL("/login?error=oidc_invalid_state", getPublicOrigin(request)));
|
||||
}
|
||||
|
||||
try {
|
||||
const config = await getOidcRuntimeConfig();
|
||||
if (!config) {
|
||||
clearOidcCookies(cookieStore);
|
||||
return NextResponse.redirect(new URL("/login?error=oidc_not_configured", getPublicOrigin(request)));
|
||||
}
|
||||
|
||||
const discovery = await fetchOidcDiscovery(config.issuerUrl);
|
||||
const discoveredIssuer = discovery.issuer || config.issuerUrl;
|
||||
const redirectUri = `${getPublicOrigin(request)}/api/auth/oidc/callback`;
|
||||
const tokenData = await exchangeOidcCode({
|
||||
tokenEndpoint: discovery.token_endpoint,
|
||||
clientId: config.clientId,
|
||||
clientSecret: config.clientSecret,
|
||||
code,
|
||||
redirectUri,
|
||||
codeVerifier,
|
||||
});
|
||||
|
||||
if (!tokenData.id_token) {
|
||||
throw new Error("OIDC provider did not return an id_token");
|
||||
}
|
||||
|
||||
const payload = await verifyOidcIdToken({
|
||||
idToken: tokenData.id_token,
|
||||
issuer: discoveredIssuer,
|
||||
audience: config.clientId,
|
||||
jwksUri: discovery.jwks_uri,
|
||||
nonce: storedNonce,
|
||||
});
|
||||
|
||||
clearOidcCookies(cookieStore);
|
||||
await setDashboardAuthCookie(cookieStore, request, {
|
||||
oidc: true,
|
||||
oidcSub: payload.sub || null,
|
||||
oidcEmail: pickOidcEmail(payload) || null,
|
||||
oidcName: pickOidcDisplayName(payload),
|
||||
});
|
||||
|
||||
return NextResponse.redirect(new URL("/dashboard", getPublicOrigin(request)));
|
||||
} catch (error) {
|
||||
clearOidcCookies(cookieStore);
|
||||
return NextResponse.redirect(new URL(`/login?error=${encodeURIComponent(error.message || "oidc_callback_failed")}`, getPublicOrigin(request)));
|
||||
}
|
||||
}
|
||||
52
src/app/api/auth/oidc/start/route.js
Normal file
52
src/app/api/auth/oidc/start/route.js
Normal file
|
|
@ -0,0 +1,52 @@
|
|||
import { NextResponse } from "next/server";
|
||||
import { cookies } from "next/headers";
|
||||
import {
|
||||
buildOidcAuthorizationUrl,
|
||||
createOidcNonce,
|
||||
createOidcState,
|
||||
createPkcePair,
|
||||
fetchOidcDiscovery,
|
||||
getOidcRuntimeConfig,
|
||||
getPublicOrigin,
|
||||
} from "@/lib/auth/oidc";
|
||||
import { shouldUseSecureCookie } from "@/lib/auth/dashboardSession";
|
||||
|
||||
export async function GET(request) {
|
||||
try {
|
||||
const config = await getOidcRuntimeConfig();
|
||||
if (!config) {
|
||||
return NextResponse.redirect(new URL("/login?error=oidc_not_configured", getPublicOrigin(request)));
|
||||
}
|
||||
|
||||
const discovery = await fetchOidcDiscovery(config.issuerUrl);
|
||||
const state = createOidcState();
|
||||
const nonce = createOidcNonce();
|
||||
const { verifier, challenge } = createPkcePair();
|
||||
const redirectUri = `${getPublicOrigin(request)}/api/auth/oidc/callback`;
|
||||
const authUrl = buildOidcAuthorizationUrl({
|
||||
authorizationEndpoint: discovery.authorization_endpoint,
|
||||
clientId: config.clientId,
|
||||
redirectUri,
|
||||
scopes: config.scopes,
|
||||
state,
|
||||
nonce,
|
||||
codeChallenge: challenge,
|
||||
});
|
||||
|
||||
const cookieStore = await cookies();
|
||||
const baseOptions = {
|
||||
httpOnly: true,
|
||||
secure: shouldUseSecureCookie(request),
|
||||
sameSite: "lax",
|
||||
path: "/",
|
||||
maxAge: 10 * 60,
|
||||
};
|
||||
cookieStore.set("oidc_state", state, baseOptions);
|
||||
cookieStore.set("oidc_nonce", nonce, baseOptions);
|
||||
cookieStore.set("oidc_code_verifier", verifier, baseOptions);
|
||||
|
||||
return NextResponse.redirect(authUrl);
|
||||
} catch (error) {
|
||||
return NextResponse.redirect(new URL(`/login?error=${encodeURIComponent(error.message || "oidc_start_failed")}`, getPublicOrigin(request)));
|
||||
}
|
||||
}
|
||||
84
src/app/api/auth/oidc/test/route.js
Normal file
84
src/app/api/auth/oidc/test/route.js
Normal file
|
|
@ -0,0 +1,84 @@
|
|||
import { NextResponse } from "next/server";
|
||||
import { cookies } from "next/headers";
|
||||
import { getSettings } from "@/lib/localDb";
|
||||
import { fetchOidcDiscovery, getPublicOrigin, probeOidcClientSecret } from "@/lib/auth/oidc";
|
||||
import { verifyDashboardAuthToken } from "@/lib/auth/dashboardSession";
|
||||
|
||||
async function canAccessTestRoute() {
|
||||
const settings = await getSettings();
|
||||
if (settings.requireLogin === false) return true;
|
||||
|
||||
const cookieStore = await cookies();
|
||||
const token = cookieStore.get("auth_token")?.value;
|
||||
return await verifyDashboardAuthToken(token);
|
||||
}
|
||||
|
||||
export async function POST(request) {
|
||||
try {
|
||||
if (!(await canAccessTestRoute())) {
|
||||
return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
|
||||
}
|
||||
|
||||
const body = await request.json().catch(() => ({}));
|
||||
const settings = await getSettings();
|
||||
|
||||
const issuerUrl = String(body.issuerUrl || settings.oidcIssuerUrl || "").trim();
|
||||
const clientId = String(body.clientId || settings.oidcClientId || "").trim();
|
||||
const scopes = String(body.scopes || settings.oidcScopes || "openid profile email").trim() || "openid profile email";
|
||||
const clientSecret = String(
|
||||
Object.prototype.hasOwnProperty.call(body, "clientSecret")
|
||||
? body.clientSecret
|
||||
: settings.oidcClientSecret || ""
|
||||
).trim();
|
||||
|
||||
if (!issuerUrl) {
|
||||
return NextResponse.json({ error: "Issuer URL is required" }, { status: 400 });
|
||||
}
|
||||
if (!clientId) {
|
||||
return NextResponse.json({ error: "Client ID is required" }, { status: 400 });
|
||||
}
|
||||
|
||||
const discovery = await fetchOidcDiscovery(issuerUrl);
|
||||
const redirectUri = `${getPublicOrigin(request)}/api/auth/oidc/callback`;
|
||||
const secretProbe = await probeOidcClientSecret({
|
||||
tokenEndpoint: discovery.token_endpoint,
|
||||
clientId,
|
||||
clientSecret,
|
||||
redirectUri,
|
||||
});
|
||||
|
||||
if (secretProbe.tested && secretProbe.valid === false) {
|
||||
return NextResponse.json({
|
||||
ok: false,
|
||||
discoveryOk: true,
|
||||
clientSecretTested: true,
|
||||
clientSecretValid: false,
|
||||
issuerUrl,
|
||||
clientId,
|
||||
scopes,
|
||||
redirectUri,
|
||||
authorizationEndpoint: discovery.authorization_endpoint || "",
|
||||
tokenEndpoint: discovery.token_endpoint || "",
|
||||
jwksUri: discovery.jwks_uri || "",
|
||||
error: `Discovery loaded, but the client secret is not valid: ${secretProbe.message}`,
|
||||
});
|
||||
}
|
||||
|
||||
return NextResponse.json({
|
||||
ok: true,
|
||||
discoveryOk: true,
|
||||
clientSecretTested: secretProbe.tested,
|
||||
clientSecretValid: secretProbe.valid,
|
||||
issuerUrl,
|
||||
clientId,
|
||||
scopes,
|
||||
redirectUri,
|
||||
authorizationEndpoint: discovery.authorization_endpoint || "",
|
||||
tokenEndpoint: discovery.token_endpoint || "",
|
||||
jwksUri: discovery.jwks_uri || "",
|
||||
message: secretProbe.message,
|
||||
});
|
||||
} catch (error) {
|
||||
return NextResponse.json({ error: error.message || "OIDC test failed" }, { status: 500 });
|
||||
}
|
||||
}
|
||||
45
src/app/api/auth/status/route.js
Normal file
45
src/app/api/auth/status/route.js
Normal file
|
|
@ -0,0 +1,45 @@
|
|||
import { NextResponse } from "next/server";
|
||||
import { cookies } from "next/headers";
|
||||
import { getSettings } from "@/lib/localDb";
|
||||
import { isOidcConfigured } from "@/lib/auth/oidc";
|
||||
import { getDashboardAuthSession } from "@/lib/auth/dashboardSession";
|
||||
|
||||
export async function GET() {
|
||||
try {
|
||||
const settings = await getSettings();
|
||||
const cookieStore = await cookies();
|
||||
const session = await getDashboardAuthSession(cookieStore.get("auth_token")?.value);
|
||||
const requireLogin = settings.requireLogin !== false;
|
||||
const authMode = settings.authMode || "password";
|
||||
const oidcName = String(session?.oidcName || "").trim();
|
||||
const oidcEmail = String(session?.oidcEmail || "").trim();
|
||||
const displayName = oidcName || oidcEmail || (session?.oidc ? "OIDC user" : "Password user");
|
||||
const loginMethod = session?.oidc ? "OIDC" : "Password";
|
||||
|
||||
return NextResponse.json({
|
||||
requireLogin,
|
||||
authMode,
|
||||
oidcConfigured: isOidcConfigured(settings),
|
||||
oidcLoginLabel: (settings.oidcLoginLabel || "Sign in with OIDC").trim() || "Sign in with OIDC",
|
||||
hasPassword: !!settings.password,
|
||||
displayName,
|
||||
loginMethod,
|
||||
oidcName: oidcName || null,
|
||||
oidcEmail: oidcEmail || null,
|
||||
oidcLogin: !!session?.oidc,
|
||||
});
|
||||
} catch {
|
||||
return NextResponse.json({
|
||||
requireLogin: true,
|
||||
authMode: "password",
|
||||
oidcConfigured: false,
|
||||
oidcLoginLabel: "Sign in with OIDC",
|
||||
hasPassword: false,
|
||||
displayName: "Password user",
|
||||
loginMethod: "Password",
|
||||
oidcName: null,
|
||||
oidcEmail: null,
|
||||
oidcLogin: false,
|
||||
});
|
||||
}
|
||||
}
|
||||
|
|
@ -14,7 +14,8 @@ const SETTINGS_RESPONSE_HEADERS = {
|
|||
export async function GET() {
|
||||
try {
|
||||
const settings = await getSettings();
|
||||
const { password, ...safeSettings } = settings;
|
||||
const { password, oidcClientSecret, ...safeSettings } = settings;
|
||||
safeSettings.oidcConfigured = !!(safeSettings.oidcIssuerUrl && safeSettings.oidcClientId && oidcClientSecret);
|
||||
|
||||
const enableRequestLogs = process.env.ENABLE_REQUEST_LOGS === "true";
|
||||
const enableTranslator = process.env.ENABLE_TRANSLATOR === "true";
|
||||
|
|
@ -63,6 +64,12 @@ export async function PATCH(request) {
|
|||
delete body.currentPassword;
|
||||
}
|
||||
|
||||
if (Object.prototype.hasOwnProperty.call(body, "oidcClientSecret")) {
|
||||
if (!body.oidcClientSecret || !String(body.oidcClientSecret).trim()) {
|
||||
delete body.oidcClientSecret;
|
||||
}
|
||||
}
|
||||
|
||||
const settings = await updateSettings(body);
|
||||
|
||||
// Apply outbound proxy settings immediately (no restart required)
|
||||
|
|
@ -83,7 +90,8 @@ export async function PATCH(request) {
|
|||
resetComboRotation();
|
||||
}
|
||||
|
||||
const { password, ...safeSettings } = settings;
|
||||
const { password, oidcClientSecret, ...safeSettings } = settings;
|
||||
safeSettings.oidcConfigured = !!(safeSettings.oidcIssuerUrl && safeSettings.oidcClientId && oidcClientSecret);
|
||||
return NextResponse.json(safeSettings, { headers: SETTINGS_RESPONSE_HEADERS });
|
||||
} catch (error) {
|
||||
console.log("Error updating settings:", error);
|
||||
|
|
|
|||
|
|
@ -9,6 +9,9 @@ export default function LoginPage() {
|
|||
const [error, setError] = useState("");
|
||||
const [loading, setLoading] = useState(false);
|
||||
const [hasPassword, setHasPassword] = useState(null);
|
||||
const [authMode, setAuthMode] = useState("password");
|
||||
const [oidcConfigured, setOidcConfigured] = useState(false);
|
||||
const [oidcLoginLabel, setOidcLoginLabel] = useState("Sign in with OIDC");
|
||||
const router = useRouter();
|
||||
|
||||
useEffect(() => {
|
||||
|
|
@ -18,7 +21,7 @@ export default function LoginPage() {
|
|||
const baseUrl = typeof window !== "undefined" ? window.location.origin : "";
|
||||
|
||||
try {
|
||||
const res = await fetch(`${baseUrl}/api/settings`, {
|
||||
const res = await fetch(`${baseUrl}/api/auth/status`, {
|
||||
signal: controller.signal,
|
||||
});
|
||||
clearTimeout(timeoutId);
|
||||
|
|
@ -31,6 +34,9 @@ export default function LoginPage() {
|
|||
return;
|
||||
}
|
||||
setHasPassword(!!data.hasPassword);
|
||||
setAuthMode(data.authMode || "password");
|
||||
setOidcConfigured(data.oidcConfigured === true);
|
||||
setOidcLoginLabel(data.oidcLoginLabel || "Sign in with OIDC");
|
||||
} else {
|
||||
// Safe fallback on non-OK response to avoid infinite loading state.
|
||||
setHasPassword(true);
|
||||
|
|
@ -69,6 +75,13 @@ export default function LoginPage() {
|
|||
}
|
||||
};
|
||||
|
||||
const handleOidcLogin = () => {
|
||||
window.location.href = "/api/auth/oidc/start";
|
||||
};
|
||||
|
||||
const oidcAvailable = oidcConfigured && ["oidc", "both"].includes(authMode);
|
||||
const passwordAvailable = authMode !== "oidc" || !oidcConfigured;
|
||||
|
||||
// Show loading state while checking password
|
||||
if (hasPassword === null) {
|
||||
return (
|
||||
|
|
@ -88,37 +101,72 @@ export default function LoginPage() {
|
|||
<div className="relative z-10 w-full max-w-md">
|
||||
<div className="text-center mb-8">
|
||||
<h1 className="text-3xl font-bold text-primary mb-2">9Router</h1>
|
||||
<p className="text-text-muted">Enter your password to access the dashboard</p>
|
||||
<p className="text-text-muted">
|
||||
{authMode === "oidc" && oidcConfigured
|
||||
? "Sign in with your OIDC provider to access the dashboard"
|
||||
: "Enter your password to access the dashboard"}
|
||||
</p>
|
||||
</div>
|
||||
|
||||
<Card>
|
||||
<form onSubmit={handleLogin} className="flex flex-col gap-4">
|
||||
<div className="flex flex-col gap-2">
|
||||
<label className="text-sm font-medium">Password</label>
|
||||
<Input
|
||||
type="password"
|
||||
placeholder="Enter password"
|
||||
value={password}
|
||||
onChange={(e) => setPassword(e.target.value)}
|
||||
required
|
||||
autoFocus
|
||||
/>
|
||||
{error && <p className="text-xs text-red-500">{error}</p>}
|
||||
</div>
|
||||
<div className="flex flex-col gap-4">
|
||||
{oidcAvailable && (
|
||||
<Button type="button" variant="primary" className="w-full" onClick={handleOidcLogin}>
|
||||
{oidcLoginLabel}
|
||||
</Button>
|
||||
)}
|
||||
|
||||
<Button
|
||||
type="submit"
|
||||
variant="primary"
|
||||
className="w-full"
|
||||
loading={loading}
|
||||
>
|
||||
Login
|
||||
</Button>
|
||||
{oidcAvailable && passwordAvailable && <div className="h-px bg-border/60" />}
|
||||
|
||||
<p className="text-xs text-center text-text-muted mt-2">
|
||||
Default password is <code className="bg-sidebar px-1 rounded">123456</code>
|
||||
</p>
|
||||
</form>
|
||||
{passwordAvailable ? (
|
||||
<form onSubmit={handleLogin} className="flex flex-col gap-4">
|
||||
{((authMode === "oidc" && !oidcConfigured) || (authMode === "both" && !oidcConfigured)) && (
|
||||
<p className="text-xs text-amber-600 dark:text-amber-400 text-center">
|
||||
OIDC login is enabled, but the issuer/client fields are not configured yet. Password login is still available for recovery.
|
||||
</p>
|
||||
)}
|
||||
|
||||
{authMode === "both" && oidcConfigured && (
|
||||
<p className="text-xs text-text-muted text-center">
|
||||
Password and OIDC login are both enabled.
|
||||
</p>
|
||||
)}
|
||||
|
||||
<div className="flex flex-col gap-2">
|
||||
<label className="text-sm font-medium">Password</label>
|
||||
<Input
|
||||
type="password"
|
||||
placeholder="Enter password"
|
||||
value={password}
|
||||
onChange={(e) => setPassword(e.target.value)}
|
||||
required
|
||||
autoFocus={!oidcAvailable}
|
||||
/>
|
||||
{error && <p className="text-xs text-red-500">{error}</p>}
|
||||
</div>
|
||||
|
||||
<Button
|
||||
type="submit"
|
||||
variant="primary"
|
||||
className="w-full"
|
||||
loading={loading}
|
||||
>
|
||||
Login
|
||||
</Button>
|
||||
|
||||
<p className="text-xs text-center text-text-muted mt-2">
|
||||
Default password is <code className="bg-sidebar px-1 rounded">123456</code>
|
||||
</p>
|
||||
{hasPassword === false && (
|
||||
<p className="text-xs text-center text-text-muted">
|
||||
No custom password is set yet. The default password above will work until you change it.
|
||||
</p>
|
||||
)}
|
||||
</form>
|
||||
) : (
|
||||
error && <p className="text-xs text-red-500">{error}</p>
|
||||
)}
|
||||
</div>
|
||||
</Card>
|
||||
</div>
|
||||
</div>
|
||||
|
|
|
|||
|
|
@ -1,11 +1,7 @@
|
|||
import { NextResponse } from "next/server";
|
||||
import { jwtVerify } from "jose";
|
||||
import { getSettings } from "@/lib/localDb";
|
||||
import { getConsistentMachineId } from "@/shared/utils/machineId";
|
||||
|
||||
const SECRET = new TextEncoder().encode(
|
||||
process.env.JWT_SECRET || "9router-default-secret-change-me"
|
||||
);
|
||||
import { verifyDashboardAuthToken } from "@/lib/auth/dashboardSession";
|
||||
|
||||
const CLI_TOKEN_HEADER = "x-9r-cli-token";
|
||||
const CLI_TOKEN_SALT = "9r-cli-auth";
|
||||
|
|
@ -38,13 +34,7 @@ const PROTECTED_API_PATHS = [
|
|||
|
||||
async function hasValidToken(request) {
|
||||
const token = request.cookies.get("auth_token")?.value;
|
||||
if (!token) return false;
|
||||
try {
|
||||
await jwtVerify(token, SECRET);
|
||||
return true;
|
||||
} catch {
|
||||
return false;
|
||||
}
|
||||
return await verifyDashboardAuthToken(token);
|
||||
}
|
||||
|
||||
// Read settings directly from DB to avoid self-fetch deadlock in proxy
|
||||
|
|
@ -112,10 +102,9 @@ export async function proxy(request) {
|
|||
// Verify JWT token
|
||||
const token = request.cookies.get("auth_token")?.value;
|
||||
if (token) {
|
||||
try {
|
||||
await jwtVerify(token, SECRET);
|
||||
if (await verifyDashboardAuthToken(token)) {
|
||||
return NextResponse.next();
|
||||
} catch {
|
||||
} else {
|
||||
return NextResponse.redirect(new URL("/login", request.url));
|
||||
}
|
||||
}
|
||||
|
|
|
|||
54
src/lib/auth/dashboardSession.js
Normal file
54
src/lib/auth/dashboardSession.js
Normal file
|
|
@ -0,0 +1,54 @@
|
|||
import { SignJWT, jwtVerify } from "jose";
|
||||
|
||||
const SECRET = new TextEncoder().encode(
|
||||
process.env.JWT_SECRET || "9router-default-secret-change-me"
|
||||
);
|
||||
|
||||
export function shouldUseSecureCookie(request) {
|
||||
const forceSecureCookie = process.env.AUTH_COOKIE_SECURE === "true";
|
||||
const forwardedProto = request?.headers?.get?.("x-forwarded-proto");
|
||||
const isHttpsRequest = forwardedProto === "https";
|
||||
return forceSecureCookie || isHttpsRequest;
|
||||
}
|
||||
|
||||
export async function createDashboardAuthToken(claims = {}) {
|
||||
return new SignJWT({ authenticated: true, ...claims })
|
||||
.setProtectedHeader({ alg: "HS256" })
|
||||
.setIssuedAt()
|
||||
.setExpirationTime("24h")
|
||||
.sign(SECRET);
|
||||
}
|
||||
|
||||
export async function verifyDashboardAuthToken(token) {
|
||||
if (!token) return false;
|
||||
try {
|
||||
await jwtVerify(token, SECRET);
|
||||
return true;
|
||||
} catch {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
export async function getDashboardAuthSession(token) {
|
||||
if (!token) return null;
|
||||
try {
|
||||
const { payload } = await jwtVerify(token, SECRET);
|
||||
return payload;
|
||||
} catch {
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
export async function setDashboardAuthCookie(cookieStore, request, claims = {}) {
|
||||
const token = await createDashboardAuthToken(claims);
|
||||
cookieStore.set("auth_token", token, {
|
||||
httpOnly: true,
|
||||
secure: shouldUseSecureCookie(request),
|
||||
sameSite: "lax",
|
||||
path: "/",
|
||||
});
|
||||
}
|
||||
|
||||
export function clearDashboardAuthCookie(cookieStore) {
|
||||
cookieStore.delete("auth_token");
|
||||
}
|
||||
234
src/lib/auth/oidc.js
Normal file
234
src/lib/auth/oidc.js
Normal file
|
|
@ -0,0 +1,234 @@
|
|||
import crypto from "node:crypto";
|
||||
import { createRemoteJWKSet, jwtVerify } from "jose";
|
||||
import { getSettings } from "@/lib/localDb";
|
||||
|
||||
export const OIDC_COOKIE_NAMES = {
|
||||
state: "oidc_state",
|
||||
nonce: "oidc_nonce",
|
||||
verifier: "oidc_code_verifier",
|
||||
};
|
||||
|
||||
const DEFAULT_SCOPES = "openid profile email";
|
||||
const DEFAULT_LOGIN_LABEL = "Sign in with OIDC";
|
||||
|
||||
function trimTrailingSlashes(value) {
|
||||
return (value || "").trim().replace(/\/+$/, "");
|
||||
}
|
||||
|
||||
function normalizeScopes(value) {
|
||||
return (value || DEFAULT_SCOPES).trim() || DEFAULT_SCOPES;
|
||||
}
|
||||
|
||||
export function getPublicOrigin(request) {
|
||||
const configuredBaseUrl =
|
||||
process.env.BASE_URL ||
|
||||
process.env.NEXT_PUBLIC_BASE_URL ||
|
||||
"";
|
||||
|
||||
if (configuredBaseUrl) {
|
||||
return trimTrailingSlashes(configuredBaseUrl);
|
||||
}
|
||||
|
||||
const forwardedProto = request?.headers?.get?.("x-forwarded-proto") || "";
|
||||
const forwardedHost = request?.headers?.get?.("x-forwarded-host") || "";
|
||||
const host = forwardedHost || request?.headers?.get?.("host") || "";
|
||||
if (host) {
|
||||
const protocol = (forwardedProto || new URL(request.url).protocol || "http:").replace(/:$/, "");
|
||||
return `${protocol}://${host}`.replace(/\/+$/, "");
|
||||
}
|
||||
|
||||
return trimTrailingSlashes(new URL(request.url).origin);
|
||||
}
|
||||
|
||||
export function isOidcConfigured(settings) {
|
||||
return !!(
|
||||
trimTrailingSlashes(settings?.oidcIssuerUrl) &&
|
||||
(settings?.oidcClientId || "").trim() &&
|
||||
(settings?.oidcClientSecret || "").trim()
|
||||
);
|
||||
}
|
||||
|
||||
export async function getOidcRuntimeConfig() {
|
||||
const settings = await getSettings();
|
||||
if (!["oidc", "both"].includes(settings.authMode) || !isOidcConfigured(settings)) return null;
|
||||
|
||||
const issuerUrl = trimTrailingSlashes(settings.oidcIssuerUrl);
|
||||
return {
|
||||
issuerUrl,
|
||||
clientId: settings.oidcClientId.trim(),
|
||||
clientSecret: settings.oidcClientSecret.trim(),
|
||||
scopes: normalizeScopes(settings.oidcScopes),
|
||||
loginLabel: (settings.oidcLoginLabel || DEFAULT_LOGIN_LABEL).trim() || DEFAULT_LOGIN_LABEL,
|
||||
};
|
||||
}
|
||||
|
||||
export async function fetchOidcDiscovery(issuerUrl) {
|
||||
const discoveryUrl = `${trimTrailingSlashes(issuerUrl)}/.well-known/openid-configuration`;
|
||||
const res = await fetch(discoveryUrl, { cache: "no-store" });
|
||||
if (!res.ok) {
|
||||
throw new Error(`Failed to load OIDC discovery document from ${discoveryUrl}`);
|
||||
}
|
||||
return await res.json();
|
||||
}
|
||||
|
||||
export function createPkcePair() {
|
||||
const verifier = crypto.randomBytes(32).toString("base64url");
|
||||
const challenge = crypto.createHash("sha256").update(verifier).digest("base64url");
|
||||
return { verifier, challenge };
|
||||
}
|
||||
|
||||
export function createOidcState() {
|
||||
return crypto.randomBytes(16).toString("base64url");
|
||||
}
|
||||
|
||||
export function createOidcNonce() {
|
||||
return crypto.randomBytes(16).toString("base64url");
|
||||
}
|
||||
|
||||
export function buildOidcAuthorizationUrl({
|
||||
authorizationEndpoint,
|
||||
clientId,
|
||||
redirectUri,
|
||||
scopes = DEFAULT_SCOPES,
|
||||
state,
|
||||
nonce,
|
||||
codeChallenge,
|
||||
}) {
|
||||
const url = new URL(authorizationEndpoint);
|
||||
url.searchParams.set("response_type", "code");
|
||||
url.searchParams.set("client_id", clientId);
|
||||
url.searchParams.set("redirect_uri", redirectUri);
|
||||
url.searchParams.set("scope", normalizeScopes(scopes));
|
||||
url.searchParams.set("state", state);
|
||||
url.searchParams.set("nonce", nonce);
|
||||
url.searchParams.set("code_challenge", codeChallenge);
|
||||
url.searchParams.set("code_challenge_method", "S256");
|
||||
return url.toString();
|
||||
}
|
||||
|
||||
export async function exchangeOidcCode({
|
||||
tokenEndpoint,
|
||||
clientId,
|
||||
clientSecret,
|
||||
code,
|
||||
redirectUri,
|
||||
codeVerifier,
|
||||
}) {
|
||||
const body = new URLSearchParams({
|
||||
grant_type: "authorization_code",
|
||||
client_id: clientId,
|
||||
code,
|
||||
redirect_uri: redirectUri,
|
||||
code_verifier: codeVerifier,
|
||||
});
|
||||
|
||||
if (clientSecret) {
|
||||
body.set("client_secret", clientSecret);
|
||||
}
|
||||
|
||||
const res = await fetch(tokenEndpoint, {
|
||||
method: "POST",
|
||||
headers: { "Content-Type": "application/x-www-form-urlencoded" },
|
||||
body,
|
||||
});
|
||||
|
||||
const data = await res.json().catch(() => ({}));
|
||||
if (!res.ok) {
|
||||
const message = data?.error_description || data?.error || `OIDC token exchange failed (${res.status})`;
|
||||
throw new Error(message);
|
||||
}
|
||||
|
||||
return data;
|
||||
}
|
||||
|
||||
export async function probeOidcClientSecret({
|
||||
tokenEndpoint,
|
||||
clientId,
|
||||
clientSecret,
|
||||
redirectUri,
|
||||
}) {
|
||||
if (!clientSecret) {
|
||||
return {
|
||||
tested: false,
|
||||
valid: null,
|
||||
message: "No client secret was provided, so secret validation was skipped.",
|
||||
};
|
||||
}
|
||||
|
||||
const body = new URLSearchParams({
|
||||
grant_type: "authorization_code",
|
||||
client_id: clientId,
|
||||
client_secret: clientSecret,
|
||||
code: "__oidc_test_invalid_code__",
|
||||
redirect_uri: redirectUri,
|
||||
code_verifier: "__oidc_test_invalid_verifier__",
|
||||
});
|
||||
|
||||
const res = await fetch(tokenEndpoint, {
|
||||
method: "POST",
|
||||
headers: { "Content-Type": "application/x-www-form-urlencoded" },
|
||||
body,
|
||||
});
|
||||
|
||||
const data = await res.json().catch(() => ({}));
|
||||
const error = (data?.error || "").toLowerCase();
|
||||
const errorDescription = data?.error_description || data?.error || "";
|
||||
|
||||
if (res.ok) {
|
||||
return {
|
||||
tested: true,
|
||||
valid: true,
|
||||
message: "Client secret was accepted by the token endpoint.",
|
||||
raw: data,
|
||||
};
|
||||
}
|
||||
|
||||
if (error === "invalid_client" || error === "unauthorized_client" || /client.*(invalid|failed|mismatch)/i.test(errorDescription)) {
|
||||
return {
|
||||
tested: true,
|
||||
valid: false,
|
||||
message: errorDescription || "Client secret is not valid.",
|
||||
raw: data,
|
||||
};
|
||||
}
|
||||
|
||||
if (error === "invalid_grant" || error === "invalid_code" || /grant|code/i.test(errorDescription)) {
|
||||
return {
|
||||
tested: true,
|
||||
valid: true,
|
||||
message: "Client secret was accepted; the token exchange failed only because the test authorization code is invalid.",
|
||||
raw: data,
|
||||
};
|
||||
}
|
||||
|
||||
return {
|
||||
tested: true,
|
||||
valid: null,
|
||||
message: errorDescription || `Token endpoint responded with ${res.status}`,
|
||||
raw: data,
|
||||
};
|
||||
}
|
||||
|
||||
export async function verifyOidcIdToken({
|
||||
idToken,
|
||||
issuer,
|
||||
audience,
|
||||
jwksUri,
|
||||
nonce,
|
||||
}) {
|
||||
const jwks = createRemoteJWKSet(new URL(jwksUri));
|
||||
const { payload } = await jwtVerify(idToken, jwks, {
|
||||
issuer,
|
||||
audience,
|
||||
nonce,
|
||||
});
|
||||
return payload;
|
||||
}
|
||||
|
||||
export function pickOidcDisplayName(payload = {}) {
|
||||
return payload.preferred_username || payload.email || payload.name || payload.given_name || payload.sub || "OIDC user";
|
||||
}
|
||||
|
||||
export function pickOidcEmail(payload = {}) {
|
||||
return payload.email || "";
|
||||
}
|
||||
|
|
@ -17,6 +17,12 @@ const DEFAULT_SETTINGS = {
|
|||
comboStrategies: {},
|
||||
requireLogin: true,
|
||||
tunnelDashboardAccess: true,
|
||||
authMode: "password",
|
||||
oidcIssuerUrl: "",
|
||||
oidcClientId: "",
|
||||
oidcClientSecret: "",
|
||||
oidcScopes: "openid profile email",
|
||||
oidcLoginLabel: "Sign in with OIDC",
|
||||
enableObservability: true,
|
||||
observabilityMaxRecords: 1000,
|
||||
observabilityBatchSize: 20,
|
||||
|
|
|
|||
|
|
@ -1,7 +1,7 @@
|
|||
"use client";
|
||||
|
||||
import { useEffect, useMemo, useState } from "react";
|
||||
import { usePathname, useRouter } from "next/navigation";
|
||||
import { useMemo } from "react";
|
||||
import Link from "next/link";
|
||||
import PropTypes from "prop-types";
|
||||
import ProviderIcon from "@/shared/components/ProviderIcon";
|
||||
|
|
@ -172,11 +172,39 @@ const getPageInfo = (pathname) => {
|
|||
export default function Header({ onMenuClick, showMenuButton = true }) {
|
||||
const pathname = usePathname();
|
||||
const router = useRouter();
|
||||
const [displayName, setDisplayName] = useState("");
|
||||
const [loginMethod, setLoginMethod] = useState("");
|
||||
|
||||
// Memoize page info to prevent unnecessary recalculations
|
||||
const pageInfo = useMemo(() => getPageInfo(pathname), [pathname]);
|
||||
const { title, description, icon, breadcrumbs } = pageInfo;
|
||||
|
||||
useEffect(() => {
|
||||
let cancelled = false;
|
||||
|
||||
async function loadAuthStatus() {
|
||||
try {
|
||||
const res = await fetch("/api/auth/status", { cache: "no-store" });
|
||||
if (!res.ok) return;
|
||||
const data = await res.json();
|
||||
if (!cancelled) {
|
||||
setDisplayName(data?.displayName || data?.oidcName || data?.oidcEmail || "");
|
||||
setLoginMethod(data?.loginMethod || "");
|
||||
}
|
||||
} catch {
|
||||
if (!cancelled) {
|
||||
setDisplayName("");
|
||||
setLoginMethod("");
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
loadAuthStatus();
|
||||
return () => {
|
||||
cancelled = true;
|
||||
};
|
||||
}, []);
|
||||
|
||||
const handleLogout = async () => {
|
||||
try {
|
||||
const res = await fetch("/api/auth/logout", { method: "POST" });
|
||||
|
|
@ -266,6 +294,17 @@ export default function Header({ onMenuClick, showMenuButton = true }) {
|
|||
|
||||
{/* Right actions */}
|
||||
<div className="flex items-center gap-1 shrink-0">
|
||||
{displayName && (
|
||||
<div className="hidden sm:flex items-center max-w-[220px] px-3 py-1.5 rounded-full border border-border bg-surface/70 text-xs text-text-muted truncate">
|
||||
<span className="material-symbols-outlined text-[14px] mr-1.5 text-primary">person</span>
|
||||
<span className="truncate">{displayName}</span>
|
||||
{loginMethod && (
|
||||
<span className="ml-2 shrink-0 rounded-full bg-primary/10 px-2 py-0.5 text-[10px] font-semibold uppercase tracking-wide text-primary">
|
||||
{loginMethod}
|
||||
</span>
|
||||
)}
|
||||
</div>
|
||||
)}
|
||||
<HeaderSearch />
|
||||
<ThemeToggle />
|
||||
<HeaderMenu onLogout={handleLogout} />
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue