0aec9e6e54
- ACME TLS challenge on 443 (no port 80 or nginx decoy) - Auto-generate and persist obfs password per server - Update client export, HTML catalog, and vault examples - Document Salamander vs main and ACME auto-renewal in README
335 lines
10 KiB
Django/Jinja
335 lines
10 KiB
Django/Jinja
<!DOCTYPE html>
|
|
<html lang="ru">
|
|
<head>
|
|
<meta charset="UTF-8">
|
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
|
<title>Hysteria2 — {{ hysteria2_output_name }}</title>
|
|
<style>
|
|
:root {
|
|
color-scheme: light dark;
|
|
--bg: #0f1419;
|
|
--bg-card: #1a2332;
|
|
--bg-input: #0d1117;
|
|
--border: #2d3a4f;
|
|
--text: #e6edf3;
|
|
--muted: #8b949e;
|
|
--accent: #58a6ff;
|
|
--accent-soft: rgba(88, 166, 255, 0.15);
|
|
--success: #3fb950;
|
|
--success-soft: rgba(63, 185, 80, 0.15);
|
|
--radius: 12px;
|
|
--shadow: 0 8px 32px rgba(0, 0, 0, 0.35);
|
|
}
|
|
@media (prefers-color-scheme: light) {
|
|
:root {
|
|
--bg: #f0f4f8;
|
|
--bg-card: #ffffff;
|
|
--bg-input: #f6f8fa;
|
|
--border: #d0d7de;
|
|
--text: #1f2328;
|
|
--muted: #656d76;
|
|
--accent: #0969da;
|
|
--accent-soft: rgba(9, 105, 218, 0.1);
|
|
--success: #1a7f37;
|
|
--success-soft: rgba(26, 127, 55, 0.1);
|
|
--shadow: 0 8px 24px rgba(31, 35, 40, 0.08);
|
|
}
|
|
}
|
|
* { box-sizing: border-box; margin: 0; padding: 0; }
|
|
body {
|
|
font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "Helvetica Neue", Arial, sans-serif;
|
|
background: var(--bg);
|
|
color: var(--text);
|
|
line-height: 1.5;
|
|
min-height: 100vh;
|
|
padding: 2rem 1rem 3rem;
|
|
}
|
|
.container { max-width: 720px; margin: 0 auto; }
|
|
header {
|
|
text-align: center;
|
|
margin-bottom: 2.5rem;
|
|
padding: 2rem 1.5rem;
|
|
background: linear-gradient(135deg, var(--accent-soft), transparent);
|
|
border: 1px solid var(--border);
|
|
border-radius: var(--radius);
|
|
box-shadow: var(--shadow);
|
|
}
|
|
header h1 {
|
|
font-size: 1.75rem;
|
|
font-weight: 700;
|
|
letter-spacing: -0.02em;
|
|
margin-bottom: 0.5rem;
|
|
}
|
|
header .meta {
|
|
color: var(--muted);
|
|
font-size: 0.9rem;
|
|
}
|
|
header .meta strong { color: var(--text); font-weight: 600; }
|
|
.badge {
|
|
display: inline-block;
|
|
margin-top: 1rem;
|
|
padding: 0.25rem 0.75rem;
|
|
background: var(--accent-soft);
|
|
color: var(--accent);
|
|
border-radius: 999px;
|
|
font-size: 0.8rem;
|
|
font-weight: 600;
|
|
}
|
|
.badge-salamander {
|
|
margin-left: 0.35rem;
|
|
background: rgba(163, 113, 247, 0.18);
|
|
color: #a371f7;
|
|
}
|
|
.server-obfs {
|
|
margin-top: 1.25rem;
|
|
max-width: 100%;
|
|
}
|
|
.user-card {
|
|
background: var(--bg-card);
|
|
border: 1px solid var(--border);
|
|
border-radius: var(--radius);
|
|
box-shadow: var(--shadow);
|
|
padding: 1.5rem;
|
|
margin-bottom: 1.25rem;
|
|
}
|
|
.user-card h2 {
|
|
font-size: 1.25rem;
|
|
margin-bottom: 1rem;
|
|
display: flex;
|
|
align-items: center;
|
|
gap: 0.5rem;
|
|
}
|
|
.user-card h2::before {
|
|
content: "";
|
|
width: 8px;
|
|
height: 8px;
|
|
border-radius: 50%;
|
|
background: var(--success);
|
|
box-shadow: 0 0 8px var(--success);
|
|
}
|
|
.field { margin-bottom: 1rem; }
|
|
.field label {
|
|
display: block;
|
|
font-size: 0.75rem;
|
|
font-weight: 600;
|
|
text-transform: uppercase;
|
|
letter-spacing: 0.05em;
|
|
color: var(--muted);
|
|
margin-bottom: 0.4rem;
|
|
}
|
|
.input-row {
|
|
display: flex;
|
|
gap: 0.5rem;
|
|
align-items: stretch;
|
|
}
|
|
.input-row input {
|
|
flex: 1;
|
|
min-width: 0;
|
|
padding: 0.65rem 0.85rem;
|
|
background: var(--bg-input);
|
|
border: 1px solid var(--border);
|
|
border-radius: 8px;
|
|
color: var(--text);
|
|
font-family: ui-monospace, SFMono-Regular, "SF Mono", Menlo, Consolas, monospace;
|
|
font-size: 0.8rem;
|
|
}
|
|
.input-row input:focus {
|
|
outline: none;
|
|
border-color: var(--accent);
|
|
box-shadow: 0 0 0 3px var(--accent-soft);
|
|
}
|
|
.btn-copy {
|
|
display: flex;
|
|
align-items: center;
|
|
justify-content: center;
|
|
width: 42px;
|
|
flex-shrink: 0;
|
|
background: var(--bg-input);
|
|
border: 1px solid var(--border);
|
|
border-radius: 8px;
|
|
color: var(--muted);
|
|
cursor: pointer;
|
|
transition: color 0.15s, border-color 0.15s, background 0.15s;
|
|
}
|
|
.btn-copy:hover {
|
|
color: var(--accent);
|
|
border-color: var(--accent);
|
|
background: var(--accent-soft);
|
|
}
|
|
.btn-copy.copied {
|
|
color: var(--success);
|
|
border-color: var(--success);
|
|
background: var(--success-soft);
|
|
}
|
|
.btn-copy svg { width: 18px; height: 18px; }
|
|
.qr-block {
|
|
text-align: center;
|
|
margin: 1.25rem 0;
|
|
padding: 1rem;
|
|
background: #fff;
|
|
border-radius: 8px;
|
|
border: 1px solid var(--border);
|
|
}
|
|
@media (prefers-color-scheme: dark) {
|
|
.qr-block { background: #fff; }
|
|
}
|
|
.qr-block img {
|
|
max-width: 220px;
|
|
width: 100%;
|
|
height: auto;
|
|
image-rendering: pixelated;
|
|
}
|
|
.files {
|
|
display: flex;
|
|
flex-wrap: wrap;
|
|
gap: 0.5rem;
|
|
margin-top: 1rem;
|
|
padding-top: 1rem;
|
|
border-top: 1px solid var(--border);
|
|
}
|
|
.files a {
|
|
display: inline-flex;
|
|
align-items: center;
|
|
gap: 0.35rem;
|
|
padding: 0.35rem 0.75rem;
|
|
background: var(--bg-input);
|
|
border: 1px solid var(--border);
|
|
border-radius: 6px;
|
|
color: var(--accent);
|
|
text-decoration: none;
|
|
font-size: 0.8rem;
|
|
font-weight: 500;
|
|
transition: background 0.15s, border-color 0.15s;
|
|
}
|
|
.files a:hover {
|
|
background: var(--accent-soft);
|
|
border-color: var(--accent);
|
|
}
|
|
footer {
|
|
text-align: center;
|
|
margin-top: 2rem;
|
|
color: var(--muted);
|
|
font-size: 0.8rem;
|
|
}
|
|
.toast {
|
|
position: fixed;
|
|
bottom: 1.5rem;
|
|
left: 50%;
|
|
transform: translateX(-50%) translateY(100px);
|
|
background: var(--bg-card);
|
|
border: 1px solid var(--success);
|
|
color: var(--success);
|
|
padding: 0.6rem 1.2rem;
|
|
border-radius: 999px;
|
|
font-size: 0.85rem;
|
|
font-weight: 600;
|
|
box-shadow: var(--shadow);
|
|
opacity: 0;
|
|
transition: transform 0.25s, opacity 0.25s;
|
|
pointer-events: none;
|
|
z-index: 100;
|
|
}
|
|
.toast.show {
|
|
transform: translateX(-50%) translateY(0);
|
|
opacity: 1;
|
|
}
|
|
</style>
|
|
</head>
|
|
<body>
|
|
<div class="container">
|
|
<header>
|
|
<h1>Hysteria2</h1>
|
|
<p class="meta">
|
|
Сервер: <strong>{{ hysteria2_output_name }}</strong><br>
|
|
Домен: <strong>{{ hysteria2_domain }}:{{ hysteria2_listen_port }}</strong>
|
|
</p>
|
|
<span class="badge">{{ hysteria2_export_users | length }} {{ 'пользователь' if hysteria2_export_users | length == 1 else 'пользователей' }}</span>
|
|
<span class="badge badge-salamander">Salamander</span>
|
|
|
|
<div class="field server-obfs">
|
|
<label for="obfs-server">Пароль обфускации (Salamander)</label>
|
|
<div class="input-row">
|
|
<input type="text" id="obfs-server" value="{{ hysteria2_obfs_password | e }}" readonly>
|
|
<button type="button" class="btn-copy" onclick="copyField('obfs-server', this)" title="Копировать obfs-password" aria-label="Копировать obfs-password">
|
|
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><rect x="9" y="9" width="13" height="13" rx="2"/><path d="M5 15H4a2 2 0 0 1-2-2V4a2 2 0 0 1 2-2h9a2 2 0 0 1 2 2v1"/></svg>
|
|
</button>
|
|
</div>
|
|
</div>
|
|
</header>
|
|
|
|
{% for user in hysteria2_export_users %}
|
|
<article class="user-card">
|
|
<h2>{{ user.name }}</h2>
|
|
|
|
<div class="field">
|
|
<label for="pwd-{{ loop.index }}">Пароль</label>
|
|
<div class="input-row">
|
|
<input type="text" id="pwd-{{ loop.index }}" value="{{ user.password | e }}" readonly>
|
|
<button type="button" class="btn-copy" onclick="copyField('pwd-{{ loop.index }}', this)" title="Копировать пароль" aria-label="Копировать пароль">
|
|
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><rect x="9" y="9" width="13" height="13" rx="2"/><path d="M5 15H4a2 2 0 0 1-2-2V4a2 2 0 0 1 2-2h9a2 2 0 0 1 2 2v1"/></svg>
|
|
</button>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="field">
|
|
<label for="url-{{ loop.index }}">Ссылка подключения</label>
|
|
<div class="input-row">
|
|
<input type="text" id="url-{{ loop.index }}" value="{{ user.url | e }}" readonly>
|
|
<button type="button" class="btn-copy" onclick="copyField('url-{{ loop.index }}', this)" title="Копировать ссылку" aria-label="Копировать ссылку">
|
|
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><rect x="9" y="9" width="13" height="13" rx="2"/><path d="M5 15H4a2 2 0 0 1-2-2V4a2 2 0 0 1 2-2h9a2 2 0 0 1 2 2v1"/></svg>
|
|
</button>
|
|
</div>
|
|
</div>
|
|
|
|
{% if user.has_png %}
|
|
<div class="qr-block">
|
|
<img src="{{ user.name | e }}.png" alt="QR-код {{ user.name | e }}" width="220" height="220">
|
|
</div>
|
|
{% endif %}
|
|
|
|
<nav class="files" aria-label="Файлы">
|
|
<a href="{{ user.name | e }}.url" download>{{ user.name | e }}.url</a>
|
|
<a href="{{ user.name | e }}.txt" download>{{ user.name | e }}.txt</a>
|
|
{% if user.has_png %}
|
|
<a href="{{ user.name | e }}.png" download>{{ user.name | e }}.png</a>
|
|
{% endif %}
|
|
<a href="{{ user.name | e }}.qr.txt" download>{{ user.name | e }}.qr.txt</a>
|
|
</nav>
|
|
</article>
|
|
{% endfor %}
|
|
|
|
<footer>
|
|
Сгенерировано Ansible · {{ generated_at | default('') }}
|
|
</footer>
|
|
</div>
|
|
|
|
<div class="toast" id="toast">Скопировано!</div>
|
|
|
|
<script>
|
|
function copyField(id, btn) {
|
|
var el = document.getElementById(id);
|
|
var text = el.value;
|
|
function onSuccess() {
|
|
btn.classList.add('copied');
|
|
showToast();
|
|
setTimeout(function() { btn.classList.remove('copied'); }, 2000);
|
|
}
|
|
if (navigator.clipboard && navigator.clipboard.writeText) {
|
|
navigator.clipboard.writeText(text).then(onSuccess).catch(function() { fallbackCopy(el, onSuccess); });
|
|
} else {
|
|
fallbackCopy(el, onSuccess);
|
|
}
|
|
}
|
|
function fallbackCopy(el, cb) {
|
|
el.select();
|
|
el.setSelectionRange(0, 99999);
|
|
}
|
|
function showToast() {
|
|
var t = document.getElementById('toast');
|
|
t.classList.add('show');
|
|
setTimeout(function() { t.classList.remove('show'); }, 1800);
|
|
}
|
|
</script>
|
|
</body>
|
|
</html>
|