feat: add FORM_LOGIN_ENABLED auth toggle (#334)

This commit is contained in:
Daniel Volz
2026-02-27 00:48:58 +01:00
committed by GitHub
parent 8b3901c1e1
commit 19ba4bb7d2
7 changed files with 61 additions and 32 deletions
+6 -6
View File
@@ -20,7 +20,7 @@ export interface User {
export interface AuthState {
authEnabled: boolean;
registrationEnabled: boolean;
localAuthEnabled: boolean;
formLoginEnabled: boolean;
oidcEnabled: boolean;
oidcProviderName: string;
hasUsers: boolean;
@@ -425,7 +425,7 @@ export function LoginForm({
</svg>
{t("auth.loginWithSSO", "Login with {{provider}}", { provider: authState.oidcProviderName || "SSO" })}
</button>
{authState?.localAuthEnabled && (
{authState?.formLoginEnabled && (
<div className="auth-divider">
<span>{t("auth.or", "or")}</span>
</div>
@@ -434,7 +434,7 @@ export function LoginForm({
)}
{/* Local Login Form - only show if local auth is enabled */}
{authState?.localAuthEnabled && (
{authState?.formLoginEnabled && (
<form onSubmit={handleSubmit} className="auth-form">
{error && <div className="auth-error">{error}</div>}
@@ -474,7 +474,7 @@ export function LoginForm({
</form>
)}
{authState?.registrationEnabled && authState?.localAuthEnabled && onSwitchToRegister && (
{authState?.registrationEnabled && authState?.formLoginEnabled && onSwitchToRegister && (
<div className="auth-links">
<button type="button" className="auth-link-btn" onClick={onSwitchToRegister}>
{t("auth.createAccount", "Create account")}
@@ -540,7 +540,7 @@ export function RegisterForm({ onSuccess, onSwitchToLogin }: { onSuccess?: () =>
</svg>
{t("auth.loginWithSSO", "Login with {{provider}}", { provider: authState.oidcProviderName || "SSO" })}
</button>
{authState?.localAuthEnabled && (
{authState?.formLoginEnabled && (
<div className="auth-divider">
<span>{t("auth.or", "or")}</span>
</div>
@@ -549,7 +549,7 @@ export function RegisterForm({ onSuccess, onSwitchToLogin }: { onSuccess?: () =>
)}
{/* Local Registration Form - only show if local auth is enabled */}
{authState?.localAuthEnabled && (
{authState?.formLoginEnabled && (
<form onSubmit={handleSubmit} className="auth-form">
{error && <div className="auth-error">{error}</div>}
@@ -36,7 +36,7 @@ describe("AppHeader", () => {
json: () =>
Promise.resolve({
authEnabled: false,
localAuthEnabled: true,
formLoginEnabled: true,
hasUsers: false,
needsSetup: false,
}),
@@ -171,7 +171,7 @@ describe("AppHeader", () => {
json: () =>
Promise.resolve({
authEnabled: false,
localAuthEnabled: true,
formLoginEnabled: true,
hasUsers: false,
needsSetup: false,
}),
@@ -205,7 +205,7 @@ describe("AppHeader", () => {
json: () =>
Promise.resolve({
authEnabled: false,
localAuthEnabled: true,
formLoginEnabled: true,
hasUsers: false,
needsSetup: false,
}),
@@ -239,7 +239,7 @@ describe("AppHeader", () => {
json: () =>
Promise.resolve({
authEnabled: false,
localAuthEnabled: true,
formLoginEnabled: true,
hasUsers: false,
needsSetup: false,
}),
@@ -322,7 +322,7 @@ describe("AppHeader", () => {
Promise.resolve({
authEnabled: true,
registrationEnabled: true,
localAuthEnabled: true,
formLoginEnabled: true,
oidcEnabled: false,
oidcProviderName: "",
hasUsers: true,
+11 -11
View File
@@ -11,7 +11,7 @@ describe("AuthProvider", () => {
vi.resetAllMocks();
(global.fetch as ReturnType<typeof vi.fn>).mockResolvedValue({
ok: true,
json: () => Promise.resolve({ authEnabled: true, localAuthEnabled: true }),
json: () => Promise.resolve({ authEnabled: true, formLoginEnabled: true }),
});
});
@@ -79,7 +79,7 @@ describe("AuthProvider", () => {
(global.fetch as ReturnType<typeof vi.fn>)
.mockResolvedValueOnce({
ok: true,
json: () => Promise.resolve({ authEnabled: false, localAuthEnabled: true }),
json: () => Promise.resolve({ authEnabled: false, formLoginEnabled: true }),
})
.mockResolvedValueOnce({ ok: false, status: 401 })
.mockResolvedValueOnce({ ok: true, status: 200 })
@@ -116,7 +116,7 @@ describe("AuthProvider", () => {
(global.fetch as ReturnType<typeof vi.fn>)
.mockResolvedValueOnce({
ok: true,
json: () => Promise.resolve({ authEnabled: true, localAuthEnabled: true }),
json: () => Promise.resolve({ authEnabled: true, formLoginEnabled: true }),
})
.mockResolvedValueOnce({ ok: true, status: 200, json: () => Promise.resolve({ id: 1, username: "tester" }) })
.mockResolvedValueOnce({ ok: false, status: 401 })
@@ -141,7 +141,7 @@ describe("AuthProvider", () => {
(global.fetch as ReturnType<typeof vi.fn>)
.mockResolvedValueOnce({
ok: true,
json: () => Promise.resolve({ authEnabled: true, localAuthEnabled: true }),
json: () => Promise.resolve({ authEnabled: true, formLoginEnabled: true }),
})
.mockResolvedValueOnce({ ok: true, json: () => Promise.resolve({ id: 1, username: "timer-user" }) })
.mockResolvedValueOnce({ ok: true, status: 200 });
@@ -167,7 +167,7 @@ describe("AuthProvider", () => {
describe("LoginForm", () => {
const mockAuthState = {
authEnabled: true,
localAuthEnabled: true,
formLoginEnabled: true,
oidcEnabled: false,
registrationEnabled: true,
hasUsers: true,
@@ -281,7 +281,7 @@ describe("LoginForm", () => {
json: () =>
Promise.resolve({
authEnabled: false,
localAuthEnabled: true,
formLoginEnabled: true,
oidcEnabled: false,
registrationEnabled: true,
hasUsers: true,
@@ -317,7 +317,7 @@ describe("LoginForm", () => {
describe("RegisterForm", () => {
const mockAuthState = {
authEnabled: true,
localAuthEnabled: true,
formLoginEnabled: true,
oidcEnabled: false,
registrationEnabled: true,
hasUsers: false,
@@ -404,7 +404,7 @@ describe("RegisterForm", () => {
json: () =>
Promise.resolve({
authEnabled: true,
localAuthEnabled: true,
formLoginEnabled: true,
oidcEnabled: false,
registrationEnabled: true,
hasUsers: false,
@@ -439,7 +439,7 @@ describe("RegisterForm", () => {
describe("AuthPage", () => {
const mockAuthState = {
authEnabled: true,
localAuthEnabled: true,
formLoginEnabled: true,
oidcEnabled: false,
registrationEnabled: true,
hasUsers: true,
@@ -504,7 +504,7 @@ describe("UserProfile", () => {
(global.fetch as ReturnType<typeof vi.fn>)
.mockResolvedValueOnce({
ok: true,
json: () => Promise.resolve({ authEnabled: true, localAuthEnabled: true }),
json: () => Promise.resolve({ authEnabled: true, formLoginEnabled: true }),
})
.mockResolvedValueOnce({
ok: true,
@@ -724,7 +724,7 @@ describe("AuthProvider methods", () => {
it("refreshUser retries after token refresh on 401", async () => {
vi.clearAllMocks();
(global.fetch as ReturnType<typeof vi.fn>)
.mockResolvedValueOnce({ ok: true, json: () => Promise.resolve({ authEnabled: false, localAuthEnabled: true }) })
.mockResolvedValueOnce({ ok: true, json: () => Promise.resolve({ authEnabled: false, formLoginEnabled: true }) })
.mockResolvedValueOnce({ ok: false, status: 401 })
.mockResolvedValueOnce({ ok: true })
.mockResolvedValueOnce({ ok: true, json: () => Promise.resolve({ id: 1, username: "refreshed-user" }) });