Files
tiny-forge/web/src/routes/login/+page.svelte
T
alexei.dolgolyov 37cfa090ac feat: global Docker health indicator and graceful degradation
- GET /api/health endpoint returning Docker connectivity status
- Sidebar shows Docker connection dot (green=connected, red=disconnected)
- Stale scanner returns store-only results when Docker is unavailable
- Polls health every 30s
2026-03-30 13:43:33 +03:00

153 lines
5.6 KiB
Svelte

<script lang="ts">
import { goto } from '$app/navigation';
import { page } from '$app/stores';
import { onMount } from 'svelte';
import { t } from '$lib/i18n';
import { resolvedTheme, applyTheme } from '$lib/stores/theme';
import { IconLoader } from '$lib/components/icons';
import { setAuthToken, isAuthenticated } from '$lib/auth';
let username = $state('');
let password = $state('');
let error = $state('');
let loading = $state(false);
// Apply theme on login page too.
$effect(() => {
applyTheme($resolvedTheme);
});
onMount(async () => {
const urlToken = $page.url.searchParams.get('token');
if (urlToken) {
// Validate the token against the backend before trusting it.
try {
const res = await fetch('/api/auth/me', {
headers: { 'Authorization': `Bearer ${urlToken}` }
});
if (res.ok) {
setAuthToken(urlToken);
// Remove token from URL to prevent leakage via history/referrer.
history.replaceState(null, '', '/login');
goto('/');
return;
}
} catch {
// Token validation failed — fall through to login form.
}
// Remove invalid token from URL.
history.replaceState(null, '', '/login');
}
if (isAuthenticated()) {
goto('/');
}
});
async function handleLogin() {
error = '';
loading = true;
try {
const res = await fetch('/api/auth/login', {
method: 'POST',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({ username, password })
});
const envelope = await res.json();
if (!envelope.success) {
error = envelope.error ?? $t('login.loginFailed');
return;
}
setAuthToken(envelope.data.token);
goto('/');
} catch (err: unknown) {
error = err instanceof Error ? err.message : $t('login.networkError');
} finally {
loading = false;
}
}
function handleOIDCLogin() {
window.location.href = '/api/auth/oidc/login';
}
</script>
<div class="flex min-h-screen items-center justify-center bg-[var(--surface-page)] px-4">
<div class="w-full max-w-sm">
<div class="rounded-2xl border border-[var(--border-primary)] bg-[var(--surface-card)] p-8 shadow-[var(--shadow-lg)]">
<!-- Logo -->
<div class="mb-6 text-center">
<div class="mx-auto flex h-12 w-12 items-center justify-center rounded-xl bg-[var(--color-brand-600)] shadow-md">
<svg class="h-6 w-6 text-white" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
<path stroke-linecap="round" stroke-linejoin="round" d="M21 7.5l-2.25-1.313M21 7.5v2.25m0-2.25l-2.25 1.313M3 7.5l2.25-1.313M3 7.5l2.25 1.313M3 7.5v2.25m9 3l2.25-1.313M12 12.75l-2.25-1.313M12 12.75V15m0 6.75l2.25-1.313M12 21.75V19.5m0 2.25l-2.25-1.313m0-16.875L12 2.25l2.25 1.313M21 14.25v2.25l-2.25 1.313m-13.5 0L3 16.5v-2.25" />
</svg>
</div>
<h1 class="mt-4 text-xl font-bold text-[var(--text-primary)]">{$t('login.title')}</h1>
<p class="mt-1 text-sm text-[var(--text-secondary)]">{$t('login.subtitle')}</p>
</div>
{#if error}
<div class="mb-4 rounded-lg bg-[var(--color-danger-light)] p-3 text-sm text-[var(--color-danger)]">
{error}
</div>
{/if}
<form onsubmit={(e) => { e.preventDefault(); handleLogin(); }} class="space-y-4">
<div class="flex flex-col gap-1.5">
<label for="username" class="text-sm font-medium text-[var(--text-primary)]">{$t('login.username')}</label>
<input
id="username"
type="text"
bind:value={username}
required
autocomplete="username"
class="w-full rounded-lg border border-[var(--border-input)] bg-[var(--surface-input)] px-3 py-2.5 text-sm text-[var(--text-primary)] placeholder:text-[var(--text-tertiary)] focus:border-[var(--color-brand-500)] focus:outline-none focus:ring-2 focus:ring-[var(--color-brand-500)]"
/>
</div>
<div class="flex flex-col gap-1.5">
<label for="password" class="text-sm font-medium text-[var(--text-primary)]">{$t('login.password')}</label>
<input
id="password"
type="password"
bind:value={password}
required
autocomplete="current-password"
class="w-full rounded-lg border border-[var(--border-input)] bg-[var(--surface-input)] px-3 py-2.5 text-sm text-[var(--text-primary)] placeholder:text-[var(--text-tertiary)] focus:border-[var(--color-brand-500)] focus:outline-none focus:ring-2 focus:ring-[var(--color-brand-500)]"
/>
</div>
<button
type="submit"
disabled={loading}
class="w-full inline-flex items-center justify-center gap-2 rounded-lg bg-[var(--color-brand-600)] px-4 py-2.5 text-sm font-medium text-white shadow-sm transition-all duration-150 hover:bg-[var(--color-brand-700)] focus:outline-none focus:ring-2 focus:ring-[var(--color-brand-500)] focus:ring-offset-2 disabled:opacity-50 active:animate-press"
>
{#if loading}
<IconLoader size={16} />
{$t('login.signingIn')}
{:else}
{$t('login.signIn')}
{/if}
</button>
</form>
<div class="mt-5">
<div class="relative">
<div class="absolute inset-0 flex items-center">
<div class="w-full border-t border-[var(--border-primary)]"></div>
</div>
<div class="relative flex justify-center text-xs">
<span class="bg-[var(--surface-card)] px-3 text-[var(--text-tertiary)]">{$t('login.or')}</span>
</div>
</div>
<button
onclick={handleOIDCLogin}
class="mt-4 w-full rounded-lg border border-[var(--border-primary)] bg-[var(--surface-card)] px-4 py-2.5 text-sm font-medium text-[var(--text-secondary)] shadow-sm transition-all duration-150 hover:bg-[var(--surface-card-hover)] focus:outline-none focus:ring-2 focus:ring-[var(--color-brand-500)] focus:ring-offset-2"
>
{$t('login.ssoButton')}
</button>
</div>
</div>
</div>
</div>