Two user-visible polish passes on top of the walking-skeleton install: - Console welcome: live ISO's getty no longer shows the bare Arch prompt. `/etc/hostname` is now `proksi` so avahi advertises `proksi.local`; a systemd oneshot (`furtka-issue.service`, runs after network-online.target) regenerates `/etc/issue` via `/usr/local/bin/furtka-update-issue` to show both `http://proksi.local:5000` (preferred, via mDNS — avahi and nss-mdns are already in `packages.extra`) and the raw IP as a fallback for networks where mDNS is flaky. `agetty --reload` nudges the already- running login prompt to redraw. - /install/log now polls a JSON endpoint (`/install/log.json`) every 3 s instead of meta-refresh, so expanding the collapsed log `<details>` doesn't get eaten by the refresh. Noscript fallback keeps the meta-refresh for JS-off users. When the install finishes, the Done state shows a Reboot-now button that POSTs to `/install/reboot` (guarded server-side to only reboot once status is "done", so a panicked click mid-pacstrap can't brick the box). A confirm() reminds the user to pull the USB / eject the ISO first. End-to-end tested on a Proxmox VM 2026-04-14: boot → wizard → archinstall → Done state → Reboot now → VM came back up → login as created user → `docker ps` worked. Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
97 lines
4.2 KiB
HTML
97 lines
4.2 KiB
HTML
{% extends "base.html" %}
|
|
|
|
{% block title %}Installing… · Furtka Installer{% endblock %}
|
|
{% block head_extra %}
|
|
{# Fallback for users with JS disabled — otherwise the JS below takes over
|
|
and updates in-place so the log <details> doesn't re-collapse. #}
|
|
<noscript>
|
|
{% if progress.status == "running" %}<meta http-equiv="refresh" content="3">{% endif %}
|
|
</noscript>
|
|
{% endblock %}
|
|
{% block step_indicator %}<span class="step-indicator">Installing</span>{% endblock %}
|
|
|
|
{% block content %}
|
|
<h1 id="install-heading">
|
|
{% if progress.status == "done" %}Furtka is ready
|
|
{% elif progress.status == "error" %}Installation hit a snag
|
|
{% else %}Installing Furtka{% endif %}
|
|
</h1>
|
|
<p class="lede" id="install-lede">
|
|
{% if progress.status == "done" %}Installation finished. <strong>Remove the installer USB / eject the ISO</strong>, then click Reboot.
|
|
{% elif progress.status == "error" %}Something went wrong. Open the details below and share them so we can help.
|
|
{% else %}This takes a few minutes. Don't close this page or power off the machine.{% endif %}
|
|
</p>
|
|
|
|
{% if progress.status == "done" %}
|
|
<form method="post" action="{{ url_for('install_reboot') }}"
|
|
onsubmit="return confirm('Have you removed the installer USB / ejected the ISO? Click OK to reboot.');">
|
|
<div class="actions">
|
|
<button type="submit" class="btn btn-primary">Reboot now</button>
|
|
</div>
|
|
</form>
|
|
{% endif %}
|
|
|
|
<div class="progress" role="progressbar" aria-valuemin="0" aria-valuemax="100" aria-valuenow="{{ progress.percent }}">
|
|
<div id="progress-bar" class="progress-bar{% if progress.status == 'error' %} progress-bar-error{% endif %}{% if progress.status == 'done' %} progress-bar-done{% endif %}" style="width: {{ progress.percent }}%;"></div>
|
|
</div>
|
|
<p class="progress-phase"><span id="progress-phase">{{ progress.phase }}</span> · <span id="progress-percent">{{ progress.percent }}</span>%</p>
|
|
|
|
<details class="log-details">
|
|
<summary>Show details</summary>
|
|
<pre id="install-log" class="log">{{ log or "(waiting for install to start)" }}</pre>
|
|
</details>
|
|
|
|
<script>
|
|
(function () {
|
|
var initialStatus = {{ progress.status | tojson }};
|
|
if (initialStatus !== 'running') return;
|
|
|
|
var bar = document.getElementById('progress-bar');
|
|
var phaseEl = document.getElementById('progress-phase');
|
|
var percentEl = document.getElementById('progress-percent');
|
|
var logEl = document.getElementById('install-log');
|
|
var headingEl = document.getElementById('install-heading');
|
|
var ledeEl = document.getElementById('install-lede');
|
|
|
|
var HEADINGS = {
|
|
done: 'Furtka is ready',
|
|
error: 'Installation hit a snag',
|
|
};
|
|
var LEDES = {
|
|
done: 'Installation finished. Remove the USB / eject the installer image, then reboot.',
|
|
error: 'Something went wrong. Open the details below and share them so we can help.',
|
|
};
|
|
|
|
function tick() {
|
|
fetch('{{ url_for("install_log_json") }}', { cache: 'no-store' })
|
|
.then(function (r) { return r.json(); })
|
|
.then(function (data) {
|
|
var p = data.progress;
|
|
bar.style.width = p.percent + '%';
|
|
phaseEl.textContent = p.phase;
|
|
percentEl.textContent = p.percent;
|
|
if (logEl.textContent !== data.log) {
|
|
var atBottom = logEl.scrollTop + logEl.clientHeight >= logEl.scrollHeight - 8;
|
|
logEl.textContent = data.log || '(waiting for install to start)';
|
|
if (atBottom) logEl.scrollTop = logEl.scrollHeight;
|
|
}
|
|
if (p.status === 'done') {
|
|
// Reload so the server-rendered Done state (with the
|
|
// Reboot button) replaces the running-state markup.
|
|
window.location.reload();
|
|
return;
|
|
}
|
|
if (p.status === 'error') {
|
|
bar.classList.add('progress-bar-error');
|
|
headingEl.textContent = HEADINGS.error;
|
|
ledeEl.textContent = LEDES.error;
|
|
return;
|
|
}
|
|
setTimeout(tick, 3000);
|
|
})
|
|
.catch(function () { setTimeout(tick, 3000); });
|
|
}
|
|
setTimeout(tick, 3000);
|
|
})();
|
|
</script>
|
|
{% endblock %}
|