feat(docker-watcher): phase 9 - SvelteKit dashboard & project views

SvelteKit project with Svelte 5, TypeScript, Tailwind CSS v4.
Dashboard with project cards, project detail with stage/instance
management, deploy history, instance controls. Shared API client
and reusable components (StatusBadge, InstanceCard, ProjectCard,
ConfirmDialog). Add Phase 14 (Volumes & Environment) to plan.
This commit is contained in:
2026-03-27 22:15:54 +03:00
parent 757c72eea1
commit 09d185d94e
13 changed files with 1787 additions and 53 deletions
+278
View File
@@ -0,0 +1,278 @@
<script lang="ts">
import { getSettings, updateSettings, getWebhookUrl, regenerateWebhookUrl } from '$lib/api';
import type { Settings } from '$lib/types';
import FormField from '$lib/components/FormField.svelte';
import { toasts } from '$lib/stores/toast';
let loading = $state(true);
let saving = $state(false);
let webhookUrl = $state('');
let regenerating = $state(false);
// Settings fields
let domain = $state('');
let serverIp = $state('');
let network = $state('');
let subdomainPattern = $state('');
let pollingInterval = $state('');
let notificationUrl = $state('');
let errors = $state<Record<string, string>>({});
function validateDomain(value: string): string {
if (!value.trim()) return 'Domain is required';
if (!/^[a-zA-Z0-9][a-zA-Z0-9.\-]+\.[a-zA-Z]{2,}$/.test(value.trim())) {
return 'Invalid domain format';
}
return '';
}
function validateIp(value: string): string {
if (!value.trim()) return '';
if (!/^(\d{1,3}\.){3}\d{1,3}$/.test(value.trim())) {
return 'Invalid IP address format';
}
return '';
}
function validatePollingInterval(value: string): string {
if (!value.trim()) return '';
const num = parseInt(value, 10);
if (isNaN(num) || num < 10 || num > 86400) {
return 'Polling interval must be between 10 and 86400 seconds';
}
return '';
}
function validateUrl(value: string): string {
if (!value.trim()) return '';
try {
new URL(value.trim());
return '';
} catch {
return 'Invalid URL format';
}
}
function validateAll(): boolean {
const newErrors: Record<string, string> = {};
const domainErr = validateDomain(domain);
if (domainErr) newErrors.domain = domainErr;
const ipErr = validateIp(serverIp);
if (ipErr) newErrors.serverIp = ipErr;
const intervalErr = validatePollingInterval(pollingInterval);
if (intervalErr) newErrors.pollingInterval = intervalErr;
const urlErr = validateUrl(notificationUrl);
if (urlErr) newErrors.notificationUrl = urlErr;
errors = newErrors;
return Object.keys(newErrors).length === 0;
}
async function loadSettings() {
loading = true;
try {
const settings = await getSettings();
domain = settings.domain ?? '';
serverIp = settings.server_ip ?? '';
network = settings.network ?? '';
subdomainPattern = settings.subdomain_pattern ?? '';
pollingInterval = settings.polling_interval ?? '';
notificationUrl = settings.notification_url ?? '';
} catch (err) {
const message = err instanceof Error ? err.message : 'Failed to load settings';
toasts.error(message);
} finally {
loading = false;
}
}
async function loadWebhookUrlValue() {
try {
const result = await getWebhookUrl();
webhookUrl = result.url;
} catch {
// Webhook URL may not be configured yet
}
}
async function handleSave() {
if (!validateAll()) return;
saving = true;
try {
const payload: Partial<Settings> = {
domain: domain.trim(),
server_ip: serverIp.trim(),
network: network.trim(),
subdomain_pattern: subdomainPattern.trim(),
polling_interval: pollingInterval.trim(),
notification_url: notificationUrl.trim()
};
await updateSettings(payload);
toasts.success('Settings saved successfully');
} catch (err) {
const message = err instanceof Error ? err.message : 'Failed to save settings';
toasts.error(message);
} finally {
saving = false;
}
}
async function handleRegenerateWebhook() {
regenerating = true;
try {
const result = await regenerateWebhookUrl();
webhookUrl = result.url;
toasts.success('Webhook URL regenerated');
} catch (err) {
const message = err instanceof Error ? err.message : 'Failed to regenerate webhook URL';
toasts.error(message);
} finally {
regenerating = false;
}
}
$effect(() => {
loadSettings();
loadWebhookUrlValue();
});
</script>
<svelte:head>
<title>General Settings - Docker Watcher</title>
</svelte:head>
<div class="space-y-6">
{#if loading}
<div class="flex items-center justify-center py-12">
<svg class="h-8 w-8 animate-spin text-blue-600" viewBox="0 0 24 24" fill="none">
<circle
class="opacity-25"
cx="12"
cy="12"
r="10"
stroke="currentColor"
stroke-width="4"
></circle>
<path
class="opacity-75"
fill="currentColor"
d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4z"
></path>
</svg>
</div>
{:else}
<!-- Global settings form -->
<div class="rounded-lg border border-gray-200 bg-white p-6 shadow-sm">
<h2 class="mb-4 text-lg font-semibold text-gray-800">Global Configuration</h2>
<div class="grid grid-cols-1 gap-4 md:grid-cols-2">
<FormField
label="Domain"
name="domain"
bind:value={domain}
placeholder="example.com"
required
error={errors.domain ?? ''}
helpText="Base domain for subdomain routing"
/>
<FormField
label="Server IP"
name="serverIp"
bind:value={serverIp}
placeholder="93.84.96.191"
error={errors.serverIp ?? ''}
helpText="Public IP address of the server"
/>
<FormField
label="Docker Network"
name="network"
bind:value={network}
placeholder="staging-net"
helpText="Docker network for deployed containers"
/>
<FormField
label="Subdomain Pattern"
name="subdomainPattern"
bind:value={subdomainPattern}
placeholder="stage-{stage}-{project}"
helpText="Pattern for auto-generated subdomains"
/>
<FormField
label="Polling Interval (seconds)"
name="pollingInterval"
type="number"
bind:value={pollingInterval}
placeholder="60"
error={errors.pollingInterval ?? ''}
helpText="How often to check registries for new tags (10-86400)"
/>
<FormField
label="Notification URL"
name="notificationUrl"
bind:value={notificationUrl}
placeholder="https://notify.example.com/webhook"
error={errors.notificationUrl ?? ''}
helpText="Webhook URL for deploy notifications"
/>
</div>
<div class="mt-6">
<button
onclick={handleSave}
disabled={saving}
class="rounded-md bg-blue-600 px-4 py-2 text-sm font-medium text-white transition-colors hover:bg-blue-700 disabled:cursor-not-allowed disabled:opacity-50"
>
{saving ? 'Saving...' : 'Save Settings'}
</button>
</div>
</div>
<!-- Webhook URL section -->
<div class="rounded-lg border border-gray-200 bg-white p-6 shadow-sm">
<h2 class="mb-4 text-lg font-semibold text-gray-800">Webhook URL</h2>
<p class="mb-3 text-sm text-gray-500">
This secret URL receives image push notifications from your CI pipeline.
</p>
{#if webhookUrl}
<div class="flex items-center gap-3">
<code
class="flex-1 rounded-md border border-gray-200 bg-gray-50 px-3 py-2 text-sm font-mono text-gray-700 break-all"
>
{webhookUrl}
</code>
<button
onclick={() => {
navigator.clipboard.writeText(webhookUrl);
toasts.info('Webhook URL copied to clipboard');
}}
class="rounded-md border border-gray-300 px-3 py-2 text-sm font-medium text-gray-700 transition-colors hover:bg-gray-50"
>
Copy
</button>
</div>
{:else}
<p class="text-sm text-gray-400 italic">No webhook URL configured</p>
{/if}
<div class="mt-4">
<button
onclick={handleRegenerateWebhook}
disabled={regenerating}
class="rounded-md border border-red-300 px-4 py-2 text-sm font-medium text-red-600 transition-colors hover:bg-red-50 disabled:opacity-50"
>
{regenerating ? 'Regenerating...' : 'Regenerate URL'}
</button>
<p class="mt-1 text-xs text-gray-500">
Warning: regenerating will invalidate the current URL. Update your CI pipelines.
</p>
</div>
</div>
{/if}
</div>