+418
@@ -0,0 +1,418 @@
|
||||
<!doctype html>
|
||||
<html lang="en">
|
||||
<head>
|
||||
<meta charset="utf-8">
|
||||
<meta name="viewport" content="width=device-width,initial-scale=1">
|
||||
<title>weircon-random-proxy · API tester</title>
|
||||
<style>
|
||||
:root {
|
||||
--bg: #f7f7f8;
|
||||
--panel: #ffffff;
|
||||
--text: #1f2328;
|
||||
--muted: #656d76;
|
||||
--border: #d0d7de;
|
||||
--accent: #0969da;
|
||||
--ok: #1a7f37;
|
||||
--warn: #9a6700;
|
||||
--err: #cf222e;
|
||||
--code-bg: #f1f3f5;
|
||||
--pill-bg: #ddf4ff;
|
||||
--mono: ui-monospace, "SF Mono", Menlo, Consolas, "Liberation Mono", monospace;
|
||||
}
|
||||
@media (prefers-color-scheme: dark) {
|
||||
:root {
|
||||
--bg: #0d1117;
|
||||
--panel: #161b22;
|
||||
--text: #e6edf3;
|
||||
--muted: #8b949e;
|
||||
--border: #30363d;
|
||||
--accent: #58a6ff;
|
||||
--ok: #3fb950;
|
||||
--warn: #d29922;
|
||||
--err: #f85149;
|
||||
--code-bg: #1e242b;
|
||||
--pill-bg: #133156;
|
||||
}
|
||||
}
|
||||
* { box-sizing: border-box; }
|
||||
html, body { margin: 0; padding: 0; }
|
||||
body {
|
||||
font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Helvetica, Arial, sans-serif;
|
||||
background: var(--bg);
|
||||
color: var(--text);
|
||||
line-height: 1.5;
|
||||
}
|
||||
.wrap { max-width: 920px; margin: 0 auto; padding: 24px; }
|
||||
header { margin-bottom: 32px; }
|
||||
h1 { margin: 0 0 4px; font-size: 24px; letter-spacing: -0.01em; }
|
||||
h1 .tag { font-family: var(--mono); font-size: 13px; color: var(--muted); margin-left: 8px; font-weight: 400; }
|
||||
header p { margin: 0; color: var(--muted); }
|
||||
section { background: var(--panel); border: 1px solid var(--border); border-radius: 8px; padding: 20px; margin-bottom: 16px; }
|
||||
section h2 { margin: 0 0 4px; font-size: 18px; }
|
||||
section h2 .method { font-family: var(--mono); font-size: 12px; padding: 2px 8px; border-radius: 4px; background: var(--pill-bg); color: var(--accent); margin-right: 8px; vertical-align: 1px; }
|
||||
section h2 code { font-size: 16px; color: var(--text); background: transparent; padding: 0; }
|
||||
section > p { margin: 0 0 12px; color: var(--muted); font-size: 14px; }
|
||||
label { display: block; font-size: 13px; color: var(--muted); margin-bottom: 4px; }
|
||||
input, select, textarea {
|
||||
width: 100%;
|
||||
padding: 8px 10px;
|
||||
border: 1px solid var(--border);
|
||||
border-radius: 6px;
|
||||
background: var(--bg);
|
||||
color: var(--text);
|
||||
font-family: inherit;
|
||||
font-size: 14px;
|
||||
}
|
||||
input:focus, select:focus, textarea:focus { outline: 2px solid var(--accent); outline-offset: -1px; border-color: transparent; }
|
||||
textarea { font-family: var(--mono); resize: vertical; min-height: 80px; }
|
||||
.grid { display: grid; grid-template-columns: 2fr 1fr 1fr; gap: 12px; margin-bottom: 12px; }
|
||||
@media (max-width: 600px) { .grid { grid-template-columns: 1fr; } }
|
||||
.field { margin-bottom: 12px; }
|
||||
button {
|
||||
padding: 8px 16px;
|
||||
background: var(--accent);
|
||||
color: white;
|
||||
border: none;
|
||||
border-radius: 6px;
|
||||
font-size: 14px;
|
||||
font-weight: 500;
|
||||
cursor: pointer;
|
||||
}
|
||||
button:hover { filter: brightness(0.92); }
|
||||
button.secondary { background: var(--panel); color: var(--text); border: 1px solid var(--border); }
|
||||
button.secondary:hover { background: var(--code-bg); }
|
||||
button:disabled { opacity: 0.5; cursor: not-allowed; }
|
||||
pre, code { font-family: var(--mono); }
|
||||
pre {
|
||||
background: var(--code-bg);
|
||||
border: 1px solid var(--border);
|
||||
border-radius: 6px;
|
||||
padding: 12px;
|
||||
overflow: auto;
|
||||
font-size: 12px;
|
||||
max-height: 360px;
|
||||
margin: 0;
|
||||
}
|
||||
code { background: var(--code-bg); padding: 1px 4px; border-radius: 3px; font-size: 0.92em; }
|
||||
table { width: 100%; border-collapse: collapse; font-size: 13px; margin-bottom: 12px; }
|
||||
th, td { text-align: left; padding: 6px 8px; border-bottom: 1px solid var(--border); vertical-align: top; }
|
||||
th { font-weight: 600; color: var(--muted); font-size: 12px; text-transform: uppercase; letter-spacing: 0.04em; }
|
||||
td code { font-size: 12px; }
|
||||
.status-pill { display: inline-block; padding: 2px 8px; border-radius: 10px; font-size: 12px; font-family: var(--mono); font-weight: 600; }
|
||||
.status-2xx { background: rgba(63, 185, 80, 0.15); color: var(--ok); }
|
||||
.status-3xx { background: rgba(210, 153, 34, 0.15); color: var(--warn); }
|
||||
.status-4xx, .status-5xx { background: rgba(248, 81, 73, 0.15); color: var(--err); }
|
||||
.row { display: flex; gap: 8px; align-items: center; flex-wrap: wrap; }
|
||||
.row > * { flex: 0 0 auto; }
|
||||
details summary { cursor: pointer; color: var(--muted); font-size: 13px; padding: 4px 0; }
|
||||
details summary:hover { color: var(--text); }
|
||||
.muted { color: var(--muted); font-size: 13px; }
|
||||
.tunnel-grid { display: grid; grid-template-columns: repeat(auto-fill, minmax(160px, 1fr)); gap: 8px; margin-top: 12px; }
|
||||
.tunnel { padding: 10px 12px; background: var(--code-bg); border: 1px solid var(--border); border-radius: 6px; font-family: var(--mono); font-size: 12px; }
|
||||
.tunnel .id { color: var(--muted); font-size: 11px; }
|
||||
.tunnel .ip { font-size: 13px; font-weight: 600; margin-top: 2px; }
|
||||
.tunnel.err { border-color: var(--err); }
|
||||
.tunnel.err .ip { color: var(--err); }
|
||||
.tunnel.loading .ip { color: var(--muted); }
|
||||
</style>
|
||||
</head>
|
||||
<body>
|
||||
<div class="wrap">
|
||||
|
||||
<header>
|
||||
<h1>weircon-random-proxy <span class="tag" id="version-tag">api tester</span></h1>
|
||||
<p>HTTP fetch service that forwards a request through one of N upstream WireGuard tunnels. This page lets you exercise the API without writing any code.</p>
|
||||
</header>
|
||||
|
||||
<section id="config-section">
|
||||
<h2>Configuration</h2>
|
||||
<p>If a reverse proxy in front (e.g. NginxProxyManager) validates an API key header, paste it here. Stored in your browser's localStorage. Leave blank when accessing the service directly on LAN.</p>
|
||||
<div class="grid">
|
||||
<div>
|
||||
<label for="apikey">X-WEIRCON-RANDOM-IP (API key)</label>
|
||||
<input id="apikey" type="password" autocomplete="off" placeholder="optional">
|
||||
</div>
|
||||
<div>
|
||||
<label for="base-url">Service base URL</label>
|
||||
<input id="base-url" type="url" placeholder="(this origin)">
|
||||
</div>
|
||||
<div style="display: flex; align-items: end;">
|
||||
<button class="secondary" onclick="forgetCreds()">Forget</button>
|
||||
</div>
|
||||
</div>
|
||||
</section>
|
||||
|
||||
<section>
|
||||
<h2><span class="method">GET</span><code>/health</code></h2>
|
||||
<p>Service health and the number of upstream tunnels configured.</p>
|
||||
<div class="row">
|
||||
<button onclick="callHealth()">Try it</button>
|
||||
<span id="health-status" class="muted"></span>
|
||||
</div>
|
||||
<pre id="health-body" style="margin-top: 8px; display: none;"></pre>
|
||||
</section>
|
||||
|
||||
<section>
|
||||
<h2><span class="method">ANY</span><code>/</code></h2>
|
||||
<p>Fetch a target URL through one of the configured tunnels. Status, headers, and body of the upstream response are streamed back to you.</p>
|
||||
|
||||
<h3 style="margin: 16px 0 8px; font-size: 14px;">Parameters</h3>
|
||||
<table>
|
||||
<thead><tr><th>Source</th><th>Name</th><th>Required</th><th>Description</th></tr></thead>
|
||||
<tbody>
|
||||
<tr><td>header</td><td><code>X-WEIRCON-RANDOM-IP-REDIRECT</code></td><td>one of</td><td>Target URL to fetch. Takes precedence over <code>?url=</code>.</td></tr>
|
||||
<tr><td>query</td><td><code>?url=</code></td><td>one of</td><td>Alternative way to pass the target URL.</td></tr>
|
||||
<tr><td>header</td><td><code>X-WEIRCON-PROXY-ID</code></td><td>no</td><td>Pin a specific tunnel (0..N-1). Default: random.</td></tr>
|
||||
<tr><td>header</td><td><code>X-WEIRCON-FORWARD-METHOD</code></td><td>no</td><td>Override the HTTP method sent to the target.</td></tr>
|
||||
<tr><td>body</td><td><em>passthrough</em></td><td>no</td><td>Forwarded as-is for POST/PUT/PATCH.</td></tr>
|
||||
</tbody>
|
||||
</table>
|
||||
|
||||
<h3 style="margin: 16px 0 8px; font-size: 14px;">Try it out</h3>
|
||||
<div class="grid">
|
||||
<div>
|
||||
<label for="target">Target URL</label>
|
||||
<input id="target" type="url" value="https://api.ipify.org" placeholder="https://example.com">
|
||||
</div>
|
||||
<div>
|
||||
<label for="method">Method</label>
|
||||
<select id="method">
|
||||
<option>GET</option><option>POST</option><option>PUT</option><option>PATCH</option><option>DELETE</option><option>HEAD</option><option>OPTIONS</option>
|
||||
</select>
|
||||
</div>
|
||||
<div>
|
||||
<label for="proxy-id">Proxy ID</label>
|
||||
<input id="proxy-id" type="number" min="0" placeholder="random">
|
||||
</div>
|
||||
</div>
|
||||
<div class="field" id="body-field" style="display: none;">
|
||||
<label for="body">Request body</label>
|
||||
<textarea id="body" placeholder='{"hello":"world"}'></textarea>
|
||||
</div>
|
||||
<div class="row">
|
||||
<button onclick="callFetch()">Send</button>
|
||||
<button class="secondary" onclick="showCurl()">Show cURL</button>
|
||||
<span id="fetch-status" class="muted"></span>
|
||||
</div>
|
||||
|
||||
<details id="curl-details" style="margin-top: 12px;">
|
||||
<summary>cURL equivalent</summary>
|
||||
<pre id="curl-out"></pre>
|
||||
</details>
|
||||
|
||||
<div id="response-block" style="display: none; margin-top: 16px;">
|
||||
<div class="row" style="margin-bottom: 8px;">
|
||||
<span id="resp-status" class="status-pill"></span>
|
||||
<span id="resp-egress" class="muted"></span>
|
||||
<span id="resp-latency" class="muted"></span>
|
||||
</div>
|
||||
<details open>
|
||||
<summary>Body</summary>
|
||||
<pre id="resp-body"></pre>
|
||||
</details>
|
||||
<details>
|
||||
<summary>Response headers</summary>
|
||||
<pre id="resp-headers"></pre>
|
||||
</details>
|
||||
</div>
|
||||
</section>
|
||||
|
||||
<section>
|
||||
<h2>Tunnel health check</h2>
|
||||
<p>Sends a request to each configured tunnel and reports the egress IP. A healthy stack returns N distinct IPs.</p>
|
||||
<div class="row">
|
||||
<input id="probe-url" type="url" value="https://api.ipify.org" style="max-width: 320px;">
|
||||
<button onclick="probeAll()">Probe all tunnels</button>
|
||||
</div>
|
||||
<div id="tunnel-grid" class="tunnel-grid"></div>
|
||||
</section>
|
||||
|
||||
<p class="muted" style="text-align: center; margin-top: 24px;">
|
||||
weircon-random-proxy · <a href="/health" style="color: var(--accent);">/health</a>
|
||||
</p>
|
||||
|
||||
</div>
|
||||
|
||||
<script>
|
||||
const HEADER_AUTH = "X-Weircon-Random-Ip";
|
||||
const HEADER_REDIRECT = "X-Weircon-Random-Ip-Redirect";
|
||||
const HEADER_PROXY_ID = "X-Weircon-Proxy-Id";
|
||||
const HEADER_METHOD = "X-Weircon-Forward-Method";
|
||||
const RESP_HEADER_EGRESS = "x-weircon-egress-proxy";
|
||||
|
||||
const $ = (id) => document.getElementById(id);
|
||||
|
||||
function loadCreds() {
|
||||
$("apikey").value = localStorage.getItem("weircon_apikey") || "";
|
||||
$("base-url").value = localStorage.getItem("weircon_base_url") || "";
|
||||
}
|
||||
function saveCreds() {
|
||||
localStorage.setItem("weircon_apikey", $("apikey").value);
|
||||
localStorage.setItem("weircon_base_url", $("base-url").value);
|
||||
}
|
||||
function forgetCreds() {
|
||||
localStorage.removeItem("weircon_apikey");
|
||||
localStorage.removeItem("weircon_base_url");
|
||||
loadCreds();
|
||||
}
|
||||
function baseURL() {
|
||||
return ($("base-url").value || window.location.origin).replace(/\/$/, "");
|
||||
}
|
||||
function authHeaders() {
|
||||
const h = {};
|
||||
const k = $("apikey").value;
|
||||
if (k) h[HEADER_AUTH] = k;
|
||||
return h;
|
||||
}
|
||||
|
||||
async function callHealth() {
|
||||
saveCreds();
|
||||
$("health-status").textContent = "calling…";
|
||||
$("health-body").style.display = "none";
|
||||
try {
|
||||
const r = await fetch(baseURL() + "/health", { headers: authHeaders() });
|
||||
const text = await r.text();
|
||||
$("health-status").textContent = r.ok ? `HTTP ${r.status}` : `HTTP ${r.status} (failed)`;
|
||||
$("health-body").textContent = tryPrettyJSON(text);
|
||||
$("health-body").style.display = "block";
|
||||
try {
|
||||
const j = JSON.parse(text);
|
||||
if (j.proxies) renderTunnelGrid(j.proxies);
|
||||
} catch (_) {}
|
||||
} catch (e) {
|
||||
$("health-status").textContent = "network error: " + e.message;
|
||||
}
|
||||
}
|
||||
|
||||
async function callFetch() {
|
||||
saveCreds();
|
||||
const target = $("target").value.trim();
|
||||
if (!target) { alert("target URL required"); return; }
|
||||
const method = $("method").value;
|
||||
const proxyId = $("proxy-id").value.trim();
|
||||
const body = $("body").value;
|
||||
|
||||
const headers = { ...authHeaders(), [HEADER_REDIRECT]: target };
|
||||
if (proxyId !== "") headers[HEADER_PROXY_ID] = proxyId;
|
||||
|
||||
const init = { method, headers };
|
||||
if (["POST", "PUT", "PATCH"].includes(method) && body) init.body = body;
|
||||
|
||||
$("fetch-status").textContent = "calling…";
|
||||
$("response-block").style.display = "none";
|
||||
|
||||
const t0 = performance.now();
|
||||
try {
|
||||
const r = await fetch(baseURL() + "/", init);
|
||||
const t1 = performance.now();
|
||||
const text = await r.text();
|
||||
showResponse(r, text, Math.round(t1 - t0));
|
||||
$("fetch-status").textContent = "";
|
||||
} catch (e) {
|
||||
$("fetch-status").textContent = "network error: " + e.message;
|
||||
}
|
||||
}
|
||||
|
||||
function showResponse(r, body, latencyMs) {
|
||||
$("response-block").style.display = "block";
|
||||
const sp = $("resp-status");
|
||||
sp.textContent = `${r.status} ${r.statusText}`;
|
||||
sp.className = "status-pill status-" + (Math.floor(r.status / 100)) + "xx";
|
||||
|
||||
const egress = r.headers.get(RESP_HEADER_EGRESS);
|
||||
$("resp-egress").textContent = egress !== null ? `via proxy${egress}` : "";
|
||||
$("resp-latency").textContent = `${latencyMs} ms`;
|
||||
|
||||
$("resp-body").textContent = tryPrettyJSON(body);
|
||||
const headerLines = [];
|
||||
r.headers.forEach((v, k) => headerLines.push(`${k}: ${v}`));
|
||||
$("resp-headers").textContent = headerLines.join("\n");
|
||||
}
|
||||
|
||||
function tryPrettyJSON(s) {
|
||||
try { return JSON.stringify(JSON.parse(s), null, 2); } catch (_) { return s; }
|
||||
}
|
||||
|
||||
function showCurl() {
|
||||
saveCreds();
|
||||
const target = $("target").value.trim();
|
||||
const method = $("method").value;
|
||||
const proxyId = $("proxy-id").value.trim();
|
||||
const body = $("body").value;
|
||||
const k = $("apikey").value;
|
||||
|
||||
let cmd = `curl -X ${method} \\\n "${baseURL()}/"`;
|
||||
if (k) cmd += ` \\\n -H '${HEADER_AUTH}: ${k}'`;
|
||||
cmd += ` \\\n -H '${HEADER_REDIRECT}: ${target}'`;
|
||||
if (proxyId !== "") cmd += ` \\\n -H '${HEADER_PROXY_ID}: ${proxyId}'`;
|
||||
if (["POST", "PUT", "PATCH"].includes(method) && body) {
|
||||
cmd += ` \\\n --data-raw '${body.replace(/'/g, "'\\''")}'`;
|
||||
}
|
||||
$("curl-out").textContent = cmd;
|
||||
$("curl-details").open = true;
|
||||
}
|
||||
|
||||
async function renderTunnelGrid(count) {
|
||||
const grid = $("tunnel-grid");
|
||||
grid.innerHTML = "";
|
||||
for (let i = 0; i < count; i++) {
|
||||
const el = document.createElement("div");
|
||||
el.className = "tunnel";
|
||||
el.id = `tunnel-${i}`;
|
||||
el.innerHTML = `<div class="id">proxy${i}</div><div class="ip">—</div>`;
|
||||
grid.appendChild(el);
|
||||
}
|
||||
}
|
||||
|
||||
async function probeAll() {
|
||||
const probeURL = $("probe-url").value.trim();
|
||||
if (!probeURL) { alert("probe URL required"); return; }
|
||||
|
||||
// ensure grid is rendered (calls /health if needed)
|
||||
if ($("tunnel-grid").children.length === 0) {
|
||||
try {
|
||||
const r = await fetch(baseURL() + "/health", { headers: authHeaders() });
|
||||
const j = await r.json();
|
||||
renderTunnelGrid(j.proxies);
|
||||
} catch (e) {
|
||||
alert("failed to read /health: " + e.message);
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
const cells = [...$("tunnel-grid").children];
|
||||
cells.forEach(c => { c.className = "tunnel loading"; c.querySelector(".ip").textContent = "…"; });
|
||||
|
||||
await Promise.all(cells.map(async (cell, i) => {
|
||||
const headers = { ...authHeaders(), [HEADER_REDIRECT]: probeURL, [HEADER_PROXY_ID]: String(i) };
|
||||
const t0 = performance.now();
|
||||
try {
|
||||
const r = await fetch(baseURL() + "/", { headers });
|
||||
const text = (await r.text()).trim();
|
||||
const ms = Math.round(performance.now() - t0);
|
||||
if (r.ok) {
|
||||
cell.className = "tunnel";
|
||||
cell.querySelector(".ip").textContent = text.slice(0, 64);
|
||||
cell.querySelector(".id").textContent = `proxy${i} · ${ms}ms`;
|
||||
} else {
|
||||
cell.className = "tunnel err";
|
||||
cell.querySelector(".ip").textContent = `HTTP ${r.status}`;
|
||||
}
|
||||
} catch (e) {
|
||||
cell.className = "tunnel err";
|
||||
cell.querySelector(".ip").textContent = "error";
|
||||
}
|
||||
}));
|
||||
}
|
||||
|
||||
$("method").addEventListener("change", () => {
|
||||
const m = $("method").value;
|
||||
$("body-field").style.display = ["POST", "PUT", "PATCH"].includes(m) ? "block" : "none";
|
||||
});
|
||||
|
||||
loadCreds();
|
||||
// auto-load /health to populate tunnel grid on first paint
|
||||
callHealth();
|
||||
</script>
|
||||
</body>
|
||||
</html>
|
||||
Reference in New Issue
Block a user