Files
IMAPSYNC/backend/static/index.html
T
2026-04-21 15:10:35 +02:00

1259 lines
46 KiB
HTML

<!DOCTYPE html>
<html lang="de">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>ImapSync Manager</title>
<style>
@import url('https://fonts.googleapis.com/css2?family=IBM+Plex+Mono:wght@300;400;500;600&family=IBM+Plex+Sans:wght@300;400;500;600&display=swap');
:root {
--bg: #0d1117;
--surface: #161b22;
--surface2: #1c2128;
--border: #30363d;
--border2: #21262d;
--text: #e6edf3;
--muted: #7d8590;
--accent: #2ea043;
--accent-dim:#1a6629;
--blue: #388bfd;
--blue-dim: #1b3d6e;
--warn: #d29922;
--danger: #f85149;
--danger-dim:#5c1a18;
--mono: 'IBM Plex Mono', monospace;
--sans: 'IBM Plex Sans', sans-serif;
}
* { box-sizing: border-box; margin: 0; padding: 0; }
body {
font-family: var(--sans);
background: var(--bg);
color: var(--text);
min-height: 100vh;
font-size: 14px;
}
/* ── Login ── */
#login-screen {
display: flex;
align-items: center;
justify-content: center;
min-height: 100vh;
background: var(--bg);
}
.login-box {
width: 360px;
background: var(--surface);
border: 1px solid var(--border);
border-radius: 8px;
padding: 40px;
}
.login-logo {
text-align: center;
margin-bottom: 32px;
}
.login-logo .brand {
font-family: var(--mono);
font-size: 11px;
letter-spacing: 4px;
text-transform: uppercase;
color: var(--muted);
display: block;
margin-bottom: 8px;
}
.login-logo h1 {
font-family: var(--mono);
font-size: 22px;
font-weight: 600;
color: var(--accent);
}
.login-logo .pulse {
display: inline-block;
width: 8px; height: 8px;
background: var(--accent);
border-radius: 50%;
margin-left: 8px;
animation: pulse 2s infinite;
}
@keyframes pulse {
0%,100% { opacity: 1; transform: scale(1); }
50% { opacity: 0.4; transform: scale(0.8); }
}
/* ── App Shell ── */
#app { display: none; min-height: 100vh; }
#app.visible { display: flex; }
.sidebar {
width: 220px;
background: var(--surface);
border-right: 1px solid var(--border);
display: flex;
flex-direction: column;
padding: 0;
flex-shrink: 0;
}
.sidebar-header {
padding: 20px 20px 16px;
border-bottom: 1px solid var(--border2);
}
.sidebar-header .brand {
font-family: var(--mono);
font-size: 10px;
letter-spacing: 3px;
color: var(--muted);
text-transform: uppercase;
}
.sidebar-header h2 {
font-family: var(--mono);
font-size: 16px;
font-weight: 600;
color: var(--accent);
margin-top: 4px;
}
.sidebar-header h2 .dot {
display: inline-block;
width: 6px; height: 6px;
background: var(--accent);
border-radius: 50%;
margin-left: 6px;
animation: pulse 2s infinite;
vertical-align: middle;
margin-bottom: 2px;
}
nav { flex: 1; padding: 12px 0; }
nav a {
display: flex;
align-items: center;
gap: 10px;
padding: 9px 20px;
color: var(--muted);
text-decoration: none;
font-size: 13px;
font-weight: 500;
border-left: 2px solid transparent;
transition: all 0.15s;
cursor: pointer;
}
nav a:hover { color: var(--text); background: var(--surface2); }
nav a.active { color: var(--text); border-left-color: var(--accent); background: var(--surface2); }
nav a .icon { font-size: 15px; width: 18px; text-align: center; }
nav .nav-section {
font-family: var(--mono);
font-size: 10px;
letter-spacing: 2px;
color: var(--muted);
padding: 16px 20px 6px;
text-transform: uppercase;
}
.sidebar-footer {
padding: 16px 20px;
border-top: 1px solid var(--border2);
display: flex;
align-items: center;
gap: 10px;
}
.user-avatar {
width: 28px; height: 28px;
background: var(--accent-dim);
border-radius: 50%;
display: flex; align-items: center; justify-content: center;
font-family: var(--mono);
font-size: 11px;
color: var(--accent);
font-weight: 600;
flex-shrink: 0;
}
.user-info { flex: 1; min-width: 0; }
.user-info .name { font-size: 12px; font-weight: 500; white-space: nowrap; overflow: hidden; text-overflow: ellipsis; }
.user-info .role { font-size: 10px; font-family: var(--mono); color: var(--muted); text-transform: uppercase; letter-spacing: 1px; }
.logout-btn {
background: none; border: none; color: var(--muted); cursor: pointer;
font-size: 14px; padding: 4px; border-radius: 4px;
transition: color 0.15s;
}
.logout-btn:hover { color: var(--danger); }
/* ── Main Content ── */
.main { flex: 1; display: flex; flex-direction: column; overflow: hidden; }
.topbar {
height: 52px;
border-bottom: 1px solid var(--border);
display: flex;
align-items: center;
padding: 0 24px;
gap: 12px;
background: var(--surface);
}
.page-title {
flex: 1;
font-size: 14px;
font-weight: 600;
font-family: var(--mono);
letter-spacing: 0.5px;
}
.content { flex: 1; overflow-y: auto; padding: 24px; }
/* ── Cards / Grid ── */
.stat-grid {
display: grid;
grid-template-columns: repeat(auto-fill, minmax(180px, 1fr));
gap: 12px;
margin-bottom: 24px;
}
.stat-card {
background: var(--surface);
border: 1px solid var(--border);
border-radius: 6px;
padding: 16px 18px;
}
.stat-card .label {
font-family: var(--mono);
font-size: 10px;
letter-spacing: 2px;
color: var(--muted);
text-transform: uppercase;
margin-bottom: 8px;
}
.stat-card .value {
font-family: var(--mono);
font-size: 28px;
font-weight: 600;
line-height: 1;
}
.stat-card .sub { font-size: 11px; color: var(--muted); margin-top: 4px; }
.stat-card.green .value { color: var(--accent); }
.stat-card.blue .value { color: var(--blue); }
.stat-card.yellow .value { color: var(--warn); }
.stat-card.red .value { color: var(--danger); }
/* ── Panel ── */
.panel {
background: var(--surface);
border: 1px solid var(--border);
border-radius: 6px;
margin-bottom: 20px;
}
.panel-header {
padding: 14px 18px;
border-bottom: 1px solid var(--border2);
display: flex;
align-items: center;
gap: 10px;
}
.panel-header h3 {
flex: 1;
font-family: var(--mono);
font-size: 12px;
letter-spacing: 1px;
text-transform: uppercase;
color: var(--muted);
}
.panel-body { padding: 0; }
/* ── Table ── */
table { width: 100%; border-collapse: collapse; }
th {
text-align: left;
padding: 10px 18px;
font-family: var(--mono);
font-size: 10px;
letter-spacing: 1.5px;
color: var(--muted);
text-transform: uppercase;
border-bottom: 1px solid var(--border2);
font-weight: 500;
}
td { padding: 11px 18px; border-bottom: 1px solid var(--border2); vertical-align: middle; }
tr:last-child td { border-bottom: none; }
tr:hover td { background: var(--surface2); }
/* ── Badges ── */
.badge {
display: inline-flex;
align-items: center;
gap: 5px;
padding: 3px 8px;
border-radius: 4px;
font-family: var(--mono);
font-size: 10px;
font-weight: 500;
letter-spacing: 0.5px;
text-transform: uppercase;
}
.badge-dot { width: 5px; height: 5px; border-radius: 50%; }
.badge.idle { background: rgba(125,133,144,0.15); color: var(--muted); }
.badge.idle .badge-dot { background: var(--muted); }
.badge.queued { background: rgba(210,153,34,0.15); color: var(--warn); }
.badge.queued .badge-dot { background: var(--warn); animation: pulse 1s infinite; }
.badge.running { background: rgba(56,139,253,0.15); color: var(--blue); }
.badge.running .badge-dot { background: var(--blue); animation: pulse 0.8s infinite; }
.badge.done { background: rgba(46,160,67,0.15); color: var(--accent); }
.badge.done .badge-dot { background: var(--accent); }
.badge.failed { background: rgba(248,81,73,0.15); color: var(--danger); }
.badge.failed .badge-dot { background: var(--danger); }
.badge.admin { background: rgba(56,139,253,0.15); color: var(--blue); }
.badge.operator { background: rgba(210,153,34,0.15); color: var(--warn); }
.badge.viewer { background: rgba(125,133,144,0.15); color: var(--muted); }
.badge.enabled { background: rgba(46,160,67,0.15); color: var(--accent); }
.badge.disabled { background: rgba(125,133,144,0.15); color: var(--muted); }
/* ── Buttons ── */
.btn {
display: inline-flex;
align-items: center;
gap: 6px;
padding: 7px 14px;
border-radius: 5px;
border: 1px solid transparent;
font-family: var(--sans);
font-size: 13px;
font-weight: 500;
cursor: pointer;
transition: all 0.15s;
white-space: nowrap;
text-decoration: none;
}
.btn-primary { background: var(--accent); color: #fff; border-color: var(--accent); }
.btn-primary:hover { background: #3fb950; }
.btn-outline { background: transparent; color: var(--text); border-color: var(--border); }
.btn-outline:hover { background: var(--surface2); border-color: var(--muted); }
.btn-danger { background: transparent; color: var(--danger); border-color: var(--danger-dim); }
.btn-danger:hover { background: rgba(248,81,73,0.1); }
.btn-sm { padding: 4px 10px; font-size: 12px; }
.btn-icon { padding: 5px 8px; font-size: 14px; }
/* ── Forms ── */
.form-grid { display: grid; grid-template-columns: 1fr 1fr; gap: 14px; }
.form-group { display: flex; flex-direction: column; gap: 6px; }
.form-group.full { grid-column: 1 / -1; }
label { font-size: 12px; font-weight: 500; color: var(--muted); font-family: var(--mono); letter-spacing: 0.5px; }
input, select, textarea {
background: var(--bg);
border: 1px solid var(--border);
border-radius: 5px;
color: var(--text);
padding: 8px 12px;
font-family: var(--sans);
font-size: 13px;
transition: border-color 0.15s;
width: 100%;
}
input:focus, select:focus, textarea:focus {
outline: none;
border-color: var(--blue);
}
input[type="checkbox"] { width: auto; }
textarea { resize: vertical; min-height: 80px; font-family: var(--mono); font-size: 12px; }
select option { background: var(--surface); }
.form-actions { display: flex; gap: 10px; justify-content: flex-end; margin-top: 20px; padding-top: 16px; border-top: 1px solid var(--border2); }
.section-divider {
grid-column: 1 / -1;
padding-top: 8px;
border-top: 1px solid var(--border2);
font-family: var(--mono);
font-size: 10px;
letter-spacing: 2px;
color: var(--muted);
text-transform: uppercase;
}
/* ── Modal ── */
.modal-overlay {
position: fixed; inset: 0;
background: rgba(0,0,0,0.7);
display: flex; align-items: center; justify-content: center;
z-index: 100;
backdrop-filter: blur(2px);
}
.modal {
background: var(--surface);
border: 1px solid var(--border);
border-radius: 8px;
width: 620px;
max-width: 95vw;
max-height: 90vh;
overflow-y: auto;
}
.modal-header {
padding: 18px 22px;
border-bottom: 1px solid var(--border2);
display: flex; align-items: center; gap: 12px;
}
.modal-header h3 { flex: 1; font-family: var(--mono); font-size: 14px; }
.modal-body { padding: 22px; }
.modal-close { background: none; border: none; color: var(--muted); cursor: pointer; font-size: 18px; padding: 4px; }
.modal-close:hover { color: var(--text); }
/* ── Log viewer ── */
.log-viewer {
background: #0a0d10;
border: 1px solid var(--border);
border-radius: 5px;
padding: 16px;
font-family: var(--mono);
font-size: 11px;
line-height: 1.7;
color: #8b949e;
max-height: 500px;
overflow-y: auto;
white-space: pre-wrap;
word-break: break-all;
}
.log-viewer .log-ok { color: var(--accent); }
.log-viewer .log-err { color: var(--danger); }
.log-viewer .log-warn { color: var(--warn); }
.log-viewer .log-info { color: var(--blue); }
/* ── Chart ── */
.chart-area { padding: 16px 18px; }
.bar-chart { display: flex; align-items: flex-end; gap: 4px; height: 80px; }
.bar-col { flex: 1; display: flex; flex-direction: column; align-items: center; gap: 3px; }
.bar { width: 100%; border-radius: 2px 2px 0 0; min-height: 2px; background: var(--accent-dim); transition: background 0.15s; }
.bar:hover { background: var(--accent); }
.bar-label { font-family: var(--mono); font-size: 9px; color: var(--muted); white-space: nowrap; }
.chart-legend { display: flex; gap: 16px; margin-top: 10px; }
.legend-item { display: flex; align-items: center; gap: 5px; font-size: 11px; color: var(--muted); }
.legend-dot { width: 8px; height: 8px; border-radius: 2px; }
/* ── Toast ── */
.toast-container {
position: fixed; bottom: 24px; right: 24px; z-index: 200;
display: flex; flex-direction: column; gap: 8px;
}
.toast {
padding: 12px 16px;
border-radius: 6px;
font-size: 13px;
border: 1px solid;
max-width: 320px;
animation: slideIn 0.2s ease;
}
@keyframes slideIn { from { transform: translateX(20px); opacity: 0; } }
.toast.success { background: rgba(46,160,67,0.2); border-color: var(--accent); color: var(--accent); }
.toast.error { background: rgba(248,81,73,0.2); border-color: var(--danger); color: var(--danger); }
.toast.info { background: rgba(56,139,253,0.2); border-color: var(--blue); color: var(--blue); }
/* ── Misc ── */
.mono { font-family: var(--mono); }
.text-muted { color: var(--muted); }
.text-sm { font-size: 12px; }
.flex { display: flex; align-items: center; }
.gap-8 { gap: 8px; }
.gap-12 { gap: 12px; }
.ml-auto { margin-left: auto; }
.empty-state { text-align: center; padding: 48px; color: var(--muted); }
.empty-state .icon { font-size: 36px; margin-bottom: 12px; }
.spinner { display: inline-block; width: 14px; height: 14px; border: 2px solid var(--border); border-top-color: var(--accent); border-radius: 50%; animation: spin 0.6s linear infinite; }
@keyframes spin { to { transform: rotate(360deg); } }
/* ── Activity feed ── */
.activity-item { display: flex; align-items: flex-start; gap: 10px; padding: 10px 18px; border-bottom: 1px solid var(--border2); }
.activity-item:last-child { border-bottom: none; }
.activity-dot { width: 6px; height: 6px; border-radius: 50%; margin-top: 5px; flex-shrink: 0; }
.activity-dot.done { background: var(--accent); }
.activity-dot.failed { background: var(--danger); }
.activity-dot.running { background: var(--blue); animation: pulse 0.8s infinite; }
.activity-meta { font-size: 11px; color: var(--muted); font-family: var(--mono); margin-top: 2px; }
</style>
</head>
<body>
<!-- Login -->
<div id="login-screen">
<div class="login-box">
<div class="login-logo">
<span class="brand">IMAP Sync Manager</span>
<h1>▸ IMS <span class="pulse"></span></h1>
</div>
<div class="form-group" style="margin-bottom:14px">
<label>BENUTZERNAME</label>
<input id="login-user" type="text" placeholder="admin" autocomplete="username">
</div>
<div class="form-group" style="margin-bottom:20px">
<label>PASSWORT</label>
<input id="login-pass" type="password" placeholder="••••••••" autocomplete="current-password">
</div>
<button class="btn btn-primary" style="width:100%;justify-content:center" onclick="doLogin()">Anmelden</button>
<div id="login-error" style="color:var(--danger);font-size:12px;text-align:center;margin-top:12px;display:none"></div>
<div style="text-align:center;margin-top:20px;font-size:11px;color:var(--muted);font-family:var(--mono)">
Standard: admin / admin
</div>
</div>
</div>
<!-- App -->
<div id="app">
<aside class="sidebar">
<div class="sidebar-header">
<div class="brand">IMAP Sync</div>
<h2>Manager <span class="dot"></span></h2>
</div>
<nav>
<div class="nav-section">Übersicht</div>
<a onclick="navigate('dashboard')" data-page="dashboard" class="active">
<span class="icon"></span> Dashboard
</a>
<div class="nav-section">Aufträge</div>
<a onclick="navigate('jobs')" data-page="jobs">
<span class="icon"></span> Sync-Jobs
</a>
<a onclick="navigate('runs')" data-page="runs">
<span class="icon"></span> Verlauf
</a>
<div class="nav-section" id="admin-nav" style="display:none">Administration</div>
<a onclick="navigate('users')" data-page="users" id="admin-link" style="display:none">
<span class="icon"></span> Benutzer
</a>
</nav>
<div class="sidebar-footer">
<div class="user-avatar" id="user-avatar">A</div>
<div class="user-info">
<div class="name" id="sidebar-username">-</div>
<div class="role" id="sidebar-role">-</div>
</div>
<button class="logout-btn" onclick="logout()" title="Abmelden"></button>
</div>
</aside>
<div class="main">
<div class="topbar">
<span class="page-title" id="page-title">◈ DASHBOARD</span>
<div id="topbar-actions"></div>
</div>
<div class="content" id="content"></div>
</div>
</div>
<!-- Modals -->
<div id="modal-container"></div>
<!-- Toasts -->
<div class="toast-container" id="toasts"></div>
<script>
// ── State ──────────────────────────────────────────────────────────────────
let state = { token: null, user: null, refreshTimer: null };
// ── API ────────────────────────────────────────────────────────────────────
const api = {
async req(method, path, body = null) {
const opts = {
method,
headers: { 'Content-Type': 'application/json' }
};
if (state.token) opts.headers['Authorization'] = 'Bearer ' + state.token;
if (body) opts.body = JSON.stringify(body);
const res = await fetch('/api' + path, opts);
if (res.status === 401) { logout(); return; }
const data = await res.json();
if (!res.ok) throw new Error(data.detail || 'API-Fehler');
return data;
},
get: (p) => api.req('GET', p),
post: (p, b) => api.req('POST', p, b),
put: (p, b) => api.req('PUT', p, b),
delete: (p) => api.req('DELETE', p),
};
// ── Auth ───────────────────────────────────────────────────────────────────
async function doLogin() {
const u = document.getElementById('login-user').value.trim();
const p = document.getElementById('login-pass').value;
if (!u || !p) return;
try {
const data = await api.post('/auth/login', { username: u, password: p });
state.token = data.token;
state.user = { username: data.username, role: data.role };
localStorage.setItem('ims_token', data.token);
startApp();
} catch(e) {
const el = document.getElementById('login-error');
el.textContent = e.message;
el.style.display = 'block';
}
}
document.getElementById('login-pass').addEventListener('keydown', e => {
if (e.key === 'Enter') doLogin();
});
function logout() {
state.token = null; state.user = null;
localStorage.removeItem('ims_token');
if (state.refreshTimer) clearInterval(state.refreshTimer);
document.getElementById('app').classList.remove('visible');
document.getElementById('login-screen').style.display = 'flex';
}
function startApp() {
document.getElementById('login-screen').style.display = 'none';
document.getElementById('app').classList.add('visible');
// Sidebar user info
document.getElementById('sidebar-username').textContent = state.user.username;
document.getElementById('sidebar-role').textContent = state.user.role;
document.getElementById('user-avatar').textContent = state.user.username[0].toUpperCase();
if (state.user.role === 'admin') {
document.getElementById('admin-nav').style.display = '';
document.getElementById('admin-link').style.display = '';
}
navigate('dashboard');
}
// Check saved token
(async () => {
const saved = localStorage.getItem('ims_token');
if (saved) {
state.token = saved;
try {
const me = await api.get('/auth/me');
state.user = { username: me.sub, role: me.role };
startApp();
} catch { logout(); }
}
})();
// ── Navigation ──────────────────────────────────────────────────────────────
function navigate(page) {
document.querySelectorAll('nav a').forEach(a => a.classList.remove('active'));
const link = document.querySelector(`nav a[data-page="${page}"]`);
if (link) link.classList.add('active');
if (state.refreshTimer) clearInterval(state.refreshTimer);
const titles = { dashboard:'◈ DASHBOARD', jobs:'⟳ SYNC-JOBS', runs:'◎ VERLAUF', users:'◉ BENUTZER' };
document.getElementById('page-title').textContent = titles[page] || page.toUpperCase();
document.getElementById('topbar-actions').innerHTML = '';
const pages = { dashboard: renderDashboard, jobs: renderJobs, runs: renderRuns, users: renderUsers };
if (pages[page]) pages[page]();
}
// ── Toast ──────────────────────────────────────────────────────────────────
function toast(msg, type = 'success') {
const t = document.createElement('div');
t.className = `toast ${type}`;
t.textContent = msg;
document.getElementById('toasts').appendChild(t);
setTimeout(() => t.remove(), 3500);
}
// ── Modal ──────────────────────────────────────────────────────────────────
function openModal(html) {
const overlay = document.createElement('div');
overlay.className = 'modal-overlay';
overlay.innerHTML = html;
overlay.addEventListener('click', e => { if (e.target === overlay) overlay.remove(); });
document.getElementById('modal-container').appendChild(overlay);
return overlay;
}
function closeModal() {
document.querySelector('.modal-overlay')?.remove();
}
// ── Dashboard ──────────────────────────────────────────────────────────────
async function renderDashboard() {
const content = document.getElementById('content');
content.innerHTML = '<div class="empty-state"><div class="spinner"></div></div>';
try {
const s = await api.get('/stats');
let dailyChart = '';
if (s.daily && s.daily.length > 0) {
const maxVal = Math.max(...s.daily.map(d => d.synced), 1);
dailyChart = s.daily.map(d => {
const h = Math.max(4, Math.round((d.synced / maxVal) * 76));
const label = d.day.substring(5); // MM-DD
return `<div class="bar-col">
<div class="bar" style="height:${h}px" title="${d.day}: ${d.synced} msgs, ${d.runs} runs"></div>
<div class="bar-label">${label}</div>
</div>`;
}).join('');
} else {
dailyChart = '<div style="color:var(--muted);font-size:12px;padding:24px 0">Noch keine Daten</div>';
}
const recentRows = (s.recent_runs || []).map(r => `
<div class="activity-item">
<div class="activity-dot ${r.status}"></div>
<div style="flex:1;min-width:0">
<div class="flex gap-8">
<span style="font-weight:500;font-size:13px">${esc(r.job_name)}</span>
<span class="badge ${r.status}">${r.status}</span>
</div>
<div class="activity-meta">
${fmtDate(r.started_at)} &nbsp;·&nbsp;
${r.messages_synced} Nachrichten
${r.duration_sec ? '&nbsp;·&nbsp;' + fmtDuration(r.duration_sec) : ''}
</div>
</div>
</div>
`).join('') || '<div class="empty-state" style="padding:24px">Keine Ausführungen</div>';
content.innerHTML = `
<div class="stat-grid">
<div class="stat-card blue">
<div class="label">Sync-Jobs</div>
<div class="value">${s.jobs.total}</div>
<div class="sub">${s.jobs.active} aktiv</div>
</div>
<div class="stat-card ${s.jobs.running > 0 ? 'blue' : ''}">
<div class="label">Laufend</div>
<div class="value">${s.jobs.running}</div>
<div class="sub">${s.jobs.queued} in Warteschlange</div>
</div>
<div class="stat-card green">
<div class="label">Nachr. synchronisiert</div>
<div class="value">${fmtNum(s.messages.synced)}</div>
<div class="sub">${s.runs.total} Ausführungen gesamt</div>
</div>
<div class="stat-card ${s.runs.failed > 0 ? 'red' : ''}">
<div class="label">Fehlgeschlagen</div>
<div class="value">${s.runs.failed}</div>
<div class="sub">Ausführungen</div>
</div>
</div>
<div style="display:grid;grid-template-columns:1fr 1fr;gap:16px">
<div class="panel">
<div class="panel-header"><h3>Aktivität (14 Tage)</h3></div>
<div class="chart-area">
<div class="bar-chart">${dailyChart}</div>
<div class="chart-legend">
<div class="legend-item"><div class="legend-dot" style="background:var(--accent)"></div> Sync. Nachrichten</div>
</div>
</div>
</div>
<div class="panel">
<div class="panel-header">
<h3>Letzte Ausführungen</h3>
<button class="btn btn-outline btn-sm" onclick="navigate('runs')">Alle →</button>
</div>
<div class="panel-body">${recentRows}</div>
</div>
</div>
`;
// Auto-refresh every 30s
state.refreshTimer = setInterval(renderDashboard, 30000);
} catch(e) {
content.innerHTML = `<div class="empty-state"><div>Fehler: ${e.message}</div></div>`;
}
}
// ── Jobs ───────────────────────────────────────────────────────────────────
async function renderJobs() {
const content = document.getElementById('content');
const canEdit = state.user.role !== 'viewer';
if (canEdit) {
document.getElementById('topbar-actions').innerHTML =
`<button class="btn btn-primary" onclick="showJobModal()">+ Neuer Job</button>`;
}
content.innerHTML = '<div class="empty-state"><div class="spinner"></div></div>';
try {
const jobs = await api.get('/jobs');
if (!jobs.length) {
content.innerHTML = `<div class="empty-state">
<div class="icon">⟳</div>
<div>Noch keine Sync-Jobs konfiguriert</div>
${canEdit ? '<button class="btn btn-primary" style="margin-top:16px" onclick="showJobModal()">+ Ersten Job erstellen</button>' : ''}
</div>`;
return;
}
content.innerHTML = `
<div class="panel">
<div class="panel-header"><h3>${jobs.length} Jobs</h3></div>
<table>
<thead>
<tr>
<th>Name</th>
<th>Quelle → Ziel</th>
<th>Status</th>
<th>Zeitplan</th>
<th>Letzte Ausführung</th>
<th>Synced</th>
<th></th>
</tr>
</thead>
<tbody>
${jobs.map(j => `
<tr>
<td>
<div style="font-weight:500">${esc(j.name)}</div>
<div class="text-muted text-sm mono">#${j.id}</div>
</td>
<td>
<div class="mono text-sm">${esc(j.src_host)}${esc(j.dst_host)}</div>
<div class="text-muted text-sm">${esc(j.src_user)}${esc(j.dst_user)}</div>
</td>
<td>
<div>${statusBadge(j.status)}</div>
<div style="margin-top:4px">${j.enabled ? '<span class="badge enabled">aktiv</span>' : '<span class="badge disabled">deaktiviert</span>'}</div>
</td>
<td class="mono text-sm">${j.schedule ? esc(j.schedule) : '<span class="text-muted">manuell</span>'}</td>
<td class="text-sm text-muted">${j.last_run ? fmtDate(j.last_run) : '—'}</td>
<td class="mono">${fmtNum(j.total_synced || 0)}</td>
<td>
<div class="flex gap-8">
${canEdit && j.status === 'idle' ? `<button class="btn btn-outline btn-sm" onclick="triggerJob(${j.id})">▶ Start</button>` : ''}
${canEdit && j.status === 'queued' ? `<button class="btn btn-outline btn-sm" onclick="stopJob(${j.id})">■ Stop</button>` : ''}
<button class="btn btn-outline btn-sm" onclick="navigate('runs');showRunsForJob(${j.id},'${esc(j.name)}')">Logs</button>
${canEdit ? `<button class="btn btn-outline btn-sm" onclick="showJobModal(${j.id})">✎</button>` : ''}
${canEdit ? `<button class="btn btn-danger btn-sm" onclick="deleteJob(${j.id}, '${esc(j.name)}')">✕</button>` : ''}
</div>
</td>
</tr>
`).join('')}
</tbody>
</table>
</div>
`;
// Auto-refresh every 15s
state.refreshTimer = setInterval(renderJobs, 15000);
} catch(e) {
content.innerHTML = `<div class="empty-state">Fehler: ${e.message}</div>`;
}
}
async function showJobModal(jobId = null) {
let job = null;
if (jobId) {
try { job = await api.get(`/jobs/${jobId}`); } catch(e) { toast(e.message, 'error'); return; }
}
const title = job ? `Job bearbeiten: ${job.name}` : 'Neuer Sync-Job';
const overlay = openModal(`
<div class="modal">
<div class="modal-header">
<h3>⟳ ${title}</h3>
<button class="modal-close" onclick="closeModal()">✕</button>
</div>
<div class="modal-body">
<div class="form-grid">
<div class="form-group full">
<label>JOB-NAME</label>
<input id="j-name" type="text" value="${esc(job?.name||'')}" placeholder="z.B. Max Mustermann Migration">
</div>
<div class="section-divider">Quell-Server (Source)</div>
<div class="form-group">
<label>HOST</label>
<input id="j-s-host" type="text" value="${esc(job?.src_host||'')}" placeholder="mail.example.com">
</div>
<div class="form-group">
<label>PORT</label>
<input id="j-s-port" type="number" value="${job?.src_port||993}">
</div>
<div class="form-group">
<label>BENUTZER</label>
<input id="j-s-user" type="text" value="${esc(job?.src_user||'')}" placeholder="user@example.com">
</div>
<div class="form-group">
<label>PASSWORT</label>
<input id="j-s-pass" type="password" value="${esc(job?.src_password||'')}" placeholder="••••••••">
</div>
<div class="form-group">
<label>SSL/TLS</label>
<select id="j-s-ssl">
<option value="1" ${(!job||job.src_ssl)?'selected':''}>SSL (empfohlen)</option>
<option value="0" ${(job&&!job.src_ssl)?'selected':''}>Kein SSL</option>
</select>
</div>
<div class="section-divider">Ziel-Server (Destination)</div>
<div class="form-group">
<label>HOST</label>
<input id="j-d-host" type="text" value="${esc(job?.dst_host||'')}" placeholder="mail.ziel.com">
</div>
<div class="form-group">
<label>PORT</label>
<input id="j-d-port" type="number" value="${job?.dst_port||993}">
</div>
<div class="form-group">
<label>BENUTZER</label>
<input id="j-d-user" type="text" value="${esc(job?.dst_user||'')}" placeholder="user@ziel.com">
</div>
<div class="form-group">
<label>PASSWORT</label>
<input id="j-d-pass" type="password" value="${esc(job?.dst_password||'')}" placeholder="••••••••">
</div>
<div class="form-group">
<label>SSL/TLS</label>
<select id="j-d-ssl">
<option value="1" ${(!job||job.dst_ssl)?'selected':''}>SSL (empfohlen)</option>
<option value="0" ${(job&&!job.dst_ssl)?'selected':''}>Kein SSL</option>
</select>
</div>
<div class="section-divider">Zeitplan & Optionen</div>
<div class="form-group">
<label>CRON-ZEITPLAN (optional)</label>
<input id="j-schedule" type="text" value="${esc(job?.schedule||'')}" placeholder="0 2 * * * (täglich 2 Uhr)">
</div>
<div class="form-group">
<label>STATUS</label>
<select id="j-enabled">
<option value="1" ${(!job||job.enabled)?'selected':''}>Aktiviert</option>
<option value="0" ${(job&&!job.enabled)?'selected':''}>Deaktiviert</option>
</select>
</div>
<div class="form-group full">
<label>EXTRA ARGUMENTE (imapsync)</label>
<input id="j-extra" type="text" value="${esc(job?.extra_args||'')}" placeholder="z.B. --delete2 --addheader">
</div>
</div>
<div class="form-actions">
<button class="btn btn-outline" onclick="closeModal()">Abbrechen</button>
<button class="btn btn-primary" onclick="saveJob(${jobId||'null'})">
${job ? 'Speichern' : 'Job erstellen'}
</button>
</div>
</div>
</div>
`);
}
async function saveJob(jobId) {
const data = {
name: document.getElementById('j-name').value.trim(),
src_host: document.getElementById('j-s-host').value.trim(),
src_port: parseInt(document.getElementById('j-s-port').value),
src_ssl: document.getElementById('j-s-ssl').value === '1',
src_user: document.getElementById('j-s-user').value.trim(),
src_password: document.getElementById('j-s-pass').value,
dst_host: document.getElementById('j-d-host').value.trim(),
dst_port: parseInt(document.getElementById('j-d-port').value),
dst_ssl: document.getElementById('j-d-ssl').value === '1',
dst_user: document.getElementById('j-d-user').value.trim(),
dst_password: document.getElementById('j-d-pass').value,
schedule: document.getElementById('j-schedule').value.trim() || null,
enabled: document.getElementById('j-enabled').value === '1',
extra_args: document.getElementById('j-extra').value.trim(),
};
if (!data.name || !data.src_host || !data.src_user || !data.dst_host || !data.dst_user) {
toast('Bitte alle Pflichtfelder ausfüllen', 'error'); return;
}
try {
if (jobId) {
await api.put(`/jobs/${jobId}`, data);
toast('Job aktualisiert');
} else {
await api.post('/jobs', data);
toast('Job erstellt');
}
closeModal();
renderJobs();
} catch(e) { toast(e.message, 'error'); }
}
async function triggerJob(id) {
try {
await api.post(`/jobs/${id}/trigger`);
toast('Job gestartet');
renderJobs();
} catch(e) { toast(e.message, 'error'); }
}
async function stopJob(id) {
try {
await api.post(`/jobs/${id}/stop`);
toast('Job gestoppt');
renderJobs();
} catch(e) { toast(e.message, 'error'); }
}
async function deleteJob(id, name) {
if (!confirm(`Job "${name}" wirklich löschen? Alle Verlaufs-Einträge werden ebenfalls gelöscht.`)) return;
try {
await api.delete(`/jobs/${id}`);
toast('Job gelöscht');
renderJobs();
} catch(e) { toast(e.message, 'error'); }
}
// ── Runs ───────────────────────────────────────────────────────────────────
let activeJobFilter = null;
function showRunsForJob(jobId, jobName) {
activeJobFilter = { id: jobId, name: jobName };
renderRuns();
}
async function renderRuns() {
const content = document.getElementById('content');
content.innerHTML = '<div class="empty-state"><div class="spinner"></div></div>';
try {
let runs = [];
if (activeJobFilter) {
runs = await api.get(`/jobs/${activeJobFilter.id}/runs?limit=100`);
document.getElementById('topbar-actions').innerHTML =
`<span class="text-muted text-sm">Filter: ${esc(activeJobFilter.name)}</span>
<button class="btn btn-outline btn-sm" onclick="activeJobFilter=null;renderRuns()">✕ Filter</button>`;
} else {
// Load all jobs and their runs
document.getElementById('topbar-actions').innerHTML = '';
const jobs = await api.get('/jobs');
for (const j of jobs.slice(0, 20)) {
const jRuns = await api.get(`/jobs/${j.id}/runs?limit=10`);
runs.push(...jRuns.map(r => ({...r, job_name: j.name})));
}
runs.sort((a, b) => b.started_at > a.started_at ? 1 : -1);
}
if (!runs.length) {
content.innerHTML = '<div class="empty-state"><div class="icon">◎</div><div>Keine Ausführungen gefunden</div></div>';
return;
}
content.innerHTML = `
<div class="panel">
<div class="panel-header"><h3>${runs.length} Ausführungen</h3></div>
<table>
<thead>
<tr>
<th>Run-ID</th>
<th>Job</th>
<th>Status</th>
<th>Gestartet</th>
<th>Dauer</th>
<th>Synced</th>
<th>Fehler</th>
<th></th>
</tr>
</thead>
<tbody>
${runs.map(r => `
<tr>
<td class="mono text-muted text-sm">#${r.id}</td>
<td>${esc(r.job_name || `Job #${r.job_id}`)}</td>
<td>${statusBadge(r.status)}</td>
<td class="text-sm text-muted">${fmtDate(r.started_at)}</td>
<td class="mono text-sm">${r.duration_sec ? fmtDuration(r.duration_sec) : r.status === 'running' ? '<span class="spinner"></span>' : '—'}</td>
<td class="mono">${fmtNum(r.messages_synced)}</td>
<td class="mono ${r.errors > 0 ? 'text-sm' : ''}" style="${r.errors > 0 ? 'color:var(--danger)' : ''}">${r.errors || 0}</td>
<td>
<button class="btn btn-outline btn-sm" onclick="showLog(${r.id})">📄 Log</button>
</td>
</tr>
`).join('')}
</tbody>
</table>
</div>
`;
} catch(e) {
content.innerHTML = `<div class="empty-state">Fehler: ${e.message}</div>`;
}
}
async function showLog(runId) {
try {
const data = await api.get(`/runs/${runId}/log`);
const highlighted = highlightLog(data.content);
openModal(`
<div class="modal" style="width:780px">
<div class="modal-header">
<h3>📄 Log — Run #${runId}</h3>
<button class="modal-close" onclick="closeModal()">✕</button>
</div>
<div class="modal-body">
<div class="log-viewer">${highlighted}</div>
</div>
</div>
`);
} catch(e) { toast('Log nicht verfügbar: ' + e.message, 'error'); }
}
function highlightLog(text) {
return esc(text)
.replace(/(ERR[A-Z]*|FAILED|ERROR)/gi, '<span class="log-err">$1</span>')
.replace(/(OK|SUCCESS|DONE|Synced|Transferred)/gi, '<span class="log-ok">$1</span>')
.replace(/(WARN[A-Z]*|SKIP[A-Z]*)/gi, '<span class="log-warn">$1</span>')
.replace(/(--[a-z]+[a-z0-9]*)/g, '<span class="log-info">$1</span>');
}
// ── Users ──────────────────────────────────────────────────────────────────
async function renderUsers() {
if (state.user.role !== 'admin') {
document.getElementById('content').innerHTML =
'<div class="empty-state">Keine Berechtigung</div>';
return;
}
document.getElementById('topbar-actions').innerHTML =
`<button class="btn btn-primary" onclick="showUserModal()">+ Benutzer erstellen</button>`;
const content = document.getElementById('content');
content.innerHTML = '<div class="empty-state"><div class="spinner"></div></div>';
try {
const users = await api.get('/users');
content.innerHTML = `
<div class="panel">
<div class="panel-header"><h3>${users.length} Benutzer</h3></div>
<table>
<thead>
<tr>
<th>Benutzer</th>
<th>Rolle</th>
<th>Erstellt am</th>
<th>Berechtigungen</th>
<th></th>
</tr>
</thead>
<tbody>
${users.map(u => `
<tr>
<td>
<div class="flex gap-8">
<div class="user-avatar" style="width:24px;height:24px;font-size:10px">${u.username[0].toUpperCase()}</div>
<span style="font-weight:500">${esc(u.username)}</span>
${u.username === state.user.username ? '<span class="text-muted text-sm">(ich)</span>' : ''}
</div>
</td>
<td><span class="badge ${u.role}">${u.role}</span></td>
<td class="text-sm text-muted">${fmtDate(u.created_at)}</td>
<td class="text-sm text-muted">${roleDesc(u.role)}</td>
<td>
<div class="flex gap-8">
<button class="btn btn-outline btn-sm" onclick="showUserModal(${u.id},'${esc(u.username)}','${u.role}')">✎ Bearbeiten</button>
${u.username !== state.user.username ?
`<button class="btn btn-danger btn-sm" onclick="deleteUser(${u.id},'${esc(u.username)}')">✕</button>` : ''}
</div>
</td>
</tr>
`).join('')}
</tbody>
</table>
</div>
<div class="panel" style="margin-top:16px">
<div class="panel-header"><h3>Rollenbeschreibung</h3></div>
<div style="padding:18px;display:grid;grid-template-columns:repeat(3,1fr);gap:16px">
<div>
<span class="badge admin" style="margin-bottom:8px">admin</span>
<div class="text-sm text-muted" style="margin-top:8px">Vollzugriff: Benutzer, Jobs, Logs erstellen/löschen</div>
</div>
<div>
<span class="badge operator" style="margin-bottom:8px">operator</span>
<div class="text-sm text-muted" style="margin-top:8px">Jobs erstellen/starten, Logs einsehen. Keine Benutzerverwaltung.</div>
</div>
<div>
<span class="badge viewer" style="margin-bottom:8px">viewer</span>
<div class="text-sm text-muted" style="margin-top:8px">Nur Lesezugriff: Statistiken und Logs ansehen</div>
</div>
</div>
</div>
`;
} catch(e) {
content.innerHTML = `<div class="empty-state">Fehler: ${e.message}</div>`;
}
}
function showUserModal(id = null, username = '', role = 'viewer') {
const isEdit = id !== null;
openModal(`
<div class="modal">
<div class="modal-header">
<h3>${isEdit ? `Benutzer bearbeiten: ${username}` : 'Neuer Benutzer'}</h3>
<button class="modal-close" onclick="closeModal()">✕</button>
</div>
<div class="modal-body">
<div class="form-grid">
${!isEdit ? `
<div class="form-group full">
<label>BENUTZERNAME</label>
<input id="u-name" type="text" placeholder="benutzername">
</div>
` : ''}
<div class="form-group ${!isEdit ? '' : 'full'}">
<label>${isEdit ? 'NEUES PASSWORT (leer = unverändert)' : 'PASSWORT'}</label>
<input id="u-pass" type="password" placeholder="••••••••">
</div>
<div class="form-group ${!isEdit ? '' : 'full'}">
<label>ROLLE</label>
<select id="u-role">
<option value="admin" ${role==='admin'?'selected':''}>Admin</option>
<option value="operator" ${role==='operator'?'selected':''}>Operator</option>
<option value="viewer" ${role==='viewer'?'selected':''}>Viewer</option>
</select>
</div>
</div>
<div class="form-actions">
<button class="btn btn-outline" onclick="closeModal()">Abbrechen</button>
<button class="btn btn-primary" onclick="saveUser(${id||'null'})">
${isEdit ? 'Speichern' : 'Erstellen'}
</button>
</div>
</div>
</div>
`);
}
async function saveUser(id) {
const pass = document.getElementById('u-pass').value;
const role = document.getElementById('u-role').value;
try {
if (id) {
const body = {};
if (pass) body.password = pass;
body.role = role;
await api.put(`/users/${id}`, body);
toast('Benutzer aktualisiert');
} else {
const name = document.getElementById('u-name').value.trim();
if (!name || !pass) { toast('Name und Passwort erforderlich', 'error'); return; }
await api.post('/users', { username: name, password: pass, role });
toast('Benutzer erstellt');
}
closeModal();
renderUsers();
} catch(e) { toast(e.message, 'error'); }
}
async function deleteUser(id, name) {
if (!confirm(`Benutzer "${name}" wirklich löschen?`)) return;
try {
await api.delete(`/users/${id}`);
toast('Benutzer gelöscht');
renderUsers();
} catch(e) { toast(e.message, 'error'); }
}
// ── Helpers ────────────────────────────────────────────────────────────────
function esc(str) {
return String(str || '').replace(/&/g,'&amp;').replace(/</g,'&lt;').replace(/>/g,'&gt;').replace(/"/g,'&quot;').replace(/'/g,'&#39;');
}
function fmtNum(n) { return Number(n||0).toLocaleString('de-DE'); }
function fmtDate(d) {
if (!d) return '—';
return new Date(d + (d.includes('Z')||d.includes('+') ? '' : 'Z'))
.toLocaleString('de-DE', { day:'2-digit', month:'2-digit', year:'2-digit', hour:'2-digit', minute:'2-digit' });
}
function fmtDuration(sec) {
if (sec < 60) return sec + 's';
if (sec < 3600) return Math.floor(sec/60) + 'm ' + (sec%60) + 's';
return Math.floor(sec/3600) + 'h ' + Math.floor((sec%3600)/60) + 'm';
}
function statusBadge(s) {
const labels = { idle:'Bereit', queued:'Warteschlange', running:'Läuft', done:'Fertig', failed:'Fehler' };
return `<span class="badge ${s}"><span class="badge-dot"></span>${labels[s]||s}</span>`;
}
function roleDesc(r) {
const d = { admin:'Vollzugriff', operator:'Jobs verwalten', viewer:'Nur Lesen' };
return d[r] || r;
}
</script>
</body>
</html>