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
413 lines
13 KiB
Django/Jinja
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>
|