Mehrere Subdomains pro Benutzer + README

- subdomains-Tabelle (n DNS-Namen je Benutzer) inkl. Migration vom alten
  Einzel-Subdomain-Schema in database.init_db()
- Benutzeranlage/Verwaltung: mehrere Subdomains hinzufuegen/entfernen
- /nic/update aktualisiert alle Subdomains des Benutzers bzw. die per
  ?hostname= gewaehlte(n); eine Antwortzeile je Subdomain
- Dashboard/Users-Templates auf das neue Modell umgestellt
- README.md mit Setup, Plesk-Konfig, Router-Einrichtung und Endpoint-Doku
- .gitignore: __pycache__/ und *.pyc

Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
This commit is contained in:
Stefan Hacker
2026-06-06 14:13:48 +02:00
parent 2542cf5455
commit 9c631992af
7 changed files with 472 additions and 109 deletions
+17 -17
View File
@@ -10,19 +10,19 @@
<div class="row g-3 mb-4">
<div class="col-sm-4">
<div class="card text-center p-3">
<div class="fs-2 fw-bold text-primary">{{ users|length }}</div>
<div class="fs-2 fw-bold text-primary">{{ user_count }}</div>
<div class="text-muted small">Benutzer gesamt</div>
</div>
</div>
<div class="col-sm-4">
<div class="card text-center p-3">
<div class="fs-2 fw-bold text-success">{{ users|selectattr('active', 'equalto', 1)|list|length }}</div>
<div class="text-muted small">Aktiv</div>
<div class="fs-2 fw-bold text-success">{{ subdomains|length }}</div>
<div class="text-muted small">Subdomains</div>
</div>
</div>
<div class="col-sm-4">
<div class="card text-center p-3">
<div class="fs-2 fw-bold text-info">{{ users|sum(attribute='update_count') }}</div>
<div class="fs-2 fw-bold text-info">{{ subdomains|sum(attribute='update_count') }}</div>
<div class="text-muted small">Updates gesamt</div>
</div>
</div>
@@ -31,15 +31,15 @@
<!-- User table -->
<div class="card mb-4">
<div class="card-header d-flex justify-content-between align-items-center">
<span class="fw-semibold">Benutzer & aktuelle IPs</span>
<span class="fw-semibold">Subdomains & aktuelle IPs</span>
<a href="{{ url_for('users') }}" class="btn btn-sm btn-outline-primary">Verwalten</a>
</div>
<div class="card-body p-0">
<table class="table table-hover mb-0">
<thead class="table-light">
<tr>
<th>Subdomain</th>
<th>Hostname</th>
<th>DynDNS-User</th>
<th>Aktuelle IP</th>
<th>Letztes Update</th>
<th>Updates</th>
@@ -47,23 +47,23 @@
</tr>
</thead>
<tbody>
{% for u in users %}
{% for s in subdomains %}
<tr>
<td class="fw-semibold">{{ u.subdomain }}</td>
<td class="text-muted small">
{% if base_domain %}{{ u.subdomain }}.{{ base_domain }}{% else %}—{% endif %}
<td class="fw-semibold font-monospace small">
{% if base_domain %}{{ s.subdomain }}.{{ base_domain }}{% else %}{{ s.subdomain }}{% endif %}
</td>
<td>{{ s.username }}</td>
<td>
{% if u.current_ip %}
<span class="badge bg-secondary badge-ip">{{ u.current_ip }}</span>
{% if s.current_ip %}
<span class="badge bg-secondary badge-ip">{{ s.current_ip }}</span>
{% else %}
<span class="text-muted">noch kein Update</span>
{% endif %}
</td>
<td class="text-muted small">{{ u.last_updated or '—' }}</td>
<td>{{ u.update_count }}</td>
<td class="text-muted small">{{ s.last_updated or '—' }}</td>
<td>{{ s.update_count }}</td>
<td>
{% if u.active %}
{% if s.active %}
<span class="badge bg-success">Aktiv</span>
{% else %}
<span class="badge bg-secondary">Inaktiv</span>
@@ -71,7 +71,7 @@
</td>
</tr>
{% else %}
<tr><td colspan="6" class="text-center text-muted py-4">Noch keine Benutzer angelegt.</td></tr>
<tr><td colspan="6" class="text-center text-muted py-4">Noch keine Subdomains angelegt.</td></tr>
{% endfor %}
</tbody>
</table>
@@ -96,7 +96,7 @@
{% for l in logs %}
<tr>
<td class="text-muted small">{{ l.timestamp }}</td>
<td>{{ l.subdomain }} <span class="text-muted small">({{ l.dyndns_username }})</span></td>
<td>{{ l.subdomain or '—' }} <span class="text-muted small">({{ l.dyndns_username }})</span></td>
<td class="badge-ip text-muted small">{{ l.old_ip or '—' }}</td>
<td><span class="badge bg-secondary badge-ip">{{ l.new_ip }}</span></td>
<td>
+66 -38
View File
@@ -11,32 +11,35 @@
<div class="card">
<div class="card-body p-0">
<table class="table table-hover mb-0">
<table class="table table-hover mb-0 align-middle">
<thead class="table-light">
<tr>
<th>Subdomain</th>
<th>Hostname</th>
<th>DynDNS-User</th>
<th>Aktuelle IP</th>
<th>Letztes Update</th>
<th>Subdomains</th>
<th>Status</th>
<th></th>
</tr>
</thead>
<tbody>
{% for u in users %}
{% for item in users %}
{% set u = item.row %}
<tr>
<td class="fw-semibold">{{ u.subdomain }}</td>
<td class="text-muted small font-monospace">
{% if base_domain %}{{ u.subdomain }}.{{ base_domain }}{% else %}—{% endif %}
</td>
<td>{{ u.username }}</td>
<td class="fw-semibold">{{ u.username }}</td>
<td>
{% if u.current_ip %}
<span class="badge bg-secondary font-monospace">{{ u.current_ip }}</span>
{% else %}<span class="text-muted"></span>{% endif %}
{% for s in item.subdomains %}
<span class="badge bg-light text-dark border me-1 mb-1 font-monospace">
{% if base_domain %}{{ s.subdomain }}.{{ base_domain }}{% else %}{{ s.subdomain }}{% endif %}
{% if s.current_ip %}<span class="text-muted">· {{ s.current_ip }}</span>{% endif %}
<form method="post" action="{{ url_for('subdomain_delete', subdomain_id=s.id) }}"
class="d-inline" onsubmit="return confirm('Subdomain {{ s.subdomain }} löschen?');">
<button type="submit" class="btn btn-link btn-sm p-0 ms-1 text-danger" title="Subdomain löschen"
style="vertical-align: baseline;"><i class="bi bi-x-lg"></i></button>
</form>
</span>
{% else %}
<span class="text-muted small">keine</span>
{% endfor %}
</td>
<td class="text-muted small">{{ u.last_updated or '—' }}</td>
<td>
{% if u.active %}
<span class="badge bg-success">Aktiv</span>
@@ -46,9 +49,13 @@
</td>
<td class="text-end">
<div class="btn-group btn-group-sm">
<button class="btn btn-outline-success"
data-bs-toggle="modal" data-bs-target="#subModal{{ u.id }}"
title="Subdomain hinzufügen">
<i class="bi bi-plus-lg"></i>
</button>
<button class="btn btn-outline-secondary"
data-bs-toggle="modal"
data-bs-target="#editModal{{ u.id }}"
data-bs-toggle="modal" data-bs-target="#editModal{{ u.id }}"
title="Bearbeiten">
<i class="bi bi-pencil"></i>
</button>
@@ -58,8 +65,7 @@
</button>
</form>
<button class="btn btn-outline-danger"
data-bs-toggle="modal"
data-bs-target="#delModal{{ u.id }}"
data-bs-toggle="modal" data-bs-target="#delModal{{ u.id }}"
title="Löschen">
<i class="bi bi-trash"></i>
</button>
@@ -67,6 +73,35 @@
</td>
</tr>
<!-- Add-subdomain modal -->
<div class="modal fade" id="subModal{{ u.id }}" tabindex="-1">
<div class="modal-dialog">
<div class="modal-content">
<div class="modal-header">
<h5 class="modal-title">Subdomain(s) zu „{{ u.username }}" hinzufügen</h5>
<button type="button" class="btn-close" data-bs-dismiss="modal"></button>
</div>
<form method="post" action="{{ url_for('subdomain_add', user_id=u.id) }}">
<div class="modal-body">
<label class="form-label">Subdomain(s)</label>
<div class="input-group">
<textarea name="subdomains" class="form-control font-monospace" rows="3"
placeholder="mypc&#10;nas&#10;router" required></textarea>
{% if base_domain %}
<span class="input-group-text text-muted">.{{ base_domain }}</span>
{% endif %}
</div>
<div class="form-text">Mehrere durch Komma, Leerzeichen oder Zeilenumbruch trennen.</div>
</div>
<div class="modal-footer">
<button type="button" class="btn btn-secondary" data-bs-dismiss="modal">Abbrechen</button>
<button type="submit" class="btn btn-primary">Hinzufügen</button>
</div>
</form>
</div>
</div>
</div>
<!-- Edit modal -->
<div class="modal fade" id="editModal{{ u.id }}" tabindex="-1">
<div class="modal-dialog">
@@ -86,16 +121,8 @@
<label class="form-label">Neues Passwort <span class="text-muted">(leer = unverändert)</span></label>
<input name="password" type="password" class="form-control">
</div>
<div class="mb-3">
<label class="form-label">Subdomain</label>
<div class="input-group">
<input name="subdomain" type="text" class="form-control font-monospace"
value="{{ u.subdomain }}" required>
{% if base_domain %}
<span class="input-group-text text-muted">.{{ base_domain }}</span>
{% endif %}
</div>
</div>
<div class="form-text">Subdomains werden direkt in der Tabelle per
<i class="bi bi-plus-lg"></i> / <i class="bi bi-x-lg"></i> verwaltet.</div>
</div>
<div class="modal-footer">
<button type="button" class="btn btn-secondary" data-bs-dismiss="modal">Abbrechen</button>
@@ -115,8 +142,8 @@
<button type="button" class="btn-close" data-bs-dismiss="modal"></button>
</div>
<div class="modal-body">
<strong>{{ u.username }}</strong> ({{ u.subdomain }}) wirklich löschen?
Alle Update-Logs werden ebenfalls entfernt.
<strong>{{ u.username }}</strong> mit {{ item.subdomains|length }} Subdomain(s)
wirklich löschen? Alle Update-Logs werden ebenfalls entfernt.
</div>
<div class="modal-footer">
<button type="button" class="btn btn-secondary" data-bs-dismiss="modal">Abbrechen</button>
@@ -129,7 +156,7 @@
</div>
{% else %}
<tr><td colspan="7" class="text-center text-muted py-4">
<tr><td colspan="4" class="text-center text-muted py-4">
Noch keine Benutzer. <a href="#" data-bs-toggle="modal" data-bs-target="#addModal">Jetzt anlegen.</a>
</td></tr>
{% endfor %}
@@ -160,17 +187,18 @@
<div class="form-text">Wird im Speedport als „Passwort" eingetragen.</div>
</div>
<div class="mb-3">
<label class="form-label">Subdomain</label>
<label class="form-label">Subdomain(s)</label>
<div class="input-group">
<input name="subdomain" type="text" class="form-control font-monospace"
placeholder="mypc" required
pattern="[a-z0-9]([a-z0-9\-]*[a-z0-9])?"
title="Kleinbuchstaben, Ziffern und Bindestriche">
<textarea name="subdomains" class="form-control font-monospace" rows="3"
placeholder="mypc&#10;nas&#10;router" required></textarea>
{% if base_domain %}
<span class="input-group-text text-muted">.{{ base_domain }}</span>
{% endif %}
</div>
<div class="form-text">Nur Kleinbuchstaben, Ziffern und Bindestriche.</div>
<div class="form-text">
Eine oder mehrere — durch Komma, Leerzeichen oder Zeilenumbruch getrennt.
Nur Kleinbuchstaben, Ziffern, Bindestriche und Punkte.
</div>
</div>
</div>
<div class="modal-footer">