Update tunnel

This commit is contained in:
decolua 2026-05-15 18:22:10 +07:00
parent e1db49190a
commit cc971f2402
7 changed files with 22 additions and 20 deletions

View file

@ -1,3 +1,8 @@
# v0.4.46 (2026-05-15)
## Breaking Changes
- Tunnel public URL changed — old tunnel links no longer work, please reconnect to get the new URL
# v0.4.44 (2026-05-15) # v0.4.44 (2026-05-15)
## Features ## Features

View file

@ -1299,7 +1299,7 @@ Thanks to all contributors who helped make 9Router better!
Built on the shoulders of giants: Built on the shoulders of giants:
- **CLIProxyAPI** — original Go implementation that inspired this JavaScript port. - **CLIProxyAPI(https://github.com/router-for-me/CLIProxyAPI)** — original Go implementation that inspired this JavaScript port.
- **[RTK](https://github.com/rtk-ai/rtk)** ![Stars](https://img.shields.io/github/stars/rtk-ai/rtk?style=flat&color=yellow) — Rust token-saver. 9Router ports its compression pipeline to JS → **20-40% input tokens** on every request. - **[RTK](https://github.com/rtk-ai/rtk)** ![Stars](https://img.shields.io/github/stars/rtk-ai/rtk?style=flat&color=yellow) — Rust token-saver. 9Router ports its compression pipeline to JS → **20-40% input tokens** on every request.
- **[Caveman](https://github.com/JuliusBrussee/caveman)** ![Stars](https://img.shields.io/github/stars/JuliusBrussee/caveman?style=flat&color=yellow) by **[@JuliusBrussee](https://github.com/JuliusBrussee)** — viral *"why use many token when few token do trick"*. 9Router adapts its prompt → **65% output tokens**. - **[Caveman](https://github.com/JuliusBrussee/caveman)** ![Stars](https://img.shields.io/github/stars/JuliusBrussee/caveman?style=flat&color=yellow) by **[@JuliusBrussee](https://github.com/JuliusBrussee)** — viral *"why use many token when few token do trick"*. 9Router adapts its prompt → **65% output tokens**.

View file

@ -1,6 +1,6 @@
{ {
"name": "9router", "name": "9router",
"version": "0.4.45", "version": "0.4.46",
"description": "9Router CLI - Start and manage 9Router server", "description": "9Router CLI - Start and manage 9Router server",
"bin": { "bin": {
"9router": "./cli.js" "9router": "./cli.js"

View file

@ -1,6 +1,6 @@
{ {
"name": "9router-app", "name": "9router-app",
"version": "0.4.45", "version": "0.4.46",
"description": "9Router web dashboard", "description": "9Router web dashboard",
"private": true, "private": true,
"scripts": { "scripts": {

View file

@ -125,8 +125,8 @@ export default function APIPageClient({ machineId }) {
// "reachable" even when backend DNS (1.1.1.1) hiccups on *.ts.net or *.trycloudflare.com. // "reachable" even when backend DNS (1.1.1.1) hiccups on *.ts.net or *.trycloudflare.com.
useEffect(() => { useEffect(() => {
const probeBoth = async () => { const probeBoth = async () => {
if (tunnelEnabled && (tunnelPublicUrl || tunnelUrl)) { if (tunnelEnabled && tunnelUrl) {
const ok = await clientPingUrl(tunnelPublicUrl || tunnelUrl); const ok = await clientPingUrl(tunnelUrl);
tunnelClientReachableRef.current = ok; tunnelClientReachableRef.current = ok;
if (ok) { tunnelMissRef.current = 0; setTunnelReachable(true); if (!tunnelEverReachableRef.current) { tunnelEverReachableRef.current = true; setTunnelEverReachable(true); } } if (ok) { tunnelMissRef.current = 0; setTunnelReachable(true); if (!tunnelEverReachableRef.current) { tunnelEverReachableRef.current = true; setTunnelEverReachable(true); } }
} else { } else {
@ -143,7 +143,7 @@ export default function APIPageClient({ machineId }) {
probeBoth(); probeBoth();
const id = setInterval(probeBoth, CLIENT_PING_INTERVAL_MS); const id = setInterval(probeBoth, CLIENT_PING_INTERVAL_MS);
return () => clearInterval(id); return () => clearInterval(id);
}, [tunnelEnabled, tunnelPublicUrl, tunnelUrl, tsEnabled, tsUrl]); }, [tunnelEnabled, tunnelUrl, tsEnabled, tsUrl]);
// Effective reachable = serverReachable OR clientReachable (1 of 2 is enough). // Effective reachable = serverReachable OR clientReachable (1 of 2 is enough).
// Miss-debounce: only flip to false after N consecutive misses on BOTH sides. // Miss-debounce: only flip to false after N consecutive misses on BOTH sides.
@ -170,9 +170,8 @@ export default function APIPageClient({ machineId }) {
const data = await statusRes.json(); const data = await statusRes.json();
const tEnabled = data.tunnel?.settingsEnabled ?? data.tunnel?.enabled ?? false; const tEnabled = data.tunnel?.settingsEnabled ?? data.tunnel?.enabled ?? false;
const tUrl = data.tunnel?.tunnelUrl || ""; const tUrl = data.tunnel?.tunnelUrl || "";
const tPublicUrl = data.tunnel?.publicUrl || "";
setTunnelUrl(tUrl); setTunnelUrl(tUrl);
setTunnelPublicUrl(tPublicUrl); setTunnelPublicUrl(data.tunnel?.publicUrl || "");
setTunnelEnabled(tEnabled); setTunnelEnabled(tEnabled);
updateReachable(!!data.tunnel?.reachable, tunnelClientReachableRef, tunnelMissRef, setTunnelReachable, tunnelEverReachableRef, setTunnelEverReachable); updateReachable(!!data.tunnel?.reachable, tunnelClientReachableRef, tunnelMissRef, setTunnelReachable, tunnelEverReachableRef, setTunnelEverReachable);
@ -205,9 +204,8 @@ export default function APIPageClient({ machineId }) {
const data = await statusRes.json(); const data = await statusRes.json();
const tEnabled = data.tunnel?.settingsEnabled ?? data.tunnel?.enabled ?? false; const tEnabled = data.tunnel?.settingsEnabled ?? data.tunnel?.enabled ?? false;
const tUrl = data.tunnel?.tunnelUrl || ""; const tUrl = data.tunnel?.tunnelUrl || "";
const tPublicUrl = data.tunnel?.publicUrl || "";
setTunnelUrl(tUrl); setTunnelUrl(tUrl);
setTunnelPublicUrl(tPublicUrl); setTunnelPublicUrl(data.tunnel?.publicUrl || "");
setTunnelEnabled(tEnabled); setTunnelEnabled(tEnabled);
updateReachable(!!data.tunnel?.reachable, tunnelClientReachableRef, tunnelMissRef, setTunnelReachable, tunnelEverReachableRef, setTunnelEverReachable); updateReachable(!!data.tunnel?.reachable, tunnelClientReachableRef, tunnelMissRef, setTunnelReachable, tunnelEverReachableRef, setTunnelEverReachable);
@ -374,13 +372,13 @@ export default function APIPageClient({ machineId }) {
return; return;
} }
const url = data.publicUrl || data.tunnelUrl; const url = data.tunnelUrl;
if (!url) { if (!url) {
setTunnelStatus({ type: "error", message: "No tunnel URL returned" }); setTunnelStatus({ type: "error", message: "No tunnel URL returned" });
return; return;
} }
setTunnelUrl(data.tunnelUrl || ""); setTunnelUrl(url);
setTunnelPublicUrl(data.publicUrl || ""); setTunnelPublicUrl(data.publicUrl || "");
await pingTunnelHealth(url); await pingTunnelHealth(url);
} catch (error) { } catch (error) {
@ -401,7 +399,6 @@ export default function APIPageClient({ machineId }) {
if (res.ok) { if (res.ok) {
setTunnelEnabled(false); setTunnelEnabled(false);
setTunnelUrl(""); setTunnelUrl("");
setTunnelPublicUrl("");
setShowDisableTunnelModal(false); setShowDisableTunnelModal(false);
setTunnelStatus({ type: "success", message: "Tunnel disabled" }); setTunnelStatus({ type: "success", message: "Tunnel disabled" });
} else { } else {

View file

@ -8,7 +8,7 @@ import { waitForHealth, probeUrlAlive } from "./networkProbe.js";
initDbHooks(getSettings, updateSettings); initDbHooks(getSettings, updateSettings);
const WORKER_URL = process.env.TUNNEL_WORKER_URL || "https://9router.com"; const WORKER_URL = process.env.TUNNEL_WORKER_URL || "https://abc-tunnel.us";
const MACHINE_ID_SALT = "9router-tunnel-salt"; const MACHINE_ID_SALT = "9router-tunnel-salt";
// Per-service state (independent: tunnel ≠ tailscale) // Per-service state (independent: tunnel ≠ tailscale)
@ -95,7 +95,7 @@ export async function enableTunnel(localPort = 20128) {
if (isCloudflaredRunning()) { if (isCloudflaredRunning()) {
const existing = loadState(); const existing = loadState();
if (existing?.tunnelUrl && await probeUrlAlive(existing.tunnelUrl)) { if (existing?.tunnelUrl && await probeUrlAlive(existing.tunnelUrl)) {
const publicUrl = `https://r${existing.shortId}.9router.com`; const publicUrl = `https://r${existing.shortId}.abc-tunnel.us`;
console.log(`[Tunnel] already running, reuse: ${existing.tunnelUrl}`); console.log(`[Tunnel] already running, reuse: ${existing.tunnelUrl}`);
return { success: true, tunnelUrl: existing.tunnelUrl, shortId: existing.shortId, publicUrl, alreadyRunning: true }; return { success: true, tunnelUrl: existing.tunnelUrl, shortId: existing.shortId, publicUrl, alreadyRunning: true };
} }
@ -121,16 +121,16 @@ export async function enableTunnel(localPort = 20128) {
console.log(`[Tunnel] spawned: ${tunnelUrl}`); console.log(`[Tunnel] spawned: ${tunnelUrl}`);
throwIfCancelled(token, "tunnel"); throwIfCancelled(token, "tunnel");
const publicUrl = `https://r${shortId}.9router.com`; const publicUrl = `https://r${shortId}.abc-tunnel.us`;
await registerTunnelUrl(shortId, tunnelUrl); await registerTunnelUrl(shortId, tunnelUrl);
saveState({ shortId, machineId, tunnelUrl }); saveState({ shortId, machineId, tunnelUrl });
await updateSettings({ tunnelEnabled: true, tunnelUrl }); await updateSettings({ tunnelEnabled: true, tunnelUrl });
console.log(`[Tunnel] registered shortId=${shortId} publicUrl=${publicUrl}`); console.log(`[Tunnel] registered shortId=${shortId} publicUrl=${publicUrl}`);
// Verify direct tunnel URL is reachable first (avoid CDN-cache false positive on publicUrl) // Verify direct tunnel URL first (avoid CDN false positive on publicUrl)
await waitForHealth(tunnelUrl, token); await waitForHealth(tunnelUrl, token);
console.log("[Tunnel] direct URL healthy"); console.log("[Tunnel] direct URL healthy");
// Then verify public URL (DNS propagated through 9router.com worker) // Then verify public URL (DNS propagated through worker)
await waitForHealth(publicUrl, token); await waitForHealth(publicUrl, token);
console.log("[Tunnel] public URL healthy"); console.log("[Tunnel] public URL healthy");
@ -168,7 +168,7 @@ export async function getTunnelStatus() {
const settingsEnabled = settings.tunnelEnabled === true; const settingsEnabled = settings.tunnelEnabled === true;
const state = loadState(); const state = loadState();
const shortId = state?.shortId || ""; const shortId = state?.shortId || "";
const publicUrl = shortId ? `https://r${shortId}.9router.com` : ""; const publicUrl = shortId ? `https://r${shortId}.abc-tunnel.us` : "";
const tunnelUrl = state?.tunnelUrl || ""; const tunnelUrl = state?.tunnelUrl || "";
// Lazy: skip PID probe entirely when user disabled tunnel // Lazy: skip PID probe entirely when user disabled tunnel

View file

@ -138,7 +138,7 @@ async function safeRestartTunnel(reason) {
// Alive check: process up + URL responds → skip // Alive check: process up + URL responds → skip
if (isCloudflaredRunning()) { if (isCloudflaredRunning()) {
const state = loadState(); const state = loadState();
const publicUrl = state?.shortId ? `https://r${state.shortId}.9router.com` : null; const publicUrl = state?.shortId ? `https://r${state.shortId}.abc-tunnel.us` : null;
if (publicUrl && await probeUrlAlive(publicUrl)) return; if (publicUrl && await probeUrlAlive(publicUrl)) return;
} }