Files
openclaw/src/gateway/gateway-config-prompts.shared.ts

94 lines
2.7 KiB
TypeScript

import type { OpenClawConfig } from "../config/config.js";
import { getTailnetHostname } from "../infra/tailscale.js";
import { isIpv6Address, parseCanonicalIpAddress } from "../shared/net/ip.js";
export const TAILSCALE_EXPOSURE_OPTIONS = [
{ value: "off", label: "Off", hint: "No Tailscale exposure" },
{
value: "serve",
label: "Serve",
hint: "Private HTTPS for your tailnet (devices on Tailscale)",
},
{
value: "funnel",
label: "Funnel",
hint: "Public HTTPS via Tailscale Funnel (internet)",
},
] as const;
export const TAILSCALE_MISSING_BIN_NOTE_LINES = [
"Tailscale binary not found in PATH or /Applications.",
"Ensure Tailscale is installed from:",
" https://tailscale.com/download/mac",
"",
"You can continue setup, but serve/funnel will fail at runtime.",
] as const;
export const TAILSCALE_DOCS_LINES = [
"Docs:",
"https://docs.openclaw.ai/gateway/tailscale",
"https://docs.openclaw.ai/web",
] as const;
function normalizeTailnetHostForUrl(rawHost: string): string | null {
const trimmed = rawHost.trim().replace(/\.$/, "");
if (!trimmed) {
return null;
}
const parsed = parseCanonicalIpAddress(trimmed);
if (parsed && isIpv6Address(parsed)) {
return `[${parsed.toString().toLowerCase()}]`;
}
return trimmed;
}
export function buildTailnetHttpsOrigin(rawHost: string): string | null {
const normalizedHost = normalizeTailnetHostForUrl(rawHost);
if (!normalizedHost) {
return null;
}
try {
return new URL(`https://${normalizedHost}`).origin;
} catch {
return null;
}
}
export function appendAllowedOrigin(existing: string[] | undefined, origin: string): string[] {
const current = existing ?? [];
const normalized = origin.toLowerCase();
if (current.some((entry) => entry.toLowerCase() === normalized)) {
return current;
}
return [...current, origin];
}
export async function maybeAddTailnetOriginToControlUiAllowedOrigins(params: {
config: OpenClawConfig;
tailscaleMode: string;
tailscaleBin?: string | null;
}): Promise<OpenClawConfig> {
if (params.tailscaleMode !== "serve" && params.tailscaleMode !== "funnel") {
return params.config;
}
const tsOrigin = await getTailnetHostname(undefined, params.tailscaleBin ?? undefined)
.then((host) => buildTailnetHttpsOrigin(host))
.catch(() => null);
if (!tsOrigin) {
return params.config;
}
const existing = params.config.gateway?.controlUi?.allowedOrigins ?? [];
const updatedOrigins = appendAllowedOrigin(existing, tsOrigin);
return {
...params.config,
gateway: {
...params.config.gateway,
controlUi: {
...params.config.gateway?.controlUi,
allowedOrigins: updatedOrigins,
},
},
};
}