Files
hysteria2/roles/hysteria2/templates/export/global-index.html.j2
T
Sergey Antropoff 0aec9e6e54 Add Salamander obfs branch: replace masquerade with packet obfuscation.
- 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
2026-07-01 02:17:22 +03:00

413 lines
13 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 — все серверы</title>
<style>
:root {
color-scheme: light dark;
--bg: #0b0f14;
--bg-soft: #121820;
--bg-card: #161d27;
--bg-input: #0d1117;
--border: #2a3544;
--text: #e6edf3;
--muted: #8b949e;
--accent: #58a6ff;
--accent-2: #a371f7;
--accent-soft: rgba(88, 166, 255, 0.12);
--accent-soft-2: rgba(163, 113, 247, 0.12);
--success: #3fb950;
--success-soft: rgba(63, 185, 80, 0.15);
--radius: 14px;
--shadow: 0 12px 40px rgba(0, 0, 0, 0.35);
}
@media (prefers-color-scheme: light) {
:root {
--bg: #eef2f7;
--bg-soft: #f6f8fb;
--bg-card: #ffffff;
--bg-input: #f6f8fa;
--border: #d8dee4;
--text: #1f2328;
--muted: #656d76;
--accent: #0969da;
--accent-2: #8250df;
--accent-soft: rgba(9, 105, 218, 0.08);
--accent-soft-2: rgba(130, 80, 223, 0.08);
--success: #1a7f37;
--success-soft: rgba(26, 127, 55, 0.1);
--shadow: 0 10px 30px 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;
}
.hero {
background:
radial-gradient(ellipse 80% 60% at 50% -10%, var(--accent-soft), transparent),
radial-gradient(ellipse 60% 40% at 90% 10%, var(--accent-soft-2), transparent),
var(--bg-soft);
border-bottom: 1px solid var(--border);
padding: 2.5rem 1.5rem 2rem;
text-align: center;
}
.hero h1 {
font-size: clamp(1.75rem, 4vw, 2.25rem);
font-weight: 800;
letter-spacing: -0.03em;
margin-bottom: 0.5rem;
background: linear-gradient(135deg, var(--accent), var(--accent-2));
-webkit-background-clip: text;
-webkit-text-fill-color: transparent;
background-clip: text;
}
.hero p { color: var(--muted); font-size: 0.95rem; }
.stats {
display: flex;
justify-content: center;
gap: 1rem;
flex-wrap: wrap;
margin-top: 1.25rem;
}
.stat {
padding: 0.5rem 1rem;
border: 1px solid var(--border);
border-radius: 999px;
background: var(--bg-card);
font-size: 0.85rem;
font-weight: 600;
}
.stat span { color: var(--accent); }
.nav {
position: sticky;
top: 0;
z-index: 50;
display: flex;
gap: 0.5rem;
flex-wrap: wrap;
justify-content: center;
padding: 0.85rem 1rem;
background: color-mix(in srgb, var(--bg-soft) 88%, transparent);
backdrop-filter: blur(10px);
border-bottom: 1px solid var(--border);
}
.nav a {
padding: 0.4rem 0.85rem;
border-radius: 999px;
border: 1px solid var(--border);
background: var(--bg-card);
color: var(--text);
text-decoration: none;
font-size: 0.82rem;
font-weight: 600;
transition: border-color 0.15s, background 0.15s, color 0.15s;
}
.nav a:hover {
border-color: var(--accent);
color: var(--accent);
background: var(--accent-soft);
}
.container {
max-width: 980px;
margin: 0 auto;
padding: 1.5rem 1rem 3rem;
}
.server-section {
scroll-margin-top: 4.5rem;
margin-bottom: 2.5rem;
}
.server-header {
display: flex;
flex-wrap: wrap;
align-items: center;
justify-content: space-between;
gap: 0.75rem;
margin-bottom: 1.25rem;
padding: 1.25rem 1.5rem;
background: var(--bg-card);
border: 1px solid var(--border);
border-radius: var(--radius);
box-shadow: var(--shadow);
border-left: 4px solid var(--accent);
}
.server-header h2 {
font-size: 1.35rem;
font-weight: 700;
}
.server-header .domain {
color: var(--muted);
font-size: 0.9rem;
margin-top: 0.15rem;
}
.server-link {
display: inline-flex;
align-items: center;
gap: 0.35rem;
padding: 0.45rem 0.9rem;
border-radius: 8px;
border: 1px solid var(--border);
background: var(--bg-input);
color: var(--accent);
text-decoration: none;
font-size: 0.82rem;
font-weight: 600;
}
.server-link:hover { background: var(--accent-soft); border-color: var(--accent); }
.badge-salamander {
display: inline-block;
margin-top: 0.35rem;
padding: 0.2rem 0.6rem;
border-radius: 999px;
font-size: 0.72rem;
font-weight: 700;
background: var(--accent-soft-2);
color: var(--accent-2);
}
.server-obfs { margin-top: 0.75rem; width: 100%; }
.users-grid {
display: grid;
grid-template-columns: repeat(auto-fill, minmax(300px, 1fr));
gap: 1rem;
}
.user-card {
background: var(--bg-card);
border: 1px solid var(--border);
border-radius: var(--radius);
padding: 1.25rem;
box-shadow: var(--shadow);
display: flex;
flex-direction: column;
gap: 0.85rem;
}
.user-card h3 {
font-size: 1.05rem;
display: flex;
align-items: center;
gap: 0.5rem;
}
.user-card h3::before {
content: "";
width: 7px;
height: 7px;
border-radius: 50%;
background: var(--success);
}
.field label {
display: block;
font-size: 0.7rem;
font-weight: 700;
text-transform: uppercase;
letter-spacing: 0.06em;
color: var(--muted);
margin-bottom: 0.35rem;
}
.input-row { display: flex; gap: 0.45rem; }
.input-row input {
flex: 1;
min-width: 0;
padding: 0.55rem 0.75rem;
background: var(--bg-input);
border: 1px solid var(--border);
border-radius: 8px;
color: var(--text);
font-family: ui-monospace, SFMono-Regular, Menlo, Consolas, monospace;
font-size: 0.72rem;
}
.btn-copy {
display: flex;
align-items: center;
justify-content: center;
width: 38px;
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: 16px; height: 16px; }
.qr-block {
text-align: center;
padding: 0.75rem;
background: #fff;
border-radius: 8px;
border: 1px solid var(--border);
}
.qr-block img {
width: 160px;
height: 160px;
image-rendering: pixelated;
}
.files {
display: flex;
flex-wrap: wrap;
gap: 0.4rem;
margin-top: auto;
padding-top: 0.75rem;
border-top: 1px solid var(--border);
}
.files a {
padding: 0.3rem 0.6rem;
border-radius: 6px;
border: 1px solid var(--border);
background: var(--bg-input);
color: var(--accent);
text-decoration: none;
font-size: 0.72rem;
font-weight: 600;
}
.files a:hover { background: var(--accent-soft); border-color: var(--accent); }
footer {
text-align: center;
padding: 1.5rem;
color: var(--muted);
font-size: 0.8rem;
border-top: 1px solid var(--border);
}
.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>
<header class="hero">
<h1>Hysteria2 · Salamander</h1>
<p>Общий каталог VPN-подключений с обфускацией Salamander</p>
<div class="stats">
<div class="stat"><span>{{ hysteria2_global_servers | length }}</span> серверов</div>
<div class="stat"><span>{{ total_users }}</span> пользователей</div>
</div>
</header>
<nav class="nav" aria-label="Серверы">
{% for server in hysteria2_global_servers %}
<a href="#server-{{ server.dir | e }}">{{ server.name | e }}</a>
{% endfor %}
</nav>
<main class="container">
{% for server in hysteria2_global_servers %}
<section class="server-section" id="server-{{ server.dir | e }}">
<div class="server-header">
<div>
<h2>{{ server.name | e }}</h2>
<p class="domain">{{ server.domain | e }}:{{ server.port }}</p>
<span class="badge-salamander">Salamander</span>
{% if server.obfs_password | default('') | length > 0 %}
<div class="field server-obfs">
<label for="obfs-{{ server.dir | e }}">Пароль обфускации</label>
<div class="input-row">
<input type="text" id="obfs-{{ server.dir | e }}" value="{{ server.obfs_password | e }}" readonly>
<button type="button" class="btn-copy" onclick="copyField('obfs-{{ server.dir | e }}', this)" title="Копировать obfs-password" aria-label="Копировать obfs-password">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><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>
{% endif %}
</div>
<a class="server-link" href="{{ server.dir | e }}/index.html">Страница сервера →</a>
</div>
<div class="users-grid">
{% for user in server.users %}
<article class="user-card">
<h3>{{ user.name | e }}</h3>
<div class="field">
<label for="pwd-{{ server.dir | e }}-{{ loop.index }}">Пароль</label>
<div class="input-row">
<input type="text" id="pwd-{{ server.dir | e }}-{{ loop.index }}" value="{{ user.password | e }}" readonly>
<button type="button" class="btn-copy" onclick="copyField('pwd-{{ server.dir | e }}-{{ loop.index }}', this)" title="Копировать пароль" aria-label="Копировать пароль">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><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-{{ server.dir | e }}-{{ loop.index }}">Ссылка</label>
<div class="input-row">
<input type="text" id="url-{{ server.dir | e }}-{{ loop.index }}" value="{{ user.url | e }}" readonly>
<button type="button" class="btn-copy" onclick="copyField('url-{{ server.dir | e }}-{{ loop.index }}', this)" title="Копировать ссылку" aria-label="Копировать ссылку">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><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 | default(false) %}
<div class="qr-block">
<img src="{{ server.dir | e }}/{{ user.name | e }}.png" alt="QR {{ user.name | e }}" width="160" height="160">
</div>
{% endif %}
<nav class="files" aria-label="Файлы">
<a href="{{ server.dir | e }}/{{ user.name | e }}.url" download>.url</a>
<a href="{{ server.dir | e }}/{{ user.name | e }}.txt" download>.txt</a>
{% if user.has_png | default(false) %}
<a href="{{ server.dir | e }}/{{ user.name | e }}.png" download>.png</a>
{% endif %}
<a href="{{ server.dir | e }}/{{ user.name | e }}.qr.txt" download>.qr.txt</a>
</nav>
</article>
{% endfor %}
</div>
</section>
{% endfor %}
</main>
<footer>Сгенерировано Ansible · {{ generated_at | default('') }}</footer>
<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>