6eb75204b6
New controller features:
- Web-based setup wizard replaces docker-setup.sh interactive config
- Dual listener: :8080 (Traefik) + :8081 (direct HTTP for LAN)
- Drive scanner finds .felhom-infra-backup/ on all block devices
- Hub recovery pull (GET /api/v1/recovery/{id}) with retrieval password
- Fresh install: Hub config download or manual wizard
- CSRF protection, state persistence, Hungarian UI
- Local infra backup written to all connected drives after each backup cycle
- .felhom-infra-backup/backup.json + metadata.json with SHA256 checksum
- Hub verification: parse customer_blocked from report push response
- Limited mode after 7 days without verification
- Recovery info page on Settings + recovery-info.txt file generation
- Pending events queue: DR events sent to Hub on next report push
- docker-setup.sh v6.0.0: removed interactive wizard, minimal controller.yaml only
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
79 lines
3.3 KiB
HTML
79 lines
3.3 KiB
HTML
{{define "setup_restore_exec"}}
|
|
<!DOCTYPE html>
|
|
<html lang="hu">
|
|
<head>
|
|
<meta charset="UTF-8">
|
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
|
<title>Visszaállítás folyamatban — Felhom</title>
|
|
<link rel="stylesheet" href="/static/style.css">
|
|
</head>
|
|
<body class="login-body">
|
|
<div class="setup-container">
|
|
<div class="setup-header">
|
|
<img src="/static/felhom-logo.svg" alt="Felhom.eu" style="width: 120px;">
|
|
<h1>Visszaállítás</h1>
|
|
</div>
|
|
|
|
<div class="setup-card">
|
|
<ul class="step-list" id="steps">
|
|
<li><span class="spinner"></span> Indítás...</li>
|
|
</ul>
|
|
</div>
|
|
|
|
<div id="done-msg" style="display: none;">
|
|
<div class="alert alert-info">Visszaállítás sikeres! A vezérlőpult újraindul...</div>
|
|
</div>
|
|
<div id="error-msg" style="display: none;">
|
|
<div class="alert alert-error" id="error-text"></div>
|
|
<div style="display: flex; gap: 0.75rem; margin-top: 1rem;">
|
|
<a href="/setup/failed" class="btn btn-outline">Tovább</a>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<script>
|
|
(function() {
|
|
function poll() {
|
|
fetch('/setup/restore/status')
|
|
.then(function(r) { return r.json(); })
|
|
.then(function(data) {
|
|
var list = document.getElementById('steps');
|
|
if (data.steps && data.steps.length > 0) {
|
|
list.innerHTML = '';
|
|
data.steps.forEach(function(step) {
|
|
var li = document.createElement('li');
|
|
var icon = '';
|
|
if (step.status === 'done') icon = '<span class="step-done">✓</span>';
|
|
else if (step.status === 'running') icon = '<span class="spinner"></span>';
|
|
else if (step.status === 'failed') icon = '<span class="step-failed">✗</span>';
|
|
else icon = '<span style="color: var(--text-secondary);">○</span>';
|
|
li.innerHTML = icon + ' ' + step.label;
|
|
if (step.error) li.innerHTML += '<br><small style="color: var(--red, #f85149);">' + step.error + '</small>';
|
|
list.appendChild(li);
|
|
});
|
|
}
|
|
if (data.error) {
|
|
document.getElementById('error-msg').style.display = 'block';
|
|
document.getElementById('error-text').textContent = data.error;
|
|
return;
|
|
}
|
|
if (data.done) {
|
|
document.getElementById('done-msg').style.display = 'block';
|
|
setTimeout(function() { window.location.href = '/'; }, 5000);
|
|
return;
|
|
}
|
|
setTimeout(poll, 1500);
|
|
})
|
|
.catch(function() {
|
|
// Connection lost — controller may be restarting
|
|
document.getElementById('done-msg').style.display = 'block';
|
|
setTimeout(function() { window.location.href = '/'; }, 5000);
|
|
});
|
|
}
|
|
poll();
|
|
})();
|
|
</script>
|
|
</body>
|
|
</html>
|
|
{{end}}
|